mirror of
https://github.com/espocrm/espocrm.git
synced 2026-06-28 06:56:05 +00:00
721 lines
21 KiB
PHP
721 lines
21 KiB
PHP
<?php
|
||
/************************************************************************
|
||
* This file is part of EspoCRM.
|
||
*
|
||
* EspoCRM – Open Source CRM application.
|
||
* Copyright (C) 2014-2025 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
|
||
* Website: https://www.espocrm.com
|
||
*
|
||
* This program is free software: you can redistribute it and/or modify
|
||
* it under the terms of the GNU Affero General Public License as published by
|
||
* the Free Software Foundation, either version 3 of the License, or
|
||
* (at your option) any later version.
|
||
*
|
||
* This program is distributed in the hope that it will be useful,
|
||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
* GNU Affero General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Affero General Public License
|
||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
*
|
||
* The interactive user interfaces in modified source and object code versions
|
||
* of this program must display Appropriate Legal Notices, as required under
|
||
* Section 5 of the GNU Affero General Public License version 3.
|
||
*
|
||
* In accordance with Section 7(b) of the GNU Affero General Public License version 3,
|
||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||
************************************************************************/
|
||
|
||
namespace Espo\Core\Mail;
|
||
|
||
use Espo\Core\FileStorage\Manager as FileStorageManager;
|
||
use Espo\Core\Mail\Exceptions\NoSmtp;
|
||
use Espo\Core\Mail\Smtp\TransportFactory;
|
||
use Espo\Core\ORM\Repository\Option\SaveOption;
|
||
use Espo\ORM\EntityCollection;
|
||
use Espo\Core\Field\DateTime;
|
||
use Espo\Core\Utils\Config;
|
||
use Espo\Core\Utils\Log;
|
||
use Espo\Core\Mail\Account\SendingAccountProvider;
|
||
use Espo\Core\Mail\Exceptions\SendingError;
|
||
use Espo\Entities\Attachment;
|
||
use Espo\Entities\Email;
|
||
use Espo\ORM\EntityManager;
|
||
|
||
use Laminas\Mail\Header\ContentType as ContentTypeHeader;
|
||
use Laminas\Mail\Header\MessageId as MessageIdHeader;
|
||
use Laminas\Mail\Header\Sender as SenderHeader;
|
||
use Laminas\Mail\Message;
|
||
use Laminas\Mail\Protocol\Exception\RuntimeException as ProtocolRuntimeException;
|
||
use Laminas\Mail\Transport\Envelope;
|
||
use Laminas\Mail\Transport\Smtp as SmtpTransport;
|
||
use Laminas\Mail\Transport\SmtpOptions;
|
||
use Laminas\Mime\Message as MimeMessage;
|
||
use Laminas\Mime\Mime as Mime;
|
||
use Laminas\Mime\Part as MimePart;
|
||
|
||
use Exception;
|
||
use InvalidArgumentException;
|
||
|
||
/**
|
||
* Sends emails. Builds parameters for sending. Should not be used directly.
|
||
*/
|
||
class Sender
|
||
{
|
||
private ?SmtpTransport $transport = null;
|
||
private bool $isGlobal = false;
|
||
/** @var array<string, mixed> */
|
||
private array $params = [];
|
||
/** @var array<string, mixed> */
|
||
private array $overrideParams = [];
|
||
private ?Envelope $envelope = null;
|
||
private ?Message $message = null;
|
||
/** @var ?iterable<Attachment> */
|
||
private $attachmentList = null;
|
||
|
||
private const ATTACHMENT_ATTR_CONTENTS = 'contents';
|
||
|
||
private const MESSAGE_TYPE_MULTIPART_RELATED = 'multipart/related';
|
||
private const MESSAGE_TYPE_MULTIPART_ALTERNATIVE = 'multipart/alternative';
|
||
private const MESSAGE_TYPE_MULTIPART_MIXED = 'multipart/mixed';
|
||
private const MESSAGE_TYPE_TEXT_PLAIN = 'text/plain';
|
||
private const MESSAGE_TYPE_TEXT_HTML = 'text/html';
|
||
|
||
public function __construct(
|
||
private Config $config,
|
||
private EntityManager $entityManager,
|
||
private Log $log,
|
||
private TransportFactory $transportFactory,
|
||
private SendingAccountProvider $accountProvider,
|
||
private FileStorageManager $fileStorageManager,
|
||
private ConfigDataProvider $configDataProvider,
|
||
) {
|
||
$this->useGlobal();
|
||
}
|
||
|
||
private function resetParams(): void
|
||
{
|
||
$this->params = [];
|
||
$this->envelope = null;
|
||
$this->message = null;
|
||
$this->attachmentList = null;
|
||
$this->overrideParams = [];
|
||
}
|
||
|
||
/**
|
||
* With parameters.
|
||
*
|
||
* @param SenderParams|array<string, mixed> $params
|
||
*/
|
||
public function withParams($params): self
|
||
{
|
||
if ($params instanceof SenderParams) {
|
||
$params = $params->toArray();
|
||
} else if (!is_array($params)) {
|
||
throw new InvalidArgumentException();
|
||
}
|
||
|
||
$paramList = [
|
||
'fromAddress',
|
||
'fromName',
|
||
'replyToAddress',
|
||
'replyToName',
|
||
];
|
||
|
||
foreach (array_keys($params) as $key) {
|
||
if (!in_array($key, $paramList)) {
|
||
unset($params[$key]);
|
||
}
|
||
}
|
||
|
||
$this->overrideParams = array_merge($this->overrideParams, $params);
|
||
|
||
return $this;
|
||
}
|
||
|
||
/**
|
||
* With specific SMTP parameters.
|
||
*
|
||
* @param SmtpParams|array<string, mixed> $params
|
||
*/
|
||
public function withSmtpParams($params): self
|
||
{
|
||
if ($params instanceof SmtpParams) {
|
||
$params = $params->toArray();
|
||
} else if (!is_array($params)) {
|
||
throw new InvalidArgumentException();
|
||
}
|
||
|
||
/** @noinspection PhpDeprecationInspection */
|
||
return $this->useSmtp($params);
|
||
}
|
||
|
||
/**
|
||
* With specific attachments.
|
||
*
|
||
* @param iterable<Attachment> $attachmentList
|
||
*/
|
||
public function withAttachments(iterable $attachmentList): self
|
||
{
|
||
$this->attachmentList = $attachmentList;
|
||
|
||
return $this;
|
||
}
|
||
|
||
/**
|
||
* With envelope options.
|
||
*
|
||
* @param array<string, mixed> $options
|
||
*/
|
||
public function withEnvelopeOptions(array $options): self
|
||
{
|
||
/** @noinspection PhpDeprecationInspection */
|
||
return $this->setEnvelopeOptions($options);
|
||
}
|
||
|
||
/**
|
||
* Set a message instance.
|
||
*/
|
||
public function withMessage(Message $message): self
|
||
{
|
||
$this->message = $message;
|
||
|
||
return $this;
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v6.0. Use withParams.
|
||
* @param array<string, mixed> $params
|
||
* @todo Remove in v10.0.
|
||
*/
|
||
public function setParams(array $params = []): self
|
||
{
|
||
$this->params = array_merge($this->params, $params);
|
||
|
||
return $this;
|
||
}
|
||
|
||
|
||
/**
|
||
* @deprecated As of 6.0. Use withSmtpParams.
|
||
* @param array<string, mixed> $params
|
||
* @todo Make private in v10.0.
|
||
*/
|
||
public function useSmtp(array $params = []): self
|
||
{
|
||
$this->isGlobal = false;
|
||
|
||
$this->applySmtp($params);
|
||
|
||
return $this;
|
||
}
|
||
|
||
private function useGlobal(): void
|
||
{
|
||
$this->params = [];
|
||
$this->isGlobal = true;
|
||
}
|
||
|
||
/**
|
||
* @param array<string, mixed> $params
|
||
*/
|
||
private function applySmtp(array $params = []): void
|
||
{
|
||
$this->params = $params;
|
||
|
||
$this->transport = $this->transportFactory->create();
|
||
|
||
$config = $this->config;
|
||
|
||
$localHostName = $config->get('smtpLocalHostName', gethostname());
|
||
|
||
$options = [
|
||
'name' => $localHostName,
|
||
'host' => $params['server'],
|
||
'port' => $params['port'],
|
||
'connectionConfig' => [],
|
||
];
|
||
|
||
$connectionOptions = $params['connectionOptions'] ?? [];
|
||
|
||
foreach ($connectionOptions as $key => $value) {
|
||
$options['connectionConfig'][$key] = $value;
|
||
}
|
||
|
||
if ($params['auth'] ?? false) {
|
||
$authMechanism = $params['authMechanism'] ?? $params['smtpAuthMechanism'] ?? null;
|
||
|
||
if ($authMechanism) {
|
||
$authMechanism = preg_replace("([.]{2,})", '', $authMechanism);
|
||
|
||
/** @noinspection SpellCheckingInspection */
|
||
if (in_array($authMechanism, ['login', 'crammd5', 'plain'])) {
|
||
$options['connectionClass'] = $authMechanism;
|
||
} else {
|
||
$options['connectionClass'] = 'login';
|
||
}
|
||
} else {
|
||
$options['connectionClass'] = 'login';
|
||
}
|
||
|
||
$options['connectionConfig']['username'] = $params['username'];
|
||
$options['connectionConfig']['password'] = $params['password'] ?? null;
|
||
}
|
||
|
||
$authClassName = $params['authClassName'] ?? $params['smtpAuthClassName'] ?? null;
|
||
|
||
if ($authClassName) {
|
||
$options['connectionClass'] = $authClassName;
|
||
}
|
||
|
||
if ($params['security'] ?? null) {
|
||
$options['connectionConfig']['ssl'] = strtolower($params['security']);
|
||
}
|
||
|
||
if (array_key_exists('fromName', $params)) {
|
||
$this->params['fromName'] = $params['fromName'];
|
||
}
|
||
|
||
if (array_key_exists('fromAddress', $params)) {
|
||
$this->params['fromAddress'] = $params['fromAddress'];
|
||
}
|
||
|
||
$this->transport->setOptions(
|
||
new SmtpOptions($options)
|
||
);
|
||
|
||
if ($this->envelope) {
|
||
$this->transport->setEnvelope($this->envelope);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @throws NoSmtp
|
||
*/
|
||
private function applyGlobal(): void
|
||
{
|
||
$systemAccount = $this->accountProvider->getSystem();
|
||
|
||
if (!$systemAccount) {
|
||
throw new NoSmtp("No system SMTP settings.");
|
||
}
|
||
|
||
$smtpParams = $systemAccount->getSmtpParams();
|
||
|
||
if (!$smtpParams) {
|
||
throw new NoSmtp("No system SMTP settings.");
|
||
}
|
||
|
||
$this->applySmtp($smtpParams->toArray());
|
||
}
|
||
|
||
/**
|
||
* Send an email.
|
||
*
|
||
* @throws SendingError
|
||
*/
|
||
public function send(Email $email): void
|
||
{
|
||
if ($this->isGlobal) {
|
||
$this->applyGlobal();
|
||
}
|
||
|
||
$message = $this->message ?? new Message();
|
||
|
||
$params = array_merge($this->params, $this->overrideParams);
|
||
|
||
$this->applyFrom($email, $message, $params);
|
||
$this->applyReplyTo($message, $params);
|
||
$this->addRecipientAddresses($email, $message);
|
||
$this->applySubject($email, $message);
|
||
$this->applyBody($email, $message);
|
||
$this->applyMessageId($email, $message);
|
||
|
||
$message->setEncoding('UTF-8');
|
||
|
||
assert($this->transport !== null);
|
||
|
||
try {
|
||
$this->transport->send($message);
|
||
} catch (Exception $e) {
|
||
$this->resetParams();
|
||
$this->useGlobal();
|
||
|
||
$this->handleException($e);
|
||
}
|
||
|
||
$email
|
||
->setStatus(Email::STATUS_SENT)
|
||
->setDateSent(DateTime::createNow())
|
||
->setSendAt(null);
|
||
|
||
$this->resetParams();
|
||
$this->useGlobal();
|
||
}
|
||
|
||
/**
|
||
* @return MimePart[]
|
||
*/
|
||
private function getInlineAttachmentPartList(Email $email): array
|
||
{
|
||
$list = [];
|
||
|
||
foreach ($email->getInlineAttachmentList() as $attachment) {
|
||
$contents = $attachment->has(self::ATTACHMENT_ATTR_CONTENTS) ?
|
||
$attachment->get(self::ATTACHMENT_ATTR_CONTENTS) :
|
||
$this->fileStorageManager->getContents($attachment);
|
||
|
||
$mimePart = new MimePart($contents);
|
||
|
||
$mimePart->disposition = Mime::DISPOSITION_INLINE;
|
||
$mimePart->encoding = Mime::ENCODING_BASE64;
|
||
$mimePart->id = $attachment->getId();
|
||
|
||
if ($attachment->getType()) {
|
||
$mimePart->type = $attachment->getType();
|
||
}
|
||
|
||
$list[] = $mimePart;
|
||
}
|
||
|
||
return $list;
|
||
}
|
||
|
||
/**
|
||
* @throws SendingError
|
||
*/
|
||
private function handleException(Exception $e): never
|
||
{
|
||
if ($e instanceof ProtocolRuntimeException) {
|
||
$message = "unknownError";
|
||
|
||
if (
|
||
stripos($e->getMessage(), 'password') !== false ||
|
||
stripos($e->getMessage(), 'credentials') !== false ||
|
||
stripos($e->getMessage(), '5.7.8') !== false
|
||
) {
|
||
$message = 'invalidCredentials';
|
||
}
|
||
|
||
$this->log->error("Email sending error: " . $e->getMessage());
|
||
|
||
throw new SendingError($message);
|
||
}
|
||
|
||
throw new SendingError($e->getMessage());
|
||
}
|
||
|
||
/**
|
||
* @deprecated Since v9.1.0. Use EmailSender::generateMessageId.
|
||
* @noinspection PhpUnused
|
||
* @todo Remove in v10.0.
|
||
*/
|
||
static public function generateMessageId(Email $email): string
|
||
{
|
||
return EmailSender::generateMessageId($email);
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v6.0. Use withEnvelopeOptions.
|
||
*
|
||
* @param array<string, mixed> $options
|
||
* @todo Remove in v10.0.
|
||
*/
|
||
public function setEnvelopeOptions(array $options): self
|
||
{
|
||
$this->envelope = new Envelope($options);
|
||
|
||
return $this;
|
||
}
|
||
|
||
private function addRecipientAddresses(Email $email, Message $message): void
|
||
{
|
||
$value = $email->get('to');
|
||
|
||
if ($value) {
|
||
foreach (explode(';', $value) as $address) {
|
||
$message->addTo(trim($address));
|
||
}
|
||
}
|
||
|
||
$value = $email->get('cc');
|
||
|
||
if ($value) {
|
||
foreach (explode(';', $value) as $address) {
|
||
$message->addCC(trim($address));
|
||
}
|
||
}
|
||
|
||
$value = $email->get('bcc');
|
||
|
||
if ($value) {
|
||
foreach (explode(';', $value) as $address) {
|
||
$message->addBCC(trim($address));
|
||
}
|
||
}
|
||
|
||
$value = $email->get('replyTo');
|
||
|
||
if ($value) {
|
||
foreach (explode(';', $value) as $address) {
|
||
$message->addReplyTo(trim($address));
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @return MimePart[]
|
||
*/
|
||
private function getAttachmentParts(Email $email): array
|
||
{
|
||
/** @var EntityCollection<Attachment> $collection */
|
||
$collection = $this->entityManager
|
||
->getCollectionFactory()
|
||
->create(Attachment::ENTITY_TYPE);
|
||
|
||
if (!$email->isNew()) {
|
||
foreach ($email->getAttachments() as $attachment) {
|
||
$collection[] = $attachment;
|
||
}
|
||
}
|
||
|
||
if ($this->attachmentList !== null) {
|
||
foreach ($this->attachmentList as $attachment) {
|
||
$collection[] = $attachment;
|
||
}
|
||
}
|
||
|
||
$attachmentPartList = [];
|
||
|
||
foreach ($collection as $attachment) {
|
||
$contents = $attachment->has(self::ATTACHMENT_ATTR_CONTENTS) ?
|
||
$attachment->get(self::ATTACHMENT_ATTR_CONTENTS) :
|
||
$this->fileStorageManager->getContents($attachment);
|
||
|
||
$mimePart = new MimePart($contents);
|
||
|
||
$mimePart->disposition = Mime::DISPOSITION_ATTACHMENT;
|
||
$mimePart->encoding = Mime::ENCODING_BASE64;
|
||
$mimePart->filename = $this->prepareAttachmentFileName($attachment);
|
||
|
||
if ($attachment->getType()) {
|
||
$mimePart->type = $attachment->getType();
|
||
}
|
||
|
||
$attachmentPartList[] = $mimePart;
|
||
}
|
||
|
||
return $attachmentPartList;
|
||
}
|
||
|
||
private function prepareAttachmentFileName(mixed $attachment): string
|
||
{
|
||
$namePart = base64_encode($attachment->getName() ?? '');
|
||
|
||
return "=?utf-8?B?$namePart?=";
|
||
}
|
||
|
||
/**
|
||
* @param array<string, mixed> $params
|
||
* @throws NoSmtp
|
||
*/
|
||
private function applyFrom(Email $email, Message $message, array $params): void
|
||
{
|
||
$fromName = $params['fromName'] ?? $this->config->get('outboundEmailFromName');
|
||
|
||
$fromAddress = $email->get('from');
|
||
|
||
if ($fromAddress) {
|
||
$fromAddress = trim($fromAddress);
|
||
} else {
|
||
if (
|
||
empty($params['fromAddress']) &&
|
||
!$this->configDataProvider->getSystemOutboundAddress()
|
||
) {
|
||
throw new NoSmtp('outboundEmailFromAddress is not specified in config.');
|
||
}
|
||
|
||
$fromAddress = $params['fromAddress'] ?? $this->configDataProvider->getSystemOutboundAddress();
|
||
|
||
$email->setFromAddress($fromAddress);
|
||
}
|
||
|
||
$message->addFrom($fromAddress, $fromName);
|
||
|
||
$fromString = '<' . $fromAddress . '>';
|
||
|
||
if ($fromName) {
|
||
$fromString = $fromName . ' ' . $fromString;
|
||
}
|
||
|
||
$email->set('fromString', $fromString);
|
||
|
||
$senderHeader = new SenderHeader();
|
||
|
||
$senderHeader->setAddress($fromAddress);
|
||
|
||
$message->getHeaders()->addHeader($senderHeader);
|
||
}
|
||
|
||
/**
|
||
* @param array<string, mixed> $params
|
||
*/
|
||
private function applyReplyTo(Message $message, array $params): void
|
||
{
|
||
$address = $params['replyToAddress'] ?? null;
|
||
$name = $params['replyToName'] ?? null;
|
||
|
||
if (!$address) {
|
||
return;
|
||
}
|
||
|
||
$message->setReplyTo($address, $name);
|
||
}
|
||
|
||
private function applyMessageId(Email $email, Message $message): void
|
||
{
|
||
$messageId = $email->getMessageId();
|
||
|
||
if (
|
||
!$messageId ||
|
||
strlen($messageId) < 4 ||
|
||
str_starts_with($messageId, 'dummy:')
|
||
) {
|
||
$messageId = EmailSender::generateMessageId($email);
|
||
|
||
$email->setMessageId('<' . $messageId . '>');
|
||
|
||
if ($email->hasId()) {
|
||
$this->entityManager->saveEntity($email, [SaveOption::SILENT => true]);
|
||
}
|
||
} else {
|
||
$messageId = substr($messageId, 1, strlen($messageId) - 2);
|
||
}
|
||
|
||
$header = (new MessageIdHeader())->setId($messageId);
|
||
|
||
$message->getHeaders()->addHeader($header);
|
||
}
|
||
|
||
private function applyBody(Email $email, Message $message): void
|
||
{
|
||
$attachmentPartList = $this->getAttachmentParts($email);
|
||
$inlineAttachmentPartList = $this->getInlineAttachmentPartList($email);
|
||
|
||
$body = new MimeMessage();
|
||
|
||
$textPart = (new MimePart($email->getBodyPlainForSending()))
|
||
->setType(self::MESSAGE_TYPE_TEXT_PLAIN)
|
||
->setEncoding(Mime::ENCODING_QUOTEDPRINTABLE)
|
||
->setCharset('utf-8');
|
||
|
||
$htmlPart = $email->isHtml() ?
|
||
(new MimePart($email->getBodyForSending()))
|
||
->setEncoding(Mime::ENCODING_QUOTEDPRINTABLE)
|
||
->setType(self::MESSAGE_TYPE_TEXT_HTML)
|
||
->setCharset('utf-8') :
|
||
null;
|
||
|
||
$messageType = null;
|
||
|
||
$hasAttachments = count($attachmentPartList) !== 0;
|
||
$hasInlineAttachments = count($inlineAttachmentPartList) !== 0;
|
||
|
||
if ($hasAttachments || $hasInlineAttachments) {
|
||
if ($htmlPart) {
|
||
$messageType = self::MESSAGE_TYPE_MULTIPART_MIXED;
|
||
|
||
$alternative = (new MimeMessage())
|
||
->addPart($textPart)
|
||
->addPart($htmlPart);
|
||
|
||
$alternativePart = (new MimePart($alternative->generateMessage()))
|
||
->setType(self::MESSAGE_TYPE_MULTIPART_ALTERNATIVE)
|
||
->setBoundary($alternative->getMime()->boundary());
|
||
|
||
if ($hasInlineAttachments && $hasAttachments) {
|
||
$related = (new MimeMessage())->addPart($alternativePart);
|
||
|
||
foreach ($inlineAttachmentPartList as $attachmentPart) {
|
||
$related->addPart($attachmentPart);
|
||
}
|
||
|
||
$body->addPart(
|
||
(new MimePart($related->generateMessage()))
|
||
->setType(self::MESSAGE_TYPE_MULTIPART_RELATED)
|
||
->setBoundary($related->getMime()->boundary())
|
||
);
|
||
}
|
||
|
||
if ($hasInlineAttachments && !$hasAttachments) {
|
||
$messageType = self::MESSAGE_TYPE_MULTIPART_RELATED;
|
||
|
||
$body->addPart($alternativePart);
|
||
|
||
foreach ($inlineAttachmentPartList as $attachmentPart) {
|
||
$body->addPart($attachmentPart);
|
||
}
|
||
}
|
||
|
||
if (!$hasInlineAttachments) {
|
||
$body->addPart($alternativePart);
|
||
}
|
||
}
|
||
|
||
if (!$htmlPart) {
|
||
$messageType = self::MESSAGE_TYPE_MULTIPART_RELATED;
|
||
|
||
$body->addPart($textPart);
|
||
|
||
foreach ($inlineAttachmentPartList as $attachmentPart) {
|
||
$body->addPart($attachmentPart);
|
||
}
|
||
}
|
||
|
||
foreach ($attachmentPartList as $attachmentPart) {
|
||
$body->addPart($attachmentPart);
|
||
}
|
||
} else {
|
||
if ($email->isHtml()) {
|
||
$body->setParts([$textPart, $htmlPart]);
|
||
|
||
$messageType = self::MESSAGE_TYPE_MULTIPART_ALTERNATIVE;
|
||
} else {
|
||
$body = $email->getBodyPlainForSending();
|
||
|
||
$messageType = self::MESSAGE_TYPE_TEXT_PLAIN;
|
||
}
|
||
}
|
||
|
||
$message->setBody($body);
|
||
|
||
if ($messageType === self::MESSAGE_TYPE_TEXT_PLAIN) {
|
||
if ($message->getHeaders()->has('content-type')) {
|
||
$message->getHeaders()->removeHeader('content-type');
|
||
}
|
||
|
||
$message->getHeaders()->addHeaderLine('Content-Type', 'text/plain; charset=UTF-8');
|
||
|
||
return;
|
||
}
|
||
|
||
if (!$message->getHeaders()->has('content-type')) {
|
||
$contentTypeHeader = new ContentTypeHeader();
|
||
|
||
$message->getHeaders()->addHeader($contentTypeHeader);
|
||
}
|
||
|
||
$contentTypeHeader = $message->getHeaders()->get('content-type');
|
||
|
||
assert($contentTypeHeader instanceof ContentTypeHeader);
|
||
|
||
$contentTypeHeader->setType($messageType);
|
||
}
|
||
|
||
private function applySubject(Email $email, Message $message): void
|
||
{
|
||
$message->setSubject($email->getSubject() ?? '');
|
||
}
|
||
}
|