181 lines
6.5 KiB
PHP
181 lines
6.5 KiB
PHP
<?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\Bridge\Mailgun\Transport;
|
|
|
|
use Psr\EventDispatcher\EventDispatcherInterface;
|
|
use Psr\Log\LoggerInterface;
|
|
use Symfony\Component\Mailer\Envelope;
|
|
use Symfony\Component\Mailer\Exception\HttpTransportException;
|
|
use Symfony\Component\Mailer\Header\MetadataHeader;
|
|
use Symfony\Component\Mailer\Header\TagHeader;
|
|
use Symfony\Component\Mailer\SentMessage;
|
|
use Symfony\Component\Mailer\Transport\AbstractApiTransport;
|
|
use Symfony\Component\Mime\Email;
|
|
use Symfony\Component\Mime\Part\Multipart\FormDataPart;
|
|
use Symfony\Contracts\HttpClient\Exception\DecodingExceptionInterface;
|
|
use Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface;
|
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
|
use Symfony\Contracts\HttpClient\ResponseInterface;
|
|
|
|
/**
|
|
* @author Kevin Verschaeve
|
|
*/
|
|
class MailgunApiTransport extends AbstractApiTransport
|
|
{
|
|
private const HOST = 'api.%region_dot%mailgun.net';
|
|
|
|
private string $key;
|
|
private string $domain;
|
|
private ?string $region;
|
|
|
|
public function __construct(string $key, string $domain, string $region = null, HttpClientInterface $client = null, EventDispatcherInterface $dispatcher = null, LoggerInterface $logger = null)
|
|
{
|
|
$this->key = $key;
|
|
$this->domain = $domain;
|
|
$this->region = $region;
|
|
|
|
parent::__construct($client, $dispatcher, $logger);
|
|
}
|
|
|
|
public function __toString(): string
|
|
{
|
|
return sprintf('mailgun+api://%s?domain=%s', $this->getEndpoint(), $this->domain);
|
|
}
|
|
|
|
protected function doSendApi(SentMessage $sentMessage, Email $email, Envelope $envelope): ResponseInterface
|
|
{
|
|
$body = new FormDataPart($this->getPayload($email, $envelope));
|
|
$headers = [];
|
|
foreach ($body->getPreparedHeaders()->all() as $header) {
|
|
$headers[] = $header->toString();
|
|
}
|
|
|
|
$endpoint = sprintf('%s/v3/%s/messages', $this->getEndpoint(), urlencode($this->domain));
|
|
$response = $this->client->request('POST', 'https://'.$endpoint, [
|
|
'auth_basic' => 'api:'.$this->key,
|
|
'headers' => $headers,
|
|
'body' => $body->bodyToIterable(),
|
|
]);
|
|
|
|
try {
|
|
$statusCode = $response->getStatusCode();
|
|
$result = $response->toArray(false);
|
|
} catch (DecodingExceptionInterface) {
|
|
throw new HttpTransportException('Unable to send an email: '.$response->getContent(false).sprintf(' (code %d).', $statusCode), $response);
|
|
} catch (TransportExceptionInterface $e) {
|
|
throw new HttpTransportException('Could not reach the remote Mailgun server.', $response, 0, $e);
|
|
}
|
|
|
|
if (200 !== $statusCode) {
|
|
throw new HttpTransportException('Unable to send an email: '.$result['message'].sprintf(' (code %d).', $statusCode), $response);
|
|
}
|
|
|
|
$sentMessage->setMessageId($result['id']);
|
|
|
|
return $response;
|
|
}
|
|
|
|
private function getPayload(Email $email, Envelope $envelope): array
|
|
{
|
|
$headers = $email->getHeaders();
|
|
$html = $email->getHtmlBody();
|
|
if (null !== $html && \is_resource($html)) {
|
|
if (stream_get_meta_data($html)['seekable'] ?? false) {
|
|
rewind($html);
|
|
}
|
|
$html = stream_get_contents($html);
|
|
}
|
|
[$attachments, $inlines, $html] = $this->prepareAttachments($email, $html);
|
|
|
|
$payload = [
|
|
'from' => $envelope->getSender()->toString(),
|
|
'to' => implode(',', $this->stringifyAddresses($this->getRecipients($email, $envelope))),
|
|
'subject' => $email->getSubject(),
|
|
'attachment' => $attachments,
|
|
'inline' => $inlines,
|
|
];
|
|
if ($emails = $email->getCc()) {
|
|
$payload['cc'] = implode(',', $this->stringifyAddresses($emails));
|
|
}
|
|
if ($emails = $email->getBcc()) {
|
|
$payload['bcc'] = implode(',', $this->stringifyAddresses($emails));
|
|
}
|
|
if ($email->getTextBody()) {
|
|
$payload['text'] = $email->getTextBody();
|
|
}
|
|
if ($html) {
|
|
$payload['html'] = $html;
|
|
}
|
|
|
|
$headersToBypass = ['from', 'to', 'cc', 'bcc', 'subject', 'content-type'];
|
|
foreach ($headers->all() as $name => $header) {
|
|
if (\in_array($name, $headersToBypass, true)) {
|
|
continue;
|
|
}
|
|
|
|
if ($header instanceof TagHeader) {
|
|
$payload[] = ['o:tag' => $header->getValue()];
|
|
|
|
continue;
|
|
}
|
|
|
|
if ($header instanceof MetadataHeader) {
|
|
$payload['v:'.$header->getKey()] = $header->getValue();
|
|
|
|
continue;
|
|
}
|
|
|
|
// Check if it is a valid prefix or header name according to Mailgun API
|
|
$prefix = substr($name, 0, 2);
|
|
if (\in_array($prefix, ['h:', 't:', 'o:', 'v:']) || \in_array($name, ['recipient-variables', 'template', 'amp-html'])) {
|
|
$headerName = $header->getName();
|
|
} else {
|
|
$headerName = 'h:'.$header->getName();
|
|
}
|
|
|
|
$payload[$headerName] = $header->getBodyAsString();
|
|
}
|
|
|
|
return $payload;
|
|
}
|
|
|
|
private function prepareAttachments(Email $email, ?string $html): array
|
|
{
|
|
$attachments = $inlines = [];
|
|
foreach ($email->getAttachments() as $attachment) {
|
|
$headers = $attachment->getPreparedHeaders();
|
|
if ('inline' === $headers->getHeaderBody('Content-Disposition')) {
|
|
// replace the cid with just a file name (the only supported way by Mailgun)
|
|
if ($html) {
|
|
$filename = $headers->getHeaderParameter('Content-Disposition', 'filename');
|
|
$new = basename($filename);
|
|
$html = str_replace('cid:'.$filename, 'cid:'.$new, $html);
|
|
$p = new \ReflectionProperty($attachment, 'filename');
|
|
$p->setValue($attachment, $new);
|
|
}
|
|
$inlines[] = $attachment;
|
|
} else {
|
|
$attachments[] = $attachment;
|
|
}
|
|
}
|
|
|
|
return [$attachments, $inlines, $html];
|
|
}
|
|
|
|
private function getEndpoint(): ?string
|
|
{
|
|
$host = $this->host ?: str_replace('%region_dot%', 'us' !== ($this->region ?: 'us') ? $this->region.'.' : '', self::HOST);
|
|
|
|
return $host.($this->port ? ':'.$this->port : '');
|
|
}
|
|
}
|