From f4b1716d2d8e255a48395255e6f4c8c0f08243be Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Kr=C3=A4tzig?= Date: Fri, 7 Jan 2022 01:13:24 +0100 Subject: [PATCH 1/4] #571: Improve ConnectionException handling --- .../Exceptions/ConnectionException.php | 26 +++++++++++++++++++ src/PhpImap/Imap.php | 9 ++----- 2 files changed, 28 insertions(+), 7 deletions(-) diff --git a/src/PhpImap/Exceptions/ConnectionException.php b/src/PhpImap/Exceptions/ConnectionException.php index 54153a8f..3e45c4d7 100644 --- a/src/PhpImap/Exceptions/ConnectionException.php +++ b/src/PhpImap/Exceptions/ConnectionException.php @@ -13,4 +13,30 @@ */ class ConnectionException extends Exception { + public function __construct($message, $code = 0, Exception $previous = null) + { + parent::__construct(json_encode($message), $code, $previous); + } + + public function getErrors($select = 'first') + { + $message = $this->getMessage(); + + switch (strtolower($select)) { + case 'all': + return json_decode($message); + break; + default: + case 'first': + $message = json_decode($message); + + return $message[0]; + break; + case 'last': + $message = json_decode($message); + + return $message[\count($message) - 1]; + break; + } + } } diff --git a/src/PhpImap/Imap.php b/src/PhpImap/Imap.php index d6c9475d..e3bfc1b9 100644 --- a/src/PhpImap/Imap.php +++ b/src/PhpImap/Imap.php @@ -14,6 +14,7 @@ use const IMAP_WRITETIMEOUT; use InvalidArgumentException; use const NIL; +use PhpImap\Exceptions\ConnectionException; use const SE_FREE; use const SORTARRIVAL; use const SORTCC; @@ -707,13 +708,7 @@ public static function open( $result = @\imap_open($mailbox, $username, $password, $options, $n_retries, $params); if (!$result) { - $lastError = \imap_last_error(); - - if ((\is_string($lastError)) && ('' !== \trim($lastError))) { - throw new UnexpectedValueException('IMAP error:'.$lastError); - } - - throw new UnexpectedValueException('Could not open mailbox!', 0, self::HandleErrors(\imap_errors(), 'imap_open')); + throw new ConnectionException(\imap_errors()); } return $result; From e5b50994e7b067719d2fdae1259d5da8a3537d73 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Kr=C3=A4tzig?= Date: Fri, 7 Jan 2022 01:32:31 +0100 Subject: [PATCH 2/4] Add examples for the new ConnectionException::getErrors() function --- README.md | 2 +- examples/get_and_parse_unseen_emails.php | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index fb03bc26..0b801e8e 100644 --- a/README.md +++ b/README.md @@ -97,7 +97,7 @@ try { // PHP.net imap_search criteria: http://php.net/manual/en/function.imap-search.php $mailsIds = $mailbox->searchMailbox('ALL'); } catch(PhpImap\Exceptions\ConnectionException $ex) { - echo "IMAP connection failed: " . $ex; + echo "IMAP connection failed: " . implode(",", $ex->getErrors('all')); die(); } diff --git a/examples/get_and_parse_unseen_emails.php b/examples/get_and_parse_unseen_emails.php index 800bb5ac..12eb8bb2 100644 --- a/examples/get_and_parse_unseen_emails.php +++ b/examples/get_and_parse_unseen_emails.php @@ -23,7 +23,7 @@ try { $mail_ids = $mailbox->searchMailbox('UNSEEN'); } catch (ConnectionException $ex) { - exit('IMAP connection failed: '.$ex->getMessage()); + exit('IMAP connection failed: '.$ex->getErrors('first')); } catch (Exception $ex) { exit('An error occured: '.$ex->getMessage()); } From 1388c68a5a81cab4622da0803b021e13953e2fb6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Kr=C3=A4tzig?= Date: Fri, 7 Jan 2022 01:39:23 +0100 Subject: [PATCH 3/4] Update PHP Unit tests after changing the exception handling --- tests/unit/ImapTest.php | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tests/unit/ImapTest.php b/tests/unit/ImapTest.php index 670eae0e..d5cd6fc3 100644 --- a/tests/unit/ImapTest.php +++ b/tests/unit/ImapTest.php @@ -8,10 +8,10 @@ use Generator; use ParagonIE\HiddenString\HiddenString; +use PhpImap\Exceptions\ConnectionException; use PHPUnit\Framework\TestCase as Base; use const SORTARRIVAL; use Throwable; -use UnexpectedValueException; /** * @psalm-type MAILBOX_ARGS = array{ @@ -35,13 +35,13 @@ class ImapTest extends Base use LiveMailboxTestingTrait; /** - * @psalm-return Generator<'CI ENV with invalid password'|'empty mailbox/username/password', array{0: UnexpectedValueException::class, 1: '/^IMAP error:.[AUTHENTICATIONFAILED]/'|'IMAP error:Can't open mailbox : no such mailbox', 2: array{0: HiddenString, 1: HiddenString, 2: HiddenString, 3: 0, 4: 0, 5: array}, 3?: true}, mixed, void> + * @psalm-return Generator<'CI ENV with invalid password'|'empty mailbox/username/password', array{0: ConnectionException::class, 1: '/^[AUTHENTICATIONFAILED]/'|'Can't open mailbox : no such mailbox', 2: array{0: HiddenString, 1: HiddenString, 2: HiddenString, 3: 0, 4: 0, 5: array}, 3?: true}, mixed, void> */ public function OpenFailure(): Generator { yield 'empty mailbox/username/password' => [ - UnexpectedValueException::class, - 'IMAP error:Can\'t open mailbox : no such mailbox', + ConnectionException::class, + 'Can\'t open mailbox : no such mailbox', [ new HiddenString(''), new HiddenString(''), @@ -58,8 +58,8 @@ public function OpenFailure(): Generator if (\is_string($imapPath) && \is_string($login) && \is_string($password)) { yield 'CI ENV with invalid password' => [ - UnexpectedValueException::class, - '/^IMAP error:.*\[AUTHENTICATIONFAILED\].*/', + ConnectionException::class, + '/^\[AUTHENTICATIONFAILED\].*/', [ new HiddenString($imapPath, true, true), new HiddenString($login, true, true), From f85dcc965ebb607955737853caf7755f2b0fbc19 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Kr=C3=A4tzig?= Date: Fri, 7 Jan 2022 01:44:56 +0100 Subject: [PATCH 4/4] Add PHP extension JSON as dependency --- README.md | 1 + composer.json | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 0b801e8e..e12199db 100644 --- a/README.md +++ b/README.md @@ -41,6 +41,7 @@ Initially released in December 2012, the PHP IMAP Mailbox is a powerful and open * PHP `iconv` extension must be present; so make sure this line is active in your php.ini: `extension=php_iconv.dll` * PHP `imap` extension must be present; so make sure this line is active in your php.ini: `extension=php_imap.dll` * PHP `mbstring` extension must be present; so make sure this line is active in your php.ini: `extension=php_mbstring.dll` +* PHP `json` extension must be present; so make sure this line is active in your php.ini: `extension=json.dll` ### Installation by Composer diff --git a/composer.json b/composer.json index 5e897814..77176bdd 100644 --- a/composer.json +++ b/composer.json @@ -27,7 +27,8 @@ "ext-fileinfo": "*", "ext-iconv": "*", "ext-imap": "*", - "ext-mbstring": "*" + "ext-mbstring": "*", + "ext-json": "*" }, "require-dev": { "friendsofphp/php-cs-fixer": "^3.4",