vendor/shopware/core/Content/Flow/Dispatching/Action/SendMailAction.php line 111

Open in your IDE?
  1. <?php declare(strict_types=1);
  2. namespace Shopware\Core\Content\Flow\Dispatching\Action;
  3. use Doctrine\DBAL\Connection;
  4. use Psr\Log\LoggerInterface;
  5. use Shopware\Core\Checkout\Document\DocumentService;
  6. use Shopware\Core\Content\Flow\Events\FlowSendMailActionEvent;
  7. use Shopware\Core\Content\Mail\Service\AbstractMailService;
  8. use Shopware\Core\Content\MailTemplate\Exception\MailEventConfigurationException;
  9. use Shopware\Core\Content\MailTemplate\Exception\SalesChannelNotFoundException;
  10. use Shopware\Core\Content\MailTemplate\MailTemplateActions;
  11. use Shopware\Core\Content\MailTemplate\MailTemplateEntity;
  12. use Shopware\Core\Content\MailTemplate\Subscriber\MailSendSubscriberConfig;
  13. use Shopware\Core\Content\Media\MediaService;
  14. use Shopware\Core\Framework\Adapter\Translation\Translator;
  15. use Shopware\Core\Framework\Context;
  16. use Shopware\Core\Framework\DataAbstractionLayer\Doctrine\FetchModeHelper;
  17. use Shopware\Core\Framework\DataAbstractionLayer\EntityCollection;
  18. use Shopware\Core\Framework\DataAbstractionLayer\EntityRepositoryInterface;
  19. use Shopware\Core\Framework\DataAbstractionLayer\Exception\InconsistentCriteriaIdsException;
  20. use Shopware\Core\Framework\DataAbstractionLayer\Search\Criteria;
  21. use Shopware\Core\Framework\Event\FlowEvent;
  22. use Shopware\Core\Framework\Event\MailAware;
  23. use Shopware\Core\Framework\Event\OrderAware;
  24. use Shopware\Core\Framework\Uuid\Uuid;
  25. use Shopware\Core\Framework\Validation\DataBag\DataBag;
  26. use Shopware\Core\System\Locale\LanguageLocaleCodeProvider;
  27. use Symfony\Contracts\EventDispatcher\Event;
  28. use Symfony\Contracts\EventDispatcher\EventDispatcherInterface;
  29. class SendMailAction extends FlowAction
  30. {
  31.     public const ACTION_NAME MailTemplateActions::MAIL_TEMPLATE_MAIL_SEND_ACTION;
  32.     public const MAIL_CONFIG_EXTENSION 'mail-attachments';
  33.     private EntityRepositoryInterface $mailTemplateRepository;
  34.     private MediaService $mediaService;
  35.     private EntityRepositoryInterface $mediaRepository;
  36.     private DocumentService $documentService;
  37.     private EntityRepositoryInterface $documentRepository;
  38.     private LoggerInterface $logger;
  39.     private AbstractMailService $emailService;
  40.     private EventDispatcherInterface $eventDispatcher;
  41.     private EntityRepositoryInterface $mailTemplateTypeRepository;
  42.     private Translator $translator;
  43.     private Connection $connection;
  44.     private LanguageLocaleCodeProvider $languageLocaleProvider;
  45.     public function __construct(
  46.         AbstractMailService $emailService,
  47.         EntityRepositoryInterface $mailTemplateRepository,
  48.         MediaService $mediaService,
  49.         EntityRepositoryInterface $mediaRepository,
  50.         EntityRepositoryInterface $documentRepository,
  51.         DocumentService $documentService,
  52.         LoggerInterface $logger,
  53.         EventDispatcherInterface $eventDispatcher,
  54.         EntityRepositoryInterface $mailTemplateTypeRepository,
  55.         Translator $translator,
  56.         Connection $connection,
  57.         LanguageLocaleCodeProvider $languageLocaleProvider
  58.     ) {
  59.         $this->mailTemplateRepository $mailTemplateRepository;
  60.         $this->mediaService $mediaService;
  61.         $this->mediaRepository $mediaRepository;
  62.         $this->documentRepository $documentRepository;
  63.         $this->documentService $documentService;
  64.         $this->logger $logger;
  65.         $this->emailService $emailService;
  66.         $this->eventDispatcher $eventDispatcher;
  67.         $this->mailTemplateTypeRepository $mailTemplateTypeRepository;
  68.         $this->translator $translator;
  69.         $this->connection $connection;
  70.         $this->languageLocaleProvider $languageLocaleProvider;
  71.     }
  72.     public static function getName(): string
  73.     {
  74.         return 'action.mail.send';
  75.     }
  76.     public static function getSubscribedEvents(): array
  77.     {
  78.         return [
  79.             self::getName() => 'handle',
  80.         ];
  81.     }
  82.     public function requirements(): array
  83.     {
  84.         return [MailAware::class];
  85.     }
  86.     /**
  87.      * @throws MailEventConfigurationException
  88.      * @throws SalesChannelNotFoundException
  89.      * @throws InconsistentCriteriaIdsException
  90.      */
  91.     public function handle(Event $event): void
  92.     {
  93.         if (!$event instanceof FlowEvent) {
  94.             return;
  95.         }
  96.         $mailEvent $event->getEvent();
  97.         $extension $event->getContext()->getExtension(self::MAIL_CONFIG_EXTENSION);
  98.         if (!$extension instanceof MailSendSubscriberConfig) {
  99.             $extension = new MailSendSubscriberConfig(false, [], []);
  100.         }
  101.         if ($extension->skip()) {
  102.             return;
  103.         }
  104.         if (!$mailEvent instanceof MailAware) {
  105.             throw new MailEventConfigurationException('Not an instance of MailAware', \get_class($mailEvent));
  106.         }
  107.         $eventConfig $event->getConfig();
  108.         if (!isset($eventConfig['mailTemplateId'])) {
  109.             return;
  110.         }
  111.         $mailTemplate $this->getMailTemplate($eventConfig['mailTemplateId'], $event->getContext());
  112.         if ($mailTemplate === null) {
  113.             return;
  114.         }
  115.         $injectedTranslator $this->injectTranslator($mailEvent);
  116.         $data = new DataBag();
  117.         $recipients $mailEvent->getMailStruct()->getRecipients();
  118.         if (!empty($eventConfig['recipient'])) {
  119.             $recipients $this->getRecipients($eventConfig['recipient'], $mailEvent);
  120.         }
  121.         $data->set('recipients'$recipients);
  122.         $data->set('senderName'$mailTemplate->getTranslation('senderName'));
  123.         $data->set('salesChannelId'$mailEvent->getSalesChannelId());
  124.         $data->set('templateId'$mailTemplate->getId());
  125.         $data->set('customFields'$mailTemplate->getCustomFields());
  126.         $data->set('contentHtml'$mailTemplate->getTranslation('contentHtml'));
  127.         $data->set('contentPlain'$mailTemplate->getTranslation('contentPlain'));
  128.         $data->set('subject'$mailTemplate->getTranslation('subject'));
  129.         $data->set('mediaIds', []);
  130.         $attachments array_unique($this->buildAttachments($mailEvent$mailTemplate$extension$eventConfig), \SORT_REGULAR);
  131.         if (!empty($attachments)) {
  132.             $data->set('binAttachments'$attachments);
  133.         }
  134.         $this->eventDispatcher->dispatch(new FlowSendMailActionEvent($data$mailTemplate$event));
  135.         if ($data->has('templateId')) {
  136.             $this->updateMailTemplateType($event$mailEvent$mailTemplate);
  137.         }
  138.         try {
  139.             $this->emailService->send(
  140.                 $data->all(),
  141.                 $event->getContext(),
  142.                 $this->getTemplateData($mailEvent)
  143.             );
  144.             $writes array_map(static function ($id) {
  145.                 return ['id' => $id'sent' => true];
  146.             }, array_column($attachments'id'));
  147.             if (!empty($writes)) {
  148.                 $this->documentRepository->update($writes$event->getContext());
  149.             }
  150.         } catch (\Exception $e) {
  151.             $this->logger->error(
  152.                 "Could not send mail:\n"
  153.                 $e->getMessage() . "\n"
  154.                 'Error Code:' $e->getCode() . "\n"
  155.                 "Template data: \n"
  156.                 json_encode($data->all()) . "\n"
  157.             );
  158.         }
  159.         if ($injectedTranslator) {
  160.             $this->translator->resetInjection();
  161.         }
  162.     }
  163.     private function updateMailTemplateType(FlowEvent $eventMailAware $mailAwareMailTemplateEntity $mailTemplate): void
  164.     {
  165.         if (!$mailTemplate->getMailTemplateTypeId()) {
  166.             return;
  167.         }
  168.         $mailTemplateTypeTranslation $this->connection->fetchOne(
  169.             'SELECT 1 FROM mail_template_type_translation WHERE language_id = :languageId AND mail_template_type_id =:mailTemplateTypeId',
  170.             [
  171.                 'languageId' => Uuid::fromHexToBytes($event->getContext()->getLanguageId()),
  172.                 'mailTemplateTypeId' => Uuid::fromHexToBytes($mailTemplate->getMailTemplateTypeId()),
  173.             ]
  174.         );
  175.         if (!$mailTemplateTypeTranslation) {
  176.             // Don't throw errors if this fails // Fix with NEXT-15475
  177.             $this->logger->error(
  178.                 "Could not update mail template type, because translation for this language does not exits:\n"
  179.                 'Flow id: ' $event->getFlowState()->flowId "\n"
  180.                 'Sequence id: ' $event->getFlowState()->sequenceId
  181.             );
  182.             return;
  183.         }
  184.         $this->mailTemplateTypeRepository->update([[
  185.             'id' => $mailTemplate->getMailTemplateTypeId(),
  186.             'templateData' => $this->getTemplateData($mailAware),
  187.         ]], $mailAware->getContext());
  188.     }
  189.     private function getMailTemplate(string $idContext $context): ?MailTemplateEntity
  190.     {
  191.         $criteria = new Criteria([$id]);
  192.         $criteria->addAssociation('media.media');
  193.         $criteria->setLimit(1);
  194.         return $this->mailTemplateRepository
  195.             ->search($criteria$context)
  196.             ->first();
  197.     }
  198.     /**
  199.      * @throws MailEventConfigurationException
  200.      */
  201.     private function getTemplateData(MailAware $event): array
  202.     {
  203.         $data = [];
  204.         foreach (array_keys($event::getAvailableData()->toArray()) as $key) {
  205.             $getter 'get' ucfirst($key);
  206.             if (!method_exists($event$getter)) {
  207.                 throw new MailEventConfigurationException('Data for ' $key ' not available.', \get_class($event));
  208.             }
  209.             $data[$key] = $event->$getter();
  210.         }
  211.         return $data;
  212.     }
  213.     private function buildAttachments(MailAware $mailEventMailTemplateEntity $mailTemplateMailSendSubscriberConfig $extensions, array $eventConfig): array
  214.     {
  215.         $attachments = [];
  216.         if ($mailTemplate->getMedia() !== null) {
  217.             foreach ($mailTemplate->getMedia() as $mailTemplateMedia) {
  218.                 if ($mailTemplateMedia->getMedia() === null) {
  219.                     continue;
  220.                 }
  221.                 if ($mailTemplateMedia->getLanguageId() !== null && $mailTemplateMedia->getLanguageId() !== $mailEvent->getContext()->getLanguageId()) {
  222.                     continue;
  223.                 }
  224.                 $attachments[] = $this->mediaService->getAttachment(
  225.                     $mailTemplateMedia->getMedia(),
  226.                     $mailEvent->getContext()
  227.                 );
  228.             }
  229.         }
  230.         if (!empty($extensions->getMediaIds())) {
  231.             $entities $this->mediaRepository->search(new Criteria($extensions->getMediaIds()), $mailEvent->getContext());
  232.             foreach ($entities as $media) {
  233.                 $attachments[] = $this->mediaService->getAttachment($media$mailEvent->getContext());
  234.             }
  235.         }
  236.         if (!empty($extensions->getDocumentIds())) {
  237.             $attachments $this->buildOrderAttachments($extensions->getDocumentIds(), $attachments$mailEvent->getContext());
  238.         }
  239.         if (empty($eventConfig['documentTypeIds']) || !\is_array($eventConfig['documentTypeIds']) || !$mailEvent instanceof OrderAware) {
  240.             return $attachments;
  241.         }
  242.         return $this->buildFlowSettingAttachments($mailEvent->getOrderId(), $eventConfig['documentTypeIds'], $attachments$mailEvent->getContext());
  243.     }
  244.     private function injectTranslator(MailAware $event): bool
  245.     {
  246.         if ($event->getSalesChannelId() === null) {
  247.             return false;
  248.         }
  249.         if ($this->translator->getSnippetSetId() !== null) {
  250.             return false;
  251.         }
  252.         $this->translator->injectSettings(
  253.             $event->getSalesChannelId(),
  254.             $event->getContext()->getLanguageId(),
  255.             $this->languageLocaleProvider->getLocaleForLanguageId($event->getContext()->getLanguageId()),
  256.             $event->getContext()
  257.         );
  258.         return true;
  259.     }
  260.     private function getRecipients(array $recipientsMailAware $mailEvent): array
  261.     {
  262.         switch ($recipients['type']) {
  263.             case 'custom':
  264.                 return $recipients['data'];
  265.             case 'admin':
  266.                 $admins $this->connection->fetchAllAssociative(
  267.                     'SELECT first_name, last_name, email FROM user WHERE admin = true'
  268.                 );
  269.                 $emails = [];
  270.                 foreach ($admins as $admin) {
  271.                     $emails[$admin['email']] = $admin['first_name'] . ' ' $admin['last_name'];
  272.                 }
  273.                 return $emails;
  274.             default:
  275.                 return $mailEvent->getMailStruct()->getRecipients();
  276.         }
  277.     }
  278.     private function buildOrderAttachments(array $documentIds, array $attachmentsContext $context): array
  279.     {
  280.         $criteria = new Criteria($documentIds);
  281.         $criteria->addAssociation('documentMediaFile');
  282.         $criteria->addAssociation('documentType');
  283.         $entities $this->documentRepository->search($criteria$context);
  284.         return $this->mappingAttachmentsInfo($entities$attachments$context);
  285.     }
  286.     private function buildFlowSettingAttachments(string $orderId, array $documentTypeIds, array $attachmentsContext $context): array
  287.     {
  288.         $documents $this->connection->fetchAllAssociative(
  289.             'SELECT
  290.                 LOWER(hex(`document`.`document_type_id`)) as doc_type,
  291.                 LOWER(hex(`document`.`id`)) as doc_id,
  292.                 `document`.`created_at` as newest_date
  293.             FROM
  294.                 `document`
  295.             WHERE
  296.                 HEX(`document`.`order_id`) = :orderId
  297.                 AND HEX(`document`.`document_type_id`) IN (:documentTypeIds)
  298.             ORDER BY `document`.`created_at` DESC',
  299.             [
  300.                 'orderId' => $orderId,
  301.                 'documentTypeIds' => $documentTypeIds,
  302.             ],
  303.             [
  304.                 'documentTypeIds' => Connection::PARAM_STR_ARRAY,
  305.             ]
  306.         );
  307.         $documentsGroupByType FetchModeHelper::group($documents);
  308.         foreach ($documentsGroupByType as $document) {
  309.             $documentIds[] = array_shift($document)['doc_id'];
  310.         }
  311.         if (empty($documentIds)) {
  312.             return $attachments;
  313.         }
  314.         $criteria = new Criteria($documentIds);
  315.         $criteria->addAssociations(['documentMediaFile''documentType']);
  316.         $entities $this->documentRepository->search($criteria$context);
  317.         return $this->mappingAttachmentsInfo($entities$attachments$context);
  318.     }
  319.     private function mappingAttachmentsInfo(EntityCollection $entities, array $attachmentsContext $context): array
  320.     {
  321.         foreach ($entities as $document) {
  322.             $documentId $document->getId();
  323.             $document $this->documentService->getDocument($document$context);
  324.             $attachments[] = [
  325.                 'id' => $documentId,
  326.                 'content' => $document->getFileBlob(),
  327.                 'fileName' => $document->getFilename(),
  328.                 'mimeType' => $document->getContentType(),
  329.             ];
  330.         }
  331.         return $attachments;
  332.     }
  333. }