vd/vendor/dipper/sms/src/Sms.php
2018-12-10 18:54:06 +08:00

254 lines
5.6 KiB
PHP

<?php
namespace Dipper\Sms;
use Closure;
use RuntimeException;
use Dipper\Sms\Support\Config;
use Dipper\Sms\Strategies\OrderStrategy;
use Dipper\Sms\Contracts\GatewayInterface;
use Dipper\Sms\Contracts\StrategyInterface;
use Dipper\Sms\Exceptions\InvalidArgumentException;
class Sms
{
/**
* @var \Dipper\Sms\Support\Config
*/
protected $config;
/**
* @var string
*/
protected $defaultGateway;
/**
* @var array
*/
protected $customCreators = [];
/**
* @var array
*/
protected $gateways = [];
/**
* @var \Dipper\Sms\Messenger
*/
protected $messenger;
/**
* @var array
*/
protected $strategies = [];
/**
* Constructor.
*
* @param array $config
*/
public function __construct(array $config)
{
$this->config = new Config($config);
if (!empty($config['default'])) {
$this->setDefaultGateway($config['default']);
}
}
/**
* Send a message.
*
* @param string|array $to
* @param \Dipper\Sms\Contracts\MessageInterface|array $message
* @param array $gateways
*
* @return array
*/
public function send($to, $message, array $gateways = [])
{
return $this->getMessenger()->send($to, $message, $gateways);
}
/**
* Create a gateway.
*
* @param string|null $name
*
* @return \Dipper\Sms\Contracts\GatewayInterface
*/
public function gateway($name = null)
{
$name = $name ?: $this->getDefaultGateway();
if (!isset($this->gateways[$name])) {
$this->gateways[$name] = $this->createGateway($name);
}
return $this->gateways[$name];
}
/**
* Get a strategy instance.
*
* @param string|null $strategy
*
* @return \Dipper\Sms\Contracts\StrategyInterface
*
* @throws \Dipper\Sms\Exceptions\InvalidArgumentException
*/
public function strategy($strategy = null)
{
if (is_null($strategy)) {
$strategy = $this->config->get('default.strategy', OrderStrategy::class);
}
if (!class_exists($strategy)) {
$strategy = __NAMESPACE__.'\Strategies\\'.ucfirst($strategy);
}
if (!class_exists($strategy)) {
throw new InvalidArgumentException("Unsupported strategy \"{$strategy}\"");
}
if (empty($this->strategies[$strategy]) || !($this->strategies[$strategy] instanceof StrategyInterface)) {
$this->strategies[$strategy] = new $strategy($this);
}
return $this->strategies[$strategy];
}
/**
* Register a custom driver creator Closure.
*
* @param string $name
* @param \Closure $callback
*
* @return $this
*/
public function extend($name, Closure $callback)
{
$this->customCreators[$name] = $callback;
return $this;
}
/**
* @return \Dipper\Sms\Support\Config
*/
public function getConfig()
{
return $this->config;
}
/**
* Get default gateway name.
*
* @return string
*
* @throws if no default gateway configured
*/
public function getDefaultGateway()
{
if (empty($this->defaultGateway)) {
throw new RuntimeException('No default gateway configured.');
}
return $this->defaultGateway;
}
/**
* Set default gateway name.
*
* @param string $name
*
* @return $this
*/
public function setDefaultGateway($name)
{
$this->defaultGateway = $name;
return $this;
}
/**
* @return \Dipper\Sms\Messenger
*/
public function getMessenger()
{
return $this->messenger ?: $this->messenger = new Messenger($this);
}
/**
* Create a new driver instance.
*
* @param string $name
*
* @throws \InvalidArgumentException
*
* @return GatewayInterface
*/
protected function createGateway($name)
{
if (isset($this->customCreators[$name])) {
$gateway = $this->callCustomCreator($name);
} else {
$className = $this->formatGatewayClassName($name);
$gateway = $this->makeGateway($className, $this->config->get("gateways.{$name}", []));
}
if (!($gateway instanceof GatewayInterface)) {
throw new InvalidArgumentException(sprintf('Gateway "%s" not inherited from %s.', $name, GatewayInterface::class));
}
return $gateway;
}
/**
* Make gateway instance.
*
* @param string $gateway
* @param array $config
*
* @return \Dipper\Sms\Contracts\GatewayInterface
*/
protected function makeGateway($gateway, $config)
{
if (!class_exists($gateway)) {
throw new InvalidArgumentException(sprintf('Gateway "%s" not exists.', $gateway));
}
return new $gateway($config);
}
/**
* Format gateway name.
*
* @param string $name
*
* @return string
*/
protected function formatGatewayClassName($name)
{
if (class_exists($name)) {
return $name;
}
$name = ucfirst(str_replace(['-', '_', ''], '', $name));
return __NAMESPACE__."\\Gateways\\{$name}Gateway";
}
/**
* Call a custom gateway creator.
*
* @param string $gateway
*
* @return mixed
*/
protected function callCustomCreator($gateway)
{
return call_user_func($this->customCreators[$gateway], $this->config->get($gateway, []));
}
}