diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index b13b82b..8701631 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -1,50 +1,47 @@ name: Tests -on: [push, pull_request] +on: + - push + - pull_request jobs: - test: - name: PHP ${{ matrix.php }} - Laravel ${{ matrix.laravel }} - runs-on: ubuntu-latest - strategy: - fail-fast: true - matrix: - php: [ 8.2, 8.1, 8.0 ] - laravel: [ 10.*, 9.*, 8.* ] - include: - - laravel: 10.* - testbench: 8.* - - laravel: 9.* - testbench: 7.* - - laravel: 8.* - testbench: 6.* - exclude: - - laravel: 10.* - php: 8.0 - - laravel: 10.* - php: 7.4 - - laravel: 9.* - php: 7.4 - - steps: - - name: Checkout code - uses: actions/checkout@v2 - - - name: Set correct PHP version - uses: shivammathur/setup-php@v2 - with: - php-version: ${{ matrix.php }} - coverage: pcov - - - name: Install dependencies - run: | - composer require "laravel/framework:${{ matrix.laravel }}" "orchestra/testbench:${{ matrix.testbench }}" --no-interaction --no-update - composer update --prefer-stable --prefer-dist --no-interaction --no-suggest - - - name: Execute tests - run: vendor/bin/phpunit - - - name: Upload coverage to Scrutinizer - run: | - wget https://scrutinizer-ci.com/ocular.phar - php ocular.phar code-coverage:upload --format=php-clover coverage.clover + test: + name: PHP ${{ matrix.php }} - Laravel ${{ matrix.laravel }} + + runs-on: ubuntu-latest + + strategy: + fail-fast: true + matrix: + php: [8.3, 8.2, 8.1] + laravel: ['8.*', '9.*', '10.*', '11.*'] + include: + - laravel: 10.* + testbench: 8.* + - laravel: 9.* + testbench: 7.* + - laravel: 8.* + testbench: 6.* + - laravel: 11.* + testbench: 9.* + exclude: + - laravel: 11.* + php: 8.1 + + steps: + - name: Checkout code + uses: actions/checkout@v2 + + - name: Set correct PHP version + uses: shivammathur/setup-php@v2 + with: + php-version: ${{ matrix.php }} + coverage: pcov + + - name: Install dependencies + run: | + composer require "laravel/framework:${{ matrix.laravel }}" "orchestra/testbench:${{ matrix.testbench }}" --no-interaction --no-update + composer update --prefer-stable --prefer-dist --no-interaction --no-suggest + + - name: Execute tests + run: vendor/bin/phpunit diff --git a/.gitignore b/.gitignore index 3f0a634..74b638a 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,5 @@ /vendor build +.phpunit.result.cache composer.phar composer.lock diff --git a/composer.json b/composer.json index 9c03776..6e60f0e 100644 --- a/composer.json +++ b/composer.json @@ -12,17 +12,20 @@ } ], "require": { - "php": "^8.0", + "php": "^8.1", "guzzlehttp/guzzle": "^7.0.1", - "illuminate/notifications": "^8.0 || ^9.0 || ^10.0", - "illuminate/support": "^8.0 || ^9.0 || ^10.0" + "illuminate/notifications": "^8.0 || ^9.0 || ^10.0 || ^11.0", + "illuminate/support": "^8.0 || ^9.0 || ^10.0 || ^11.0" }, "require-dev": { "mockery/mockery": "^1.3.1", - "phpunit/phpunit": "^9.3", - "orchestra/testbench": "^8.0", + "phpunit/phpunit": "^9.3 || ^10.5", + "orchestra/testbench": "^8.0 || ^9.0", "dms/phpunit-arraysubset-asserts": ">=0.1.0" }, + "suggest": { + "ext-exif": "Required for image attachment support" + }, "autoload": { "psr-4": { "NotificationChannels\\Pushover\\": "src" diff --git a/src/Exceptions/CouldNotSendNotification.php b/src/Exceptions/CouldNotSendNotification.php index 61e0f50..8fbca37 100644 --- a/src/Exceptions/CouldNotSendNotification.php +++ b/src/Exceptions/CouldNotSendNotification.php @@ -3,15 +3,16 @@ namespace NotificationChannels\Pushover\Exceptions; use Exception; +use Illuminate\Notifications\AnonymousNotifiable; use Psr\Http\Message\ResponseInterface; class CouldNotSendNotification extends Exception { - public static function serviceRespondedWithAnError(ResponseInterface $response, $notifiable) + public static function serviceRespondedWithAnError(ResponseInterface $response, $notifiable): static { $statusCode = $response->getStatusCode(); - $result = json_decode($response->getBody()); + $result = json_decode($response->getBody()->getContents()); $exceptionMessage = sprintf( "Pushover responded with an error (%s) for notifiable '%s' with id '%s'", @@ -29,4 +30,19 @@ public static function serviceRespondedWithAnError(ResponseInterface $response, return new static($exceptionMessage, $statusCode); } + + public static function pushoverKeyHasWrongLength($notifiable): static + { + if ($notifiable instanceof AnonymousNotifiable) { + return new static('Pushover key has wrong length. It needs to be 30 characters long.'); + } + + $exceptionMessage = sprintf( + "Pushover key has wrong length for notifiable '%s' with id '%s'. It needs to be 30 characters long.", + get_class($notifiable), + $notifiable->getKey() + ); + + return new static($exceptionMessage); + } } diff --git a/src/Exceptions/ServiceCommunicationError.php b/src/Exceptions/ServiceCommunicationError.php index 0e33c22..59ec424 100644 --- a/src/Exceptions/ServiceCommunicationError.php +++ b/src/Exceptions/ServiceCommunicationError.php @@ -6,7 +6,7 @@ class ServiceCommunicationError extends Exception { - public static function communicationFailed(Exception $exception) + public static function communicationFailed(Exception $exception): static { return new static("The communication with Pushover failed because `{$exception->getCode()} - {$exception->getMessage()}`"); } diff --git a/src/Pushover.php b/src/Pushover.php index 20519b0..b82ecef 100644 --- a/src/Pushover.php +++ b/src/Pushover.php @@ -4,9 +4,11 @@ use Exception; use GuzzleHttp\Client as HttpClient; +use GuzzleHttp\Exception\GuzzleException; use GuzzleHttp\Exception\RequestException; use NotificationChannels\Pushover\Exceptions\CouldNotSendNotification; use NotificationChannels\Pushover\Exceptions\ServiceCommunicationError; +use Psr\Http\Message\ResponseInterface; class Pushover { @@ -22,27 +24,27 @@ class Pushover * * @var string */ - protected $pushoverApiUrl = 'https://api.pushover.net/1/messages.json'; + protected string $pushoverApiUrl = 'https://api.pushover.net/1/messages.json'; /** * The HTTP client instance. * - * @var \GuzzleHttp\Client + * @var HttpClient */ - protected $http; + protected HttpClient $http; /** * Pushover App Token. * * @var string */ - protected $token; + protected string $token; /** * @param HttpClient $http * @param string $token */ - public function __construct(HttpClient $http, $token) + public function __construct(HttpClient $http, string $token) { $this->http = $http; @@ -55,11 +57,14 @@ public function __construct(HttpClient $http, $token) * @link https://pushover.net/api * * @param array $params - * @return \Psr\Http\Message\ResponseInterface + * @param mixed $notifiable + * @return ResponseInterface * * @throws CouldNotSendNotification + * @throws ServiceCommunicationError + * @throws GuzzleException */ - public function send($params) + public function send(array $params, mixed $notifiable): ResponseInterface { try { $multipart = []; @@ -67,7 +72,7 @@ public function send($params) foreach ($this->paramsWithToken($params) as $name => $contents) { if ($name !== 'image') { $multipart[] = [ - 'name' => $name, + 'name' => $name, 'contents' => $contents, ]; } else { @@ -79,6 +84,7 @@ public function send($params) } } + //dd($multipart); return $this->http->post( $this->pushoverApiUrl, [ @@ -103,7 +109,7 @@ public function send($params) * @param array $params * @return array */ - protected function paramsWithToken($params) + protected function paramsWithToken(array $params): array { return array_merge([ 'token' => $this->token, @@ -116,11 +122,17 @@ protected function paramsWithToken($params) * If there is any error (problem with reading the file, file size exceeds the limit, the file is not an image), * silently returns null and sends the message without image attachment. * - * @param $file + * @param $file * @return array|null + * + * @throws GuzzleException */ private function getImageData($file): ?array { + if (empty($file)) { + return null; + } + try { // check if $file is not too big if (is_file($file) && is_readable($file)) { @@ -161,10 +173,10 @@ private function getImageData($file): ?array return [ // name of the field holding the image must be 'attachment' (https://pushover.net/api#attachments) - 'name' => 'attachment', + 'name' => 'attachment', 'contents' => $contents, 'filename' => basename($file), - 'headers' => [ + 'headers' => [ 'Content-Type' => $contentType, ], ]; diff --git a/src/PushoverChannel.php b/src/PushoverChannel.php index c42f960..756e16a 100644 --- a/src/PushoverChannel.php +++ b/src/PushoverChannel.php @@ -2,23 +2,24 @@ namespace NotificationChannels\Pushover; +use GuzzleHttp\Exception\GuzzleException; use Illuminate\Contracts\Events\Dispatcher; use Illuminate\Notifications\Events\NotificationFailed; use Illuminate\Notifications\Notification; +use NotificationChannels\Pushover\Exceptions\CouldNotSendNotification; use NotificationChannels\Pushover\Exceptions\ServiceCommunicationError; class PushoverChannel { - /** @var Pushover */ - protected $pushover; + protected Pushover $pushover; - /** @var Dispatcher */ - protected $events; + protected Dispatcher $events; /** * Create a new Pushover channel instance. * * @param Pushover $pushover + * @param Dispatcher $events */ public function __construct(Pushover $pushover, Dispatcher $events) { @@ -30,32 +31,42 @@ public function __construct(Pushover $pushover, Dispatcher $events) * Send the given notification. * * @param mixed $notifiable - * @param \Illuminate\Notifications\Notification $notification + * @param Notification $notification * - * @throws \NotificationChannels\Pushover\Exceptions\CouldNotSendNotification + * @throws CouldNotSendNotification + * @throws GuzzleException */ - public function send($notifiable, Notification $notification) + public function send(mixed $notifiable, Notification $notification): void { if (! $pushoverReceiver = $notifiable->routeNotificationFor('pushover')) { return; } if (is_string($pushoverReceiver)) { + // From https://pushover.net/api: + // "User and group identifiers are 30 characters long, ..." + if (strlen($pushoverReceiver) !== 30) { + throw CouldNotSendNotification::pushoverKeyHasWrongLength($notifiable); + } + $pushoverReceiver = PushoverReceiver::withUserKey($pushoverReceiver); } $message = $notification->toPushover($notifiable); try { - $this->pushover->send(array_merge($message->toArray(), $pushoverReceiver->toArray())); + $this->pushover->send( + array_merge($message->toArray(), $pushoverReceiver->toArray()), + $notifiable + ); } catch (ServiceCommunicationError $serviceCommunicationError) { $this->fireFailedEvent($notifiable, $notification, $serviceCommunicationError->getMessage()); } } - protected function fireFailedEvent($notifiable, $notification, $message) + protected function fireFailedEvent($notifiable, $notification, $message): void { - $this->events->fire( + $this->events->dispatch( new NotificationFailed($notifiable, $notification, 'pushover', [$message]) ); } diff --git a/src/PushoverMessage.php b/src/PushoverMessage.php index 4ff3065..2882ac1 100644 --- a/src/PushoverMessage.php +++ b/src/PushoverMessage.php @@ -12,103 +12,101 @@ class PushoverMessage * * @var string */ - public $content; + public string $content; /** * The format of the message. * - * Either "plain", "html" or "monospace". - * - * @var string + * @var int */ - public $format = self::FORMAT_PLAIN; + public int $format = self::FORMAT_PLAIN; /** * The (optional) title of the message. * - * @var string + * @var string|null */ - public $title; + public string|null $title = null; /** * The (optional) timestamp of the message. * - * @var int + * @var int|null */ - public $timestamp; + public int|null $timestamp = null; /** * The (optional) priority of the message. * - * @var int + * @var int|null */ - public $priority; + public int|null $priority = null; /** * The (optional) timeout between retries when sending a message * with an emergency priority. The timeout is in seconds. * - * @var int + * @var int|null */ - public $retry; + public int|null $retry = null; /** * The (optional) expire time of a message with an emergency priority. * The expire time is in seconds. * - * @var int + * @var int|null */ - public $expire; + public int|null $expire = null; /** * The (optional) supplementary url of the message. * - * @var string + * @var string|null */ - public $url; + public string|null $url = null; /** * The (optional) supplementary url title of the message. * - * @var string + * @var string|null */ - public $urlTitle; + public string|null $urlTitle = null; /** * The (optional) sound of the message. * - * @var string + * @var string|null */ - public $sound; + public string|null $sound = null; /** * The (optional) image to be attached to the message. * - * @var string + * @var string|null */ - public $image; + public string|null $image = null; /** * Message formats. */ - const FORMAT_PLAIN = 0; - const FORMAT_HTML = 1; - const FORMAT_MONOSPACE = 2; + public const FORMAT_PLAIN = 0; + public const FORMAT_HTML = 1; + public const FORMAT_MONOSPACE = 2; /** * Message priorities. */ - const LOWEST_PRIORITY = -2; - const LOW_PRIORITY = -1; - const NORMAL_PRIORITY = 0; - const HIGH_PRIORITY = 1; - const EMERGENCY_PRIORITY = 2; + public const LOWEST_PRIORITY = -2; + public const LOW_PRIORITY = -1; + public const NORMAL_PRIORITY = 0; + public const HIGH_PRIORITY = 1; + public const EMERGENCY_PRIORITY = 2; /** * @param string $content * @return static */ - public static function create($content = '') + public static function create(string $content = ''): static { return new static($content); } @@ -116,7 +114,7 @@ public static function create($content = '') /** * @param string $content */ - public function __construct($content = '') + public function __construct(string $content = '') { $this->content = $content; } @@ -127,7 +125,7 @@ public function __construct($content = '') * @param string $content * @return $this */ - public function content($content) + public function content(string $content): static { $this->content = $content; @@ -139,7 +137,7 @@ public function content($content) * * @return $this */ - public function plain() + public function plain(): static { $this->format = static::FORMAT_PLAIN; @@ -151,7 +149,7 @@ public function plain() * * @return $this */ - public function html() + public function html(): static { $this->format = static::FORMAT_HTML; @@ -163,7 +161,7 @@ public function html() * * @return $this */ - public function monospace() + public function monospace(): static { $this->format = self::FORMAT_MONOSPACE; @@ -176,7 +174,7 @@ public function monospace() * @param string $title * @return $this */ - public function title($title) + public function title(string $title): static { $this->title = $title; @@ -186,13 +184,13 @@ public function title($title) /** * Set the time of the Pushover message. * - * @param Carbon|int $time + * @param int|Carbon $time * @return $this */ - public function time($time) + public function time(int|Carbon $time): static { if ($time instanceof Carbon) { - $time = $time->timestamp; + $time = (int) $time->timestamp; } $this->timestamp = $time; @@ -207,7 +205,7 @@ public function time($time) * @param string $title * @return $this */ - public function url($url, $title = null) + public function url(string $url, string $title = ''): static { $this->url = $url; $this->urlTitle = $title; @@ -221,7 +219,7 @@ public function url($url, $title = null) * @param string $sound * @return $this */ - public function sound($sound) + public function sound(string $sound): static { $this->sound = $sound; @@ -234,7 +232,7 @@ public function sound($sound) * @param string $image * @return $this */ - public function image($image) + public function image(string $image): static { $this->image = $image; @@ -246,11 +244,13 @@ public function image($image) * Retry and expire are mandatory when setting the priority to emergency. * * @param int $priority - * @param int $retryTimeout - * @param int $expireAfter + * @param int|null $retryTimeout + * @param int|null $expireAfter * @return $this + * + * @throws EmergencyNotificationRequiresRetryAndExpire */ - public function priority($priority, $retryTimeout = null, $expireAfter = null) + public function priority(int $priority, int|null $retryTimeout = null, int|null $expireAfter = null): static { $this->noEmergencyWithoutRetryOrExpire($priority, $retryTimeout, $expireAfter); @@ -265,8 +265,10 @@ public function priority($priority, $retryTimeout = null, $expireAfter = null) * Set the priority of the Pushover message to the lowest priority. * * @return $this + * + * @throws EmergencyNotificationRequiresRetryAndExpire */ - public function lowestPriority() + public function lowestPriority(): static { return $this->priority(self::LOWEST_PRIORITY); } @@ -275,8 +277,10 @@ public function lowestPriority() * Set the priority of the Pushover message to low. * * @return $this + * + * @throws EmergencyNotificationRequiresRetryAndExpire */ - public function lowPriority() + public function lowPriority(): static { return $this->priority(self::LOW_PRIORITY); } @@ -285,8 +289,10 @@ public function lowPriority() * Set the priority of the Pushover message to normal. * * @return $this + * + * @throws EmergencyNotificationRequiresRetryAndExpire */ - public function normalPriority() + public function normalPriority(): static { return $this->priority(self::NORMAL_PRIORITY); } @@ -295,8 +301,10 @@ public function normalPriority() * Set the priority of the Pushover message to high. * * @return $this + * + * @throws EmergencyNotificationRequiresRetryAndExpire */ - public function highPriority() + public function highPriority(): static { return $this->priority(self::HIGH_PRIORITY); } @@ -308,8 +316,10 @@ public function highPriority() * @param int $retryTimeout * @param int $expireAfter * @return $this + * + * @throws EmergencyNotificationRequiresRetryAndExpire */ - public function emergencyPriority($retryTimeout, $expireAfter) + public function emergencyPriority(int $retryTimeout, int $expireAfter): static { return $this->priority(self::EMERGENCY_PRIORITY, $retryTimeout, $expireAfter); } @@ -319,7 +329,7 @@ public function emergencyPriority($retryTimeout, $expireAfter) * * @return array */ - public function toArray() + public function toArray(): array { return [ 'message' => $this->content, @@ -341,14 +351,14 @@ public function toArray() * Ensure an emergency message has an retry and expiry time. * * @param int $priority - * @param int $retry - * @param int $expire + * @param int|null $retry + * @param int|null $expire * * @throws EmergencyNotificationRequiresRetryAndExpire */ - protected function noEmergencyWithoutRetryOrExpire($priority, $retry, $expire) + protected function noEmergencyWithoutRetryOrExpire(int $priority, int|null $retry, int|null $expire): void { - if ($priority == self::EMERGENCY_PRIORITY && (! isset($retry) || ! isset($expire))) { + if ($priority === self::EMERGENCY_PRIORITY && ($retry === null || $expire === null)) { throw new EmergencyNotificationRequiresRetryAndExpire(); } } diff --git a/src/PushoverReceiver.php b/src/PushoverReceiver.php index f228a41..ff8232c 100644 --- a/src/PushoverReceiver.php +++ b/src/PushoverReceiver.php @@ -4,16 +4,16 @@ class PushoverReceiver { - protected $key; - protected $token; - protected $devices = []; + protected string $key; + protected string|null $token = null; + protected array $devices = []; /** * PushoverReceiver constructor. * - * @param $key User or group key. + * @param string $key User or group key. */ - protected function __construct($key) + protected function __construct(string $key) { $this->key = $key; } @@ -21,10 +21,10 @@ protected function __construct($key) /** * Create new Pushover receiver with an user key. * - * @param $userKey Pushover user key. + * @param string $userKey Pushover user key. * @return PushoverReceiver */ - public static function withUserKey($userKey) + public static function withUserKey(string $userKey): PushoverReceiver { return new static($userKey); } @@ -32,10 +32,10 @@ public static function withUserKey($userKey) /** * Create new Pushover receiver with a group key. * - * @param $groupKey Pushover group key. + * @param string $groupKey Pushover group key. * @return PushoverReceiver */ - public static function withGroupKey($groupKey) + public static function withGroupKey(string $groupKey): PushoverReceiver { // This has exactly the same behaviour as an user key, so we // will use the same factory method as for the user key. @@ -48,13 +48,14 @@ public static function withGroupKey($groupKey) * @param array|string $device * @return PushoverReceiver */ - public function toDevice($device) + public function toDevice(array|string $device): static { if (is_array($device)) { $this->devices = array_merge($device, $this->devices); return $this; } + $this->devices[] = $device; return $this; @@ -63,10 +64,10 @@ public function toDevice($device) /** * Set the application token. * - * @param $token + * @param $token * @return PushoverReceiver */ - public function withApplicationToken($token) + public function withApplicationToken($token): static { $this->token = $token; @@ -78,7 +79,7 @@ public function withApplicationToken($token) * * @return array */ - public function toArray() + public function toArray(): array { return array_merge([ 'user' => $this->key, diff --git a/src/PushoverServiceProvider.php b/src/PushoverServiceProvider.php index 4757648..1d83106 100644 --- a/src/PushoverServiceProvider.php +++ b/src/PushoverServiceProvider.php @@ -10,7 +10,7 @@ class PushoverServiceProvider extends ServiceProvider /** * Bootstrap the application services. */ - public function boot() + public function boot(): void { $this->app->when(PushoverChannel::class) ->needs(Pushover::class) @@ -18,11 +18,4 @@ public function boot() return new Pushover(new HttpClient(), config('services.pushover.token')); }); } - - /** - * Register the application services. - */ - public function register() - { - } } diff --git a/tests/IntegrationTest.php b/tests/IntegrationTest.php index 9416812..0ee2f31 100644 --- a/tests/IntegrationTest.php +++ b/tests/IntegrationTest.php @@ -34,7 +34,7 @@ public function setUp(): void } /** @test */ - public function it_can_send_a_pushover_notification_with_a_global_token() + public function it_can_send_a_pushover_notification_with_a_global_token(): void { $message = (new PushoverMessage('Message text')) ->title('Message title') @@ -45,19 +45,19 @@ public function it_can_send_a_pushover_notification_with_a_global_token() $this->requestWillBeSentToPushoverWith([ 'token' => 'global-application-token', - 'user' => 'pushover-key', - 'device' => 'iphone,desktop', 'message' => 'Message text', 'title' => 'Message title', - 'priority' => 2, - 'retry' => 60, - 'expire' => 600, 'timestamp' => 123456789, - 'sound' => 'boing', + 'priority' => 2, 'url' => 'http://example.com', 'url_title' => 'Example Website', + 'sound' => 'boing', + 'retry' => 60, + 'expire' => 600, 'html' => false, 'monospace' => false, + 'user' => 'pushover-key-30characters-long', + 'device' => 'iphone,desktop', ]); $pushover = new Pushover($this->guzzleClient, 'global-application-token'); @@ -70,7 +70,7 @@ public function it_can_send_a_pushover_notification_with_a_global_token() } /** @test */ - public function it_can_send_a_pushover_notification_with_an_overridden_token() + public function it_can_send_a_pushover_notification_with_an_overridden_token(): void { $message = (new PushoverMessage('Message text')) ->html() @@ -82,19 +82,19 @@ public function it_can_send_a_pushover_notification_with_an_overridden_token() $this->requestWillBeSentToPushoverWith([ 'token' => 'overridden-application-token', - 'user' => 'pushover-key', - 'device' => 'iphone,desktop', 'message' => 'Message text', 'title' => 'Message title', - 'priority' => 2, - 'retry' => 60, - 'expire' => 600, 'timestamp' => 123456789, - 'sound' => 'boing', + 'priority' => 2, 'url' => 'http://example.com', 'url_title' => 'Example Website', + 'sound' => 'boing', + 'retry' => 60, + 'expire' => 600, 'html' => true, 'monospace' => false, + 'user' => 'pushover-key-30characters-long', + 'device' => 'iphone,desktop', ]); $pushover = new Pushover($this->guzzleClient, 'global-application-token'); @@ -106,19 +106,25 @@ public function it_can_send_a_pushover_notification_with_an_overridden_token() $channel->send(new NotifiableWithPushoverReceiverWithToken(), $this->notification); } - protected function requestWillBeSentToPushoverWith($params) + protected function requestWillBeSentToPushoverWith($params): void { + $multipartData = array_map( + fn ($key, $value) => ['name' => $key, 'contents' => $value], + array_keys($params), + array_values($params) + ); + $this->guzzleClient->shouldReceive('post') ->with('https://api.pushover.net/1/messages.json', [ - 'form_params' => $params, + 'multipart' => $multipartData, ]) ->once(); } - protected function ignoreEvents() + protected function ignoreEvents(): void { $dispatcher = Mockery::mock('Illuminate\Contracts\Events\Dispatcher'); - $dispatcher->shouldReceive('fire'); + $dispatcher->shouldReceive('dispatch'); app()->instance('events', $dispatcher); } } diff --git a/tests/Notifiable.php b/tests/Notifiable.php index f4e19dc..c5ce36a 100644 --- a/tests/Notifiable.php +++ b/tests/Notifiable.php @@ -6,6 +6,11 @@ class Notifiable { public function routeNotificationFor($channel) { - return 'pushover-key'; + return 'pushover-key-30characters-long'; + } + + public function getKey() + { + return '1'; } } diff --git a/tests/NotifiableWithPushoverReceiver.php b/tests/NotifiableWithPushoverReceiver.php index 072f2e7..d3beff6 100644 --- a/tests/NotifiableWithPushoverReceiver.php +++ b/tests/NotifiableWithPushoverReceiver.php @@ -8,7 +8,7 @@ class NotifiableWithPushoverReceiver extends Notifiable { public function routeNotificationFor($channel) { - return PushoverReceiver::withUserKey('pushover-key') + return PushoverReceiver::withUserKey('pushover-key-30characters-long') ->toDevice('iphone') ->toDevice('desktop'); } diff --git a/tests/NotifiableWithPushoverReceiverWithToken.php b/tests/NotifiableWithPushoverReceiverWithToken.php index bcff167..f956f35 100644 --- a/tests/NotifiableWithPushoverReceiverWithToken.php +++ b/tests/NotifiableWithPushoverReceiverWithToken.php @@ -8,7 +8,7 @@ class NotifiableWithPushoverReceiverWithToken extends Notifiable { public function routeNotificationFor($channel) { - return PushoverReceiver::withUserKey('pushover-key') + return PushoverReceiver::withUserKey('pushover-key-30characters-long') ->withApplicationToken('overridden-application-token') ->toDevice('iphone') ->toDevice('desktop'); diff --git a/tests/PushoverChannelTest.php b/tests/PushoverChannelTest.php index 6f7397f..41e8374 100644 --- a/tests/PushoverChannelTest.php +++ b/tests/PushoverChannelTest.php @@ -43,56 +43,60 @@ public function setUp(): void } /** @test */ - public function it_can_send_a_message_to_pushover() + public function it_can_send_a_message_to_pushover(): void { $notifiable = new Notifiable; $this->notification->shouldReceive('toPushover') ->with($notifiable) ->andReturn($this->message); + $this->pushover->shouldReceive('send') ->with(Mockery::subset([ - 'user' => 'pushover-key', + 'user' => 'pushover-key-30characters-long', 'device' => '', - ])) + ]), $notifiable) ->once(); $this->channel->send($notifiable, $this->notification); } /** @test */ - public function it_can_send_a_message_to_pushover_using_a_pushover_receiver() + public function it_can_send_a_message_to_pushover_using_a_pushover_receiver(): void { $notifiable = new NotifiableWithPushoverReceiver; $this->notification->shouldReceive('toPushover') ->with($notifiable) ->andReturn($this->message); + $this->pushover->shouldReceive('send') ->with(Mockery::subset([ - 'user' => 'pushover-key', + 'user' => 'pushover-key-30characters-long', 'device' => 'iphone,desktop', - ])) + ]), $notifiable) ->once(); $this->channel->send($notifiable, $this->notification); } /** @test */ - public function it_fires_a_notification_failed_event_when_the_communication_with_pushover_failed() + public function it_fires_a_notification_failed_event_when_the_communication_with_pushover_failed(): void { $this->notification->shouldReceive('toPushover')->andReturn($this->message); $this->pushover->shouldReceive('send')->andThrow( ServiceCommunicationError::communicationFailed(new Exception()) ); - $this->events->shouldReceive('fire')->with(Mockery::type(NotificationFailed::class)); + $this->events->shouldReceive('dispatch')->with(Mockery::type(NotificationFailed::class)); $this->channel->send(new Notifiable, $this->notification); + + $this->expectNotToPerformAssertions(); } /** @test */ - public function it_does_not_send_a_message_when_notifiable_does_not_have_route_notificaton_for_pushover() + public function it_does_not_send_a_message_when_notifiable_does_not_have_route_notificaton_for_pushover(): void { $this->notification->shouldReceive('toPushover')->never(); diff --git a/tests/PushoverMessageTest.php b/tests/PushoverMessageTest.php index a2c371e..1440806 100644 --- a/tests/PushoverMessageTest.php +++ b/tests/PushoverMessageTest.php @@ -19,7 +19,7 @@ public function setUp(): void } /** @test */ - public function it_can_accept_a_message_when_constructing_a_message() + public function it_can_accept_a_message_when_constructing_a_message(): void { $message = new PushoverMessage('message text'); @@ -27,7 +27,7 @@ public function it_can_accept_a_message_when_constructing_a_message() } /** @test */ - public function it_can_create_a_message() + public function it_can_create_a_message(): void { $message = PushoverMessage::create(); @@ -35,7 +35,7 @@ public function it_can_create_a_message() } /** @test */ - public function it_can_accept_a_message_when_creating_a_message() + public function it_can_accept_a_message_when_creating_a_message(): void { $message = PushoverMessage::create('message text'); @@ -43,7 +43,7 @@ public function it_can_accept_a_message_when_creating_a_message() } /** @test */ - public function it_can_set_content() + public function it_can_set_content(): void { $this->message->content('message text'); @@ -51,7 +51,7 @@ public function it_can_set_content() } /** @test */ - public function it_can_set_the_message_format_to_plain() + public function it_can_set_the_message_format_to_plain(): void { $this->message->plain(); @@ -59,7 +59,7 @@ public function it_can_set_the_message_format_to_plain() } /** @test */ - public function it_can_set_the_message_format_to_html() + public function it_can_set_the_message_format_to_html(): void { $this->message->html(); @@ -67,7 +67,7 @@ public function it_can_set_the_message_format_to_html() } /** @test */ - public function it_can_set_the_message_format_to_monospace() + public function it_can_set_the_message_format_to_monospace(): void { $this->message->monospace(); @@ -75,7 +75,7 @@ public function it_can_set_the_message_format_to_monospace() } /** @test */ - public function it_can_set_a_title() + public function it_can_set_a_title(): void { $this->message->title('message title'); @@ -83,7 +83,7 @@ public function it_can_set_a_title() } /** @test */ - public function it_can_set_a_time() + public function it_can_set_a_time(): void { $this->message->time(123456789); @@ -91,7 +91,7 @@ public function it_can_set_a_time() } /** @test */ - public function it_can_set_a_time_from_a_carbon_object() + public function it_can_set_a_time_from_a_carbon_object(): void { $carbon = Carbon::now(); @@ -101,7 +101,7 @@ public function it_can_set_a_time_from_a_carbon_object() } /** @test */ - public function it_can_set_an_url() + public function it_can_set_an_url(): void { $this->message->url('http://example.com'); @@ -109,7 +109,7 @@ public function it_can_set_an_url() } /** @test */ - public function it_can_set_an_url_with_a_title() + public function it_can_set_an_url_with_a_title(): void { $this->message->url('http://example.com', 'Go to example website'); @@ -118,7 +118,7 @@ public function it_can_set_an_url_with_a_title() } /** @test */ - public function it_can_set_a_sound() + public function it_can_set_a_sound(): void { $this->message->sound('boing'); @@ -126,7 +126,7 @@ public function it_can_set_a_sound() } /** @test */ - public function it_can_set_a_priority() + public function it_can_set_a_priority(): void { $this->message->priority(PushoverMessage::NORMAL_PRIORITY); @@ -134,7 +134,7 @@ public function it_can_set_a_priority() } /** @test */ - public function it_can_set_a_priority_with_retry_and_expire() + public function it_can_set_a_priority_with_retry_and_expire(): void { $this->message->priority(PushoverMessage::EMERGENCY_PRIORITY, 60, 600); @@ -144,7 +144,7 @@ public function it_can_set_a_priority_with_retry_and_expire() } /** @test */ - public function it_cannot_set_priority_to_emergency_when_not_providing_a_retry_and_expiry_time() + public function it_cannot_set_priority_to_emergency_when_not_providing_a_retry_and_expiry_time(): void { $this->expectException(EmergencyNotificationRequiresRetryAndExpire::class); @@ -152,7 +152,7 @@ public function it_cannot_set_priority_to_emergency_when_not_providing_a_retry_a } /** @test */ - public function it_can_set_the_priority_to_the_lowest() + public function it_can_set_the_priority_to_the_lowest(): void { $this->message->lowestPriority(); @@ -160,7 +160,7 @@ public function it_can_set_the_priority_to_the_lowest() } /** @test */ - public function it_can_set_the_priority_to_low() + public function it_can_set_the_priority_to_low(): void { $this->message->lowPriority(); @@ -168,7 +168,7 @@ public function it_can_set_the_priority_to_low() } /** @test */ - public function it_can_set_the_priority_to_normal() + public function it_can_set_the_priority_to_normal(): void { $this->message->normalPriority(); @@ -176,7 +176,7 @@ public function it_can_set_the_priority_to_normal() } /** @test */ - public function it_can_set_the_priority_to_high() + public function it_can_set_the_priority_to_high(): void { $this->message->highPriority(); @@ -184,7 +184,7 @@ public function it_can_set_the_priority_to_high() } /** @test */ - public function it_can_set_the_priority_to_emergency() + public function it_can_set_the_priority_to_emergency(): void { $this->message->emergencyPriority(60, 600); diff --git a/tests/PushoverServiceProviderTest.php b/tests/PushoverServiceProviderTest.php index b692201..000676f 100644 --- a/tests/PushoverServiceProviderTest.php +++ b/tests/PushoverServiceProviderTest.php @@ -3,6 +3,7 @@ namespace NotificationChannels\Pushover\Test; use Illuminate\Contracts\Foundation\Application; +use Illuminate\Support\Facades\Config; use Mockery; use NotificationChannels\Pushover\Pushover; use NotificationChannels\Pushover\PushoverChannel; @@ -28,8 +29,10 @@ public function setUp(): void } /** @test */ - public function it_gives_an_instantiated_pushover_object_when_the_channel_asks_for_it() + public function it_gives_an_instantiated_pushover_object_when_the_channel_asks_for_it(): void { + Config::shouldReceive('get')->with('services.pushover.token', null)->once()->andReturn('test-token'); + $this->app->shouldReceive('when')->with(PushoverChannel::class)->once()->andReturn($this->app); $this->app->shouldReceive('needs')->with(Pushover::class)->once()->andReturn($this->app); $this->app->shouldReceive('give')->with(Mockery::on(function ($pushover) { diff --git a/tests/PushoverTest.php b/tests/PushoverTest.php index d51b8b9..1e546f4 100644 --- a/tests/PushoverTest.php +++ b/tests/PushoverTest.php @@ -24,85 +24,91 @@ public function setUp(): void parent::setUp(); $this->guzzleClient = Mockery::mock(HttpClient::class); - $this->pushover = new Pushover($this->guzzleClient, 'application-token'); + $this->pushover = new Pushover($this->guzzleClient, 'new-application-token-12345678'); } /** @test */ - public function it_can_send_a_request_to_pushover() + public function it_can_send_a_request_to_pushover(): void { $this->guzzleClient->shouldReceive('post') ->with('https://api.pushover.net/1/messages.json', [ - 'form_params' => [ - 'token' => 'application-token', + 'multipart' => [ + ['name' => 'token', 'contents' => 'new-application-token-12345678'], ], ]); - $this->pushover->send([]); + $this->pushover->send([], new Notifiable()); + + $this->expectNotToPerformAssertions(); } /** @test */ - public function it_can_send_a_request_with_an_overridden_token() + public function it_can_send_a_request_with_an_overridden_token(): void { $this->guzzleClient->shouldReceive('post') ->with('https://api.pushover.net/1/messages.json', [ - 'form_params' => [ - 'token' => 'dynamic-application-token', + 'multipart' => [ + ['name' => 'token', 'contents' => 'dynamic-application-token-1234'], ], ]); - $this->pushover->send(['token' => 'dynamic-application-token']); + $this->pushover->send(['token' => 'dynamic-application-token-1234'], new Notifiable()); + + $this->expectNotToPerformAssertions(); } /** @test */ - public function it_can_accept_parameters_for_a_send_request() + public function it_can_accept_parameters_for_a_send_request(): void { $this->guzzleClient->shouldReceive('post') ->with('https://api.pushover.net/1/messages.json', [ - 'form_params' => [ - 'token' => 'application-token', - 'content' => 'content of message', + 'multipart' => [ + ['name' => 'token', 'contents' => 'new-application-token-12345678'], + ['name' => 'content', 'contents' => 'content of message'], ], ]); $this->pushover->send([ 'content' => 'content of message', - ]); + ], new Notifiable()); + + $this->expectNotToPerformAssertions(); } /** @test */ - public function it_throws_an_exception_when_pushover_returns_an_error_with_invalid_json() + public function it_throws_an_exception_when_pushover_returns_an_error_with_invalid_json(): void { $this->expectException(CouldNotSendNotification::class); - $this->expectExceptionMessage('Pushover responded with an error (400).'); + $this->expectExceptionMessage('Pushover responded with an error (400)'); $guzzleRequest = Mockery::mock(\Psr\Http\Message\RequestInterface::class); $guzzleResponse = Mockery::mock(\Psr\Http\Message\ResponseInterface::class); $guzzleResponse->shouldReceive('getStatusCode')->andReturn(400); - $guzzleResponse->shouldReceive('getBody')->andReturn(''); + $guzzleResponse->shouldReceive('getBody->getContents'); $this->guzzleClient->shouldReceive('post')->andThrow(new RequestException('message', $guzzleRequest, $guzzleResponse)); - $this->pushover->send([]); + $this->pushover->send([], new Notifiable()); } /** @test */ - public function it_throws_an_exception_when_pushover_returns_an_error_with_valid_json() + public function it_throws_an_exception_when_pushover_returns_an_error_with_valid_json(): void { $this->expectException(CouldNotSendNotification::class); - $this->expectExceptionMessage('Pushover responded with an error (400): error_message_1, error_message_2'); + $this->expectExceptionMessage("Pushover responded with an error (400) for notifiable 'NotificationChannels\Pushover\Test\Notifiable' with id '1': error_message_1, error_message_2"); $guzzleRequest = Mockery::mock(\Psr\Http\Message\RequestInterface::class); $guzzleResponse = Mockery::mock(\Psr\Http\Message\ResponseInterface::class); $guzzleResponse->shouldReceive('getStatusCode')->andReturn(400); - $guzzleResponse->shouldReceive('getBody')->andReturn('{"errors": ["error_message_1", "error_message_2"]}'); + $guzzleResponse->shouldReceive('getBody->getContents')->andReturn('{"errors": ["error_message_1", "error_message_2"]}'); $this->guzzleClient->shouldReceive('post')->andThrow(new RequestException('message', $guzzleRequest, $guzzleResponse)); - $this->pushover->send([]); + $this->pushover->send([], new Notifiable()); } /** @test */ - public function it_throws_an_exception_when_pushover_returns_nothing() + public function it_throws_an_exception_when_pushover_returns_nothing(): void { $this->expectException(ServiceCommunicationError::class); $this->expectExceptionMessage('The communication with Pushover failed because'); @@ -111,17 +117,17 @@ public function it_throws_an_exception_when_pushover_returns_nothing() $this->guzzleClient->shouldReceive('post')->andThrow(new RequestException('message', $guzzleRequest, null)); - $this->pushover->send([]); + $this->pushover->send([], new Notifiable()); } /** @test */ - public function it_throws_an_exception_when_an_unknown_communication_error_occurred() + public function it_throws_an_exception_when_an_unknown_communication_error_occurred(): void { $this->expectException(ServiceCommunicationError::class); $this->expectExceptionMessage('The communication with Pushover failed'); $this->guzzleClient->shouldReceive('post')->andThrow(new Exception); - $this->pushover->send([]); + $this->pushover->send([], new Notifiable()); } }