Skip to content

Commit

Permalink
bug #31774 [Mailer] Fix the possibility to set a From header from Mes…
Browse files Browse the repository at this point in the history
…sageListener (fabpot)

This PR was merged into the 4.3 branch.

Discussion
----------

[Mailer] Fix the possibility to set a From header from MessageListener

| Q             | A
| ------------- | ---
| Branch?       | 4.3
| Bug fix?      | yes
| New feature?  | no <!-- please update src/**/CHANGELOG.md files -->
| BC breaks?    | no     <!-- see https://symfony.com/bc -->
| Deprecations? | no <!-- please update UPGRADE-*.md and src/**/CHANGELOG.md files -->
| Tests pass?   | yes    <!-- please add some, will be required by reviewers -->
| Fixed tickets | #31733
| License       | MIT
| Doc PR        | n/a

<!--
Replace this notice by a short README for your feature/bugfix. This will help people
understand your PR and can be used as a start for the documentation.

Additionally (see https://symfony.com/roadmap):
 - Bug fixes must be submitted against the lowest maintained branch where they apply
   (lowest branches are regularly merged to upper ones so they get the fixes too).
 - Features and deprecations must be submitted against branch 4.4.
 - Legacy code removals go to the master branch.
-->

Commits
-------

f4254e6 [Mailer] fixed the possibility to set a From header from MessageListener
  • Loading branch information
fabpot committed Jun 4, 2019
2 parents 66b87ab + f4254e6 commit 2438b14
Show file tree
Hide file tree
Showing 4 changed files with 112 additions and 42 deletions.
105 changes: 105 additions & 0 deletions src/Symfony/Component/Mailer/DelayedSmtpEnvelope.php
@@ -0,0 +1,105 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\Mailer;

use Symfony\Component\Mailer\Exception\InvalidArgumentException;
use Symfony\Component\Mailer\Exception\LogicException;
use Symfony\Component\Mime\Address;
use Symfony\Component\Mime\Header\Headers;
use Symfony\Component\Mime\Message;
use Symfony\Component\Mime\RawMessage;

/**
* @author Fabien Potencier <fabien@symfony.com>
*
* @experimental in 4.3
*
* @internal
*/
final class DelayedSmtpEnvelope extends SmtpEnvelope
{
private $senderSet = false;
private $recipientsSet = false;
private $message;

public function __construct(RawMessage $message)
{
if (!$message instanceof Message) {
// FIXME: parse the raw message to create the envelope?
throw new InvalidArgumentException(sprintf('Unable to create an SmtpEnvelope from a "%s" message.', RawMessage::class));
}

$this->message = $message;
}

public function setSender(Address $sender): void
{
parent::setSender($sender);

$this->senderSet = true;
}

public function getSender(): Address
{
if ($this->senderSet) {
return parent::getSender();
}

return self::getSenderFromHeaders($this->message->getHeaders());
}

public function setRecipients(array $recipients): void
{
parent::setRecipients($recipients);

$this->recipientsSet = parent::getRecipients();
}

/**
* @return Address[]
*/
public function getRecipients(): array
{
if ($this->recipientsSet) {
return parent::getRecipients();
}

return self::getRecipientsFromHeaders($this->message->getHeaders());
}

private static function getRecipientsFromHeaders(Headers $headers): array
{
$recipients = [];
foreach (['to', 'cc', 'bcc'] as $name) {
foreach ($headers->getAll($name) as $header) {
$recipients = array_merge($recipients, $header->getAddresses());
}
}

return $recipients;
}

private static function getSenderFromHeaders(Headers $headers): Address
{
if ($return = $headers->get('Return-Path')) {
return $return->getAddress();
}
if ($sender = $headers->get('Sender')) {
return $sender->getAddress();
}
if ($from = $headers->get('From')) {
return $from->getAddresses()[0];
}

throw new LogicException('Unable to determine the sender of the message.');
}
}
39 changes: 1 addition & 38 deletions src/Symfony/Component/Mailer/SmtpEnvelope.php
Expand Up @@ -12,10 +12,7 @@
namespace Symfony\Component\Mailer;

use Symfony\Component\Mailer\Exception\InvalidArgumentException;
use Symfony\Component\Mailer\Exception\LogicException;
use Symfony\Component\Mime\Address;
use Symfony\Component\Mime\Header\Headers;
use Symfony\Component\Mime\Message;
use Symfony\Component\Mime\NamedAddress;
use Symfony\Component\Mime\RawMessage;

Expand All @@ -40,14 +37,7 @@ public function __construct(Address $sender, array $recipients)

public static function create(RawMessage $message): self
{
if ($message instanceof Message) {
$headers = $message->getHeaders();

return new self(self::getSenderFromHeaders($headers), self::getRecipientsFromHeaders($headers));
}

// FIXME: parse the raw message to create the envelope?
throw new InvalidArgumentException(sprintf('Unable to create an SmtpEnvelope from a "%s" message.', RawMessage::class));
return new DelayedSmtpEnvelope($message);
}

public function setSender(Address $sender): void
Expand Down Expand Up @@ -84,31 +74,4 @@ public function getRecipients(): array
{
return $this->recipients;
}

private static function getRecipientsFromHeaders(Headers $headers): array
{
$recipients = [];
foreach (['to', 'cc', 'bcc'] as $name) {
foreach ($headers->getAll($name) as $header) {
$recipients = array_merge($recipients, $header->getAddresses());
}
}

return $recipients;
}

private static function getSenderFromHeaders(Headers $headers): Address
{
if ($return = $headers->get('Return-Path')) {
return $return->getAddress();
}
if ($sender = $headers->get('Sender')) {
return $sender->getAddress();
}
if ($from = $headers->get('From')) {
return $from->getAddresses()[0];
}

throw new LogicException('Unable to determine the sender of the message.');
}
}
7 changes: 4 additions & 3 deletions src/Symfony/Component/Mailer/Tests/SmtpEnvelopeTest.php
Expand Up @@ -72,12 +72,13 @@ public function testSenderFromHeaders()
$this->assertEquals('from@symfony.com', $e->getSender()->getAddress());
}

public function testSenderFromHeadersWithoutData()
public function testSenderFromHeadersWithoutFrom()
{
$this->expectException(\LogicException::class);
$headers = new Headers();
$headers->addMailboxListHeader('To', ['from@symfony.com']);
SmtpEnvelope::create(new Message($headers));
$e = SmtpEnvelope::create($message = new Message($headers));
$message->getHeaders()->addMailboxListHeader('From', ['from@symfony.com']);
$this->assertEquals('from@symfony.com', $e->getSender()->getAddress());
}

public function testRecipientsFromHeaders()
Expand Down
3 changes: 2 additions & 1 deletion src/Symfony/Component/Mailer/Transport/AbstractTransport.php
Expand Up @@ -15,6 +15,7 @@
use Psr\Log\NullLogger;
use Symfony\Component\EventDispatcher\EventDispatcher;
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
use Symfony\Component\Mailer\DelayedSmtpEnvelope;
use Symfony\Component\Mailer\Event\MessageEvent;
use Symfony\Component\Mailer\Exception\TransportException;
use Symfony\Component\Mailer\SentMessage;
Expand Down Expand Up @@ -62,7 +63,7 @@ public function send(RawMessage $message, SmtpEnvelope $envelope = null): ?SentM
$envelope = clone $envelope;
} else {
try {
$envelope = SmtpEnvelope::create($message);
$envelope = new DelayedSmtpEnvelope($message);
} catch (\Exception $e) {
throw new TransportException('Cannot send message without a valid envelope.', 0, $e);
}
Expand Down

0 comments on commit 2438b14

Please sign in to comment.