set('alternate_base_url', 'http://www.mysite.com/'); $this->origThemes = SSViewer::get_themes(); SSViewer::set_themes([ 'silverstripe/framework:/tests/php/Control/Email/EmailTest', '$default', ]); } protected function tearDown(): void { parent::tearDown(); SSViewer::set_themes($this->origThemes); } public function testAddAttachment(): void { $email = new Email(); $email->addAttachment(__DIR__ . '/EmailTest/attachment.txt', null, 'text/plain'); $attachments = $email->getAttachments(); $this->assertCount(1, $attachments); $attachment = $this->getFirstAttachment($attachments); $this->assertSame('text/plain', $attachment->getContentType()); $this->assertSame('attachment.txt', $attachment->getFilename()); } public function testAddAttachmentFromData(): void { $email = new Email(); $email->addAttachmentFromData('foo bar', 'foo.txt', 'text/plain'); $attachments = $email->getAttachments(); $this->assertCount(1, $attachments); $attachment = $this->getFirstAttachment($attachments); $this->assertSame('text/plain', $attachment->getContentType()); $this->assertSame('foo.txt', $attachment->getFilename()); $this->assertSame('foo bar', $attachment->getBody()); } private function getFirstAttachment(array $attachments): DataPart { return $attachments[0]; } /** * @dataProvider provideValidEmailAddresses */ public function testValidEmailAddress($email): void { $this->assertTrue(Email::is_valid_address($email)); } /** * @dataProvider provideInvalidEmailAddresses */ public function testInvalidEmailAddress($email): void { $this->assertFalse(Email::is_valid_address($email)); } public function provideValidEmailAddresses(): array { return [ ['test@example.com', 'test-123@sub.example.com'], ]; } public function provideInvalidEmailAddresses(): array { return [ ['foo.bar@', '@example.com', 'foo@'], ]; } public function testObfuscate(): void { $emailAddress = 'test-1@example.com'; $direction = Email::obfuscate($emailAddress, 'direction'); $visible = Email::obfuscate($emailAddress, 'visible'); $hex = Email::obfuscate($emailAddress, 'hex'); $this->assertEquals('moc.elpmaxe@1-tset', $direction); $this->assertEquals('test [dash] 1 [at] example [dot] com', $visible); $this->assertEquals( 'test-1@example.com', $hex ); } private function getTemplateClass(string $templateName): string { return implode('\\', ['SilverStripe', 'Control', 'Tests', 'Email', 'EmailTest', $templateName]); } private function getMailer(): TestMailer { return Injector::inst()->get(MailerInterface::class); } private function createTestEmail(string $subject = 'My subject', $setPlain = true): Email { $email = new Email(); $email->setFrom('from@example.com'); $email->setTo('to@example.com'); $email->setSubject($subject); if ($setPlain) { $email->text("Plain body for $subject"); } $email->html("

HTML body for $subject

"); $email->setCC('cc@example.com'); $email->setBCC('bcc@example.com'); $email->addAttachment(__DIR__ . '/EmailTest/attachment.txt', null, 'text/plain'); return $email; } public function testSendPlain(): void { $email = $this->createTestEmail('Test send plain'); $email->sendPlain(); $this->assertStringNotContainsString($email->getTextBody(), 'My Plain Template'); $sentMail = $this->getMailer()->findEmail('to@example.com'); $this->assertSame('to@example.com', $sentMail['To']); $this->assertSame('from@example.com', $sentMail['From']); $this->assertSame('Test send plain', $sentMail['Subject']); $this->assertStringContainsString('Plain body for Test send plain', $sentMail['Content']); $this->assertCount(1, $sentMail['AttachedFiles']); $child = reset($sentMail['AttachedFiles']); $this->assertSame('text/plain', $child['mimetype']); $this->assertSame('attachment.txt', $child['filename']); $this->assertSame('Hello, I\'m a text document.', $child['contents']); // assert MIME types // explicitly setting $email->html(null) because sendPlain() will itself set $this->html(null), and then // revert it to its previous AFTER sending the email. For testing purposes, we need to manuall set it // to null in order to test the MIME types for what would have been sent in practice $email->html(null); $this->assertSame([ 'text/plain charset: utf-8', 'text/plain disposition: attachment filename: attachment.txt' ], array_map(fn(AbstractPart $part) => $part->asDebugString(), $email->getBody()->getParts())); } public function testSendPlainFallback(): void { $email = $this->createTestEmail('Test send plain', false); $email->sendPlain(); $sentMail = $this->getMailer()->findEmail('to@example.com'); // assert that it has HTML body with HTML tags removed $this->assertSame('HTML body for Test send plain', $sentMail['Content']); } public function testSendPlainThenNormalWithSetData(): void { $email = $this->createTestEmail('Test send plain', false); $email->setData([ 'EmailContent' => 'This is the content of the email', ]); $email->sendPlain(); $email->send(); $sentMail = $this->getMailer()->findEmail('to@example.com'); $this->assertSame('This is the content of the email', $sentMail['Content']); $email->to('to2@example.com'); $email->send(); $sentMail = $this->getMailer()->findEmail('to2@example.com'); $this->assertStringContainsString('This is the content of the email', $sentMail['Content']); } public function testSend(): void { $email = $this->createTestEmail('Test send HTML'); // email should not call render if a body is supplied $email->setHTMLTemplate($this->getTemplateClass('HtmlTemplate')); $email->send(); $this->assertStringNotContainsString($email->getHtmlBody(), 'My HTML Template'); $sentMail = $this->getMailer()->findEmail('to@example.com'); $this->assertSame('to@example.com', $sentMail['To']); $this->assertSame('from@example.com', $sentMail['From']); $this->assertSame('Test send HTML', $sentMail['Subject']); $this->assertStringContainsString('

HTML body for Test send HTML

', $sentMail['Content']); $this->assertCount(1, $sentMail['AttachedFiles']); $child = reset($sentMail['AttachedFiles']); $this->assertSame('text/plain', $child['mimetype']); $this->assertSame('attachment.txt', $child['filename']); $this->assertSame('Hello, I\'m a text document.', $child['contents']); // assert MIME types $this->assertSame([ implode("\n └ ", [ 'multipart/alternative', 'text/plain charset: utf-8', 'text/html charset: utf-8' ]), 'text/plain disposition: attachment filename: attachment.txt' ], array_map(fn(AbstractPart $part) => $part->asDebugString(), $email->getBody()->getParts())); } public function testRenderedSend(): void { $email = new Email(to: 'to@example.com'); $email->setHTMLTemplate($this->getTemplateClass('HtmlTemplate')); $email->setData([ 'EmailContent' => '

test

', ]); $email->send(); $sentMail = $this->getMailer()->findEmail('to@example.com'); $this->assertStringContainsString('My HTML Template', $sentMail['Content']); } public function testRenderedSendSubclass(): void { $email = new EmailSubClass(to: 'to@example.com'); $email->setData([ 'EmailContent' => 'test', ]); $email->send(); $sentMail = $this->getMailer()->findEmail('to@example.com'); $this->assertStringContainsString('

Email Sub-class

', $sentMail['Content']); } public function testConstructor(): void { $email = new Email( 'from@example.com', 'to@example.com', 'subject', '

body

', 'cc@example.com', 'bcc@example.com', 'bounce@example.com' ); $this->assertCount(1, $email->getFrom()); $this->assertSame('from@example.com', $email->getFrom()[0]->getAddress()); $this->assertCount(1, $email->getTo()); $this->assertSame('to@example.com', $email->getTo()[0]->getAddress()); $this->assertEquals('subject', $email->getSubject()); $this->assertEquals('

body

', $email->getHtmlBody()); $this->assertCount(1, $email->getCC()); $this->assertEquals('cc@example.com', $email->getCC()[0]->getAddress()); $this->assertCount(1, $email->getBCC()); $this->assertEquals('bcc@example.com', $email->getBcc()[0]->getAddress()); $this->assertEquals('bounce@example.com', $email->getReturnPath()->getAddress()); } public function testSetBody(): void { $email = new Email(); $email->setBody('

body

'); $this->assertSame('

body

', $email->getHtmlBody()); } public function testSetFrom(): void { $email = new Email(); $email->setFrom('from@example.com'); $this->assertCount(1, $email->getFrom()); $this->assertSame('from@example.com', $email->getFrom()[0]->getAddress()); } public function testSender(): void { $email = new Email(); $email->setSender('sender@example.com'); $this->assertSame('sender@example.com', $email->getSender()->getAddress()); } public function testSetTo(): void { $email = new Email(); $email->setTo('to@example.com'); $this->assertCount(1, $email->getTo()); $this->assertSame('to@example.com', $email->getTo()[0]->getAddress()); } public function testSetReplyTo(): void { $email = new Email(); $email->setReplyTo('reply-to@example.com'); $this->assertCount(1, $email->getReplyTo()); $this->assertSame('reply-to@example.com', $email->getReplyTo()[0]->getAddress()); } public function testSetSubject(): void { $email = new Email(); $email->setSubject('my subject'); $this->assertSame('my subject', $email->getSubject()); } public function testSetReturnPath(): void { $email = new Email(); $email->setReturnPath('return-path@example.com'); $this->assertSame('return-path@example.com', $email->getReturnPath()->getAddress()); } public function testSetPriority(): void { $email = new Email(); // Intentionally set above 5 to test that Symfony\Component\Mime\Email->priority() is being called $email->setPriority(7); $this->assertSame(5, $email->getPriority()); } public function testAdminEmailApplied() { Email::config()->set('admin_email', 'admin@example.com'); $email = new Email(); $this->assertCount(1, $email->getFrom()); $this->assertSame('admin@example.com', $email->getFrom()[0]->getAddress()); } public function testDataWithArray(): void { $email = new Email(); $this->assertSame(true, $email->getData()->IsEmail); $this->assertSame(Director::absoluteBaseURL(), $email->getData()->BaseURL); $email->setData(['Lorem' => 'Ipsum']); $this->assertSame(true, $email->getData()->IsEmail); $this->assertSame(Director::absoluteBaseURL(), $email->getData()->BaseURL); $this->assertSame('Ipsum', $email->getData()->Lorem); $email->addData('Content', 'My content'); $this->assertSame(true, $email->getData()->IsEmail); $this->assertSame(Director::absoluteBaseURL(), $email->getData()->BaseURL); $this->assertSame('Ipsum', $email->getData()->Lorem); $this->assertSame('My content', $email->getData()->Content); } public function testDataWithViewableData(): void { $email = new Email(); $viewableData = new ViewableData(); $viewableData->ABC = 'XYZ'; $email->setData($viewableData); $data = $email->getData(); $this->assertSame('XYZ', $data->ABC); $this->assertSame(true, $data->IsEmail); $this->assertSame(Director::absoluteBaseURL(), $data->BaseURL); $member = new Member(); $member->FirstName = 'First Name'; $email->setData($member); $this->assertSame($member->FirstName, $email->getData()->FirstName); $email->addData('Test', 'Test value'); $this->assertEquals('Test value', $email->getData()->Test); $email->removeData('Test'); $this->assertNull($email->getData()->Test); } public function testHTMLTemplate(): void { // Find template on disk $emailTemplate = ModuleResourceLoader::singleton()->resolveResource( 'silverstripe/framework:templates/SilverStripe/Control/Email/Email.ss' ); $subClassTemplate = ModuleResourceLoader::singleton()->resolveResource( 'silverstripe/framework:tests/php/Control/Email/EmailTest/templates/' . str_replace('\\', '/', EmailSubClass::class) . '.ss' ); $this->assertTrue($emailTemplate->exists()); $this->assertTrue($subClassTemplate->exists()); // Check template is auto-found $email = new Email(); $this->assertEquals($emailTemplate->getPath(), $email->getHTMLTemplate()); $email->setHTMLTemplate('MyTemplate'); $this->assertEquals('MyTemplate', $email->getHTMLTemplate()); // Check subclass template is found $email2 = new EmailSubClass(); $this->assertEquals($subClassTemplate->getPath(), $email2->getHTMLTemplate()); $email->setHTMLTemplate('MyTemplate'); $this->assertEquals('MyTemplate', $email->getHTMLTemplate()); } public function testPlainTemplate(): void { $email = new Email(); $this->assertEmpty($email->getPlainTemplate()); $email->setPlainTemplate('MyTemplate'); $this->assertEquals('MyTemplate', $email->getPlainTemplate()); } public function testRerender(): void { $email = new Email(); $email->setPlainTemplate($this->getTemplateClass('PlainTemplate')); $email->setData([ 'EmailContent' => '

my content

', ]); $email->send(); $this->assertStringContainsString('<p>my content</p>', $email->getHtmlBody()); // Ensure setting data causes html() to be updated $email->setData([ 'EmailContent' => '

your content

' ]); $email->send(); $this->assertStringContainsString('<p>your content</p>', $email->getHtmlBody()); // Ensure removing data causes html() to be updated $email->removeData('EmailContent'); $email->send(); $this->assertStringNotContainsString('<p>your content</p>', $email->getHtmlBody()); // Ensure adding data causes html() to be updated $email->addData([ 'EmailContent' => '

their content

' ]); $email->send(); $this->assertStringContainsString('<p>their content</p>', $email->getHtmlBody()); } public function testRenderPlainOnly(): void { $email = new Email(); $email->setData([ 'EmailContent' => 'test content', ]); $email->sendPlain(); $this->assertSame('test content', $email->getTextBody()); } public function testMultipleEmailSends(): void { $email = new Email(to: 'to@example.com'); $email->setData([ 'EmailContent' => '

Test

', ]); $this->assertSame(null, $email->getHtmlBody()); $this->assertSame(null, $email->getTextBody()); $email->send(); $this->assertStringContainsString('<p>Test</p>', $email->getHtmlBody()); $this->assertSame('Test', $email->getTextBody()); //send again $email->send(); $this->assertStringContainsString('<p>Test</p>', $email->getHtmlBody()); $this->assertSame('Test', $email->getTextBody()); } public function testGetDefaultFrom(): void { $email = new Email(); $class = new \ReflectionClass(Email::class); $method = $class->getMethod('getDefaultFrom'); $method->setAccessible(true); // default to no-reply@mydomain.com if admin_email config not set Email::config()->set('admin_email', null); $this->assertSame('no-reply@www.mysite.com', $method->invokeArgs($email, [])); // default to no-reply@mydomain.com if admin_email config is misconfigured Email::config()->set('admin_email', 123); $this->assertSame('no-reply@www.mysite.com', $method->invokeArgs($email, [])); // use admin_email config string syntax Email::config()->set('admin_email', 'myadmin@somewhere.com'); $this->assertSame('myadmin@somewhere.com', $method->invokeArgs($email, [])); $this->assertTrue(true); // use admin_email config array syntax Email::config()->set('admin_email', ['anotheradmin@somewhere.com' => 'Admin-email']); $this->assertSame('anotheradmin@somewhere.com', $method->invokeArgs($email, [])); $this->assertTrue(true); } /** * @dataProvider provideCreateAddressArray */ public function testCreateAddressArray(string|array $address, string $name, array $expected): void { $method = new \ReflectionMethod(Email::class, 'createAddressArray'); $method->setAccessible(true); $obj = new Email(); $actual = $method->invoke($obj, $address, $name); for ($i = 0; $i < count($expected); $i++) { $this->assertSame($expected[$i]->getAddress(), $actual[$i]->getAddress()); $this->assertSame($expected[$i]->getName(), $actual[$i]->getName()); } } public function provideCreateAddressArray(): array { return [ [ 'my@email.com', 'My name', [ new Address('my@email.com', 'My name'), ], ], [ [ 'my@email.com' => 'My name', 'other@email.com' => 'My other name', 'no-name@email.com' ], '', [ new Address('my@email.com', 'My name'), new Address('other@email.com', 'My other name'), new Address('no-name@email.com', ''), ], ] ]; } }