Fix phpmailer (Case sensitive Check)
authorPhilipp <admin@philipp.info>
Tue, 22 Sep 2020 20:48:34 +0000 (22:48 +0200)
committerPhilipp <admin@philipp.info>
Tue, 22 Sep 2020 20:48:34 +0000 (22:48 +0200)
src/Util/Emailer.php
tests/Util/EmailerSpy.php [new file with mode: 0644]
tests/Util/HookMockTrait.php [new file with mode: 0644]
tests/src/Util/EMailerTest.php [new file with mode: 0644]
tests/src/Util/Emailer/MailBuilderTest.php

index b402abf..203f3b4 100644 (file)
@@ -134,6 +134,18 @@ class Emailer
                        return true;
                }
 
+               // @see https://github.com/friendica/friendica/issues/9142
+               $countMessageId = 0;
+               foreach ($email->getAdditionalMailHeader() as $name => $value) {
+                       if (strtolower($name) == 'message-id') {
+                               $countMessageId += count($value);
+                       }
+               }
+               if ($countMessageId > 0) {
+                       $this->logger->warning('More than one Message-ID found - RFC violation', ['email' => $email]);
+                       return false;
+               }
+
                $email_textonly = false;
                if (!empty($email->getRecipientUid())) {
                        $email_textonly = $this->pConfig->get($email->getRecipientUid(), 'system', 'email_textonly');
diff --git a/tests/Util/EmailerSpy.php b/tests/Util/EmailerSpy.php
new file mode 100644 (file)
index 0000000..effd2ed
--- /dev/null
@@ -0,0 +1,23 @@
+<?php
+
+namespace Friendica\Test\Util;
+
+use Friendica\Util\Emailer;
+
+class EmailerSpy extends Emailer
+{
+       public static $MAIL_DATA;
+
+       protected function mail(string $to, string $subject, string $body, string $headers, string $parameters)
+       {
+               self::$MAIL_DATA = [
+                       'to' => $to,
+                       'subject' => $subject,
+                       'body' => $body,
+                       'headers' => $headers,
+                       'parameters' => $parameters,
+               ];
+
+               return true;
+       }
+}
diff --git a/tests/Util/HookMockTrait.php b/tests/Util/HookMockTrait.php
new file mode 100644 (file)
index 0000000..68a17c5
--- /dev/null
@@ -0,0 +1,30 @@
+<?php
+
+namespace Friendica\Test\Util;
+
+use Friendica\Core\Hook;
+use Mockery\MockInterface;
+
+trait HookMockTrait
+{
+
+       /**
+        * @var MockInterface The Interface for mocking a renderer
+        */
+       private $hookMock;
+
+       /**
+        * Mocking a method 'Hook::call()' call
+        *
+        */
+       public function mockHookCallAll(string $name, &$capture)
+       {
+               if (!isset($this->hookMock)) {
+                       $this->hookMock = \Mockery::mock('alias:' . Hook::class);
+               }
+
+               $this->hookMock
+                       ->shouldReceive('callAll')
+                       ->withArgs([$name, \Mockery::capture($capture)]);
+       }
+}
diff --git a/tests/src/Util/EMailerTest.php b/tests/src/Util/EMailerTest.php
new file mode 100644 (file)
index 0000000..636965e
--- /dev/null
@@ -0,0 +1,127 @@
+<?php
+
+namespace Friendica\Test\src\Util;
+
+use Friendica\App\BaseURL;
+use Friendica\Core\Config\IConfig;
+use Friendica\Core\L10n;
+use Friendica\Core\PConfig\IPConfig;
+use Friendica\Object\EMail\IEmail;
+use Friendica\Test\MockedTest;
+use Friendica\Test\Util\EmailerSpy;
+use Friendica\Test\Util\HookMockTrait;
+use Friendica\Test\Util\SampleMailBuilder;
+use Friendica\Test\Util\VFSTrait;
+use Mockery\MockInterface;
+use Psr\Log\NullLogger;
+
+/**
+ * Annotation necessary because of Hook calls
+ *
+ * @runTestsInSeparateProcesses
+ * @preserveGlobalState false
+ */
+class EMailerTest extends MockedTest
+{
+       use VFSTrait;
+       use HookMockTrait;
+
+       /** @var IConfig|MockInterface */
+       private $config;
+       /** @var IPConfig|MockInterface */
+       private $pConfig;
+       /** @var L10n|MockInterface */
+       private $l10n;
+       /** @var BaseURL|MockInterface */
+       private $baseUrl;
+
+       /** @var string */
+       private $defaultHeaders;
+
+       protected function setUp()
+       {
+               parent::setUp();
+
+               $this->setUpVfsDir();
+
+               $this->config  = \Mockery::mock(IConfig::class);
+               $this->config->shouldReceive('get')->withArgs(['config', 'sender_email'])->andReturn('test@friendica.local')->once();
+               $this->config->shouldReceive('get')->withArgs(['config', 'sitename', 'Friendica Social Network'])->andReturn('Friendica Social Network')->once();
+               $this->config->shouldReceive('get')->withArgs(['system', 'sendmail_params', true])->andReturn(true);
+
+               $this->pConfig = \Mockery::mock(IPConfig::class);
+               $this->l10n    = \Mockery::mock(L10n::class);
+               $this->baseUrl = \Mockery::mock(BaseURL::class);
+               $this->baseUrl->shouldReceive('getHostname')->andReturn('friendica.local');
+               $this->baseUrl->shouldReceive('get')->andReturn('http://friendica.local');
+
+               $this->defaultHeaders = [];
+       }
+
+       protected function tearDown()
+       {
+               EmailerSpy::$MAIL_DATA = [];
+
+               parent::tearDown();
+       }
+
+       public function testEmail()
+       {
+               $this->pConfig->shouldReceive('get')->withArgs(['1', 'system', 'email_textonly'])->andReturn(false)->once();
+
+               $builder = new SampleMailBuilder($this->l10n, $this->baseUrl, $this->config, new NullLogger());
+
+               $testEmail = $builder
+                       ->withRecipient('recipient@friendica.local')
+                       ->withMessage('Test Subject', "Test Message<b>Bold</b>", 'Test Text')
+                       ->withSender('Sender', 'sender@friendica.loca')
+                       ->forUser(['uid' => 1])
+                       ->addHeader('Message-ID', 'first Id')
+                       ->build(true);
+
+               $emailer = new EmailerSpy($this->config, $this->pConfig, $this->baseUrl, new NullLogger(), $this->l10n);
+
+               $this->assertTrue($emailer->send($testEmail));
+
+               $this->assertContains("X-Friendica-Host : friendica.local", EmailerSpy::$MAIL_DATA['headers']);
+               $this->assertContains("X-Friendica-Platform : Friendica", EmailerSpy::$MAIL_DATA['headers']);
+               $this->assertContains("List-ID : <notification.friendica.local>", EmailerSpy::$MAIL_DATA['headers']);
+               $this->assertContains("List-Archive : <http://friendica.local/notifications/system>", EmailerSpy::$MAIL_DATA['headers']);
+               $this->assertContains("Reply-To: Sender <sender@friendica.loca>", EmailerSpy::$MAIL_DATA['headers']);
+               $this->assertContains("MIME-Version: 1.0", EmailerSpy::$MAIL_DATA['headers']);
+               // Base64 "Test Text"
+               $this->assertContains(chunk_split(base64_encode('Test Text')), EmailerSpy::$MAIL_DATA['body']);
+               // Base64 "Test Message<b>Bold</b>"
+               $this->assertContains(chunk_split(base64_encode("Test Message<b>Bold</b>")), EmailerSpy::$MAIL_DATA['body']);
+               $this->assertEquals("Test Subject", EmailerSpy::$MAIL_DATA['subject']);
+               $this->assertEquals("recipient@friendica.local", EmailerSpy::$MAIL_DATA['to']);
+               $this->assertEquals("-f sender@friendica.local", EmailerSpy::$MAIL_DATA['parameters']);
+       }
+
+       public function testWrongReturnTwoMessageIds()
+       {
+               /** @var IEmail $returnMail */
+               $returnMail = null;
+
+               $this->mockHookCallAll('emailer_send_prepare', $returnMail);
+
+               $builder = new SampleMailBuilder($this->l10n, $this->baseUrl, $this->config, new NullLogger());
+
+               $testEmail = $builder
+                       ->withRecipient('recipient@friendica.local')
+                       ->withMessage('Test Subject', "Test Message<b>Bold</b>", 'Test Text')
+                       ->withSender('Sender', 'sender@friendica.loca')
+                       ->forUser(['uid' => 1])
+                       ->addHeader('Message-ID', 'first Id')
+                       ->addHeader('Message-Id', 'second Id')
+                       ->build(true);
+
+               $emailer = new EmailerSpy($this->config, $this->pConfig, $this->baseUrl, new NullLogger(), $this->l10n);
+
+               $this->assertFalse($emailer->send($testEmail));
+
+               // check case sensitive key problem
+               $this->assertArrayHasKey('Message-ID', $testEmail->getAdditionalMailHeader());
+               $this->assertArrayHasKey('Message-Id', $testEmail->getAdditionalMailHeader());
+       }
+}
index 202ad58..b890224 100644 (file)
@@ -29,6 +29,7 @@ use Friendica\Test\MockedTest;
 use Friendica\Test\Util\SampleMailBuilder;
 use Friendica\Test\Util\VFSTrait;
 use Friendica\Util\EMailer\MailBuilder;
+use Mockery\MockInterface;
 use Psr\Log\NullLogger;
 
 /**
@@ -39,11 +40,11 @@ class MailBuilderTest extends MockedTest
 {
        use VFSTrait;
 
-       /** @var IConfig */
+       /** @var IConfig|MockInterface */
        private $config;
-       /** @var L10n */
+       /** @var L10n|MockInterface */
        private $l10n;
-       /** @var BaseURL */
+       /** @var BaseURL|MockInterface */
        private $baseUrl;
 
        /** @var string */