vendor/uvdesk/mailbox-component/Services/MailboxService.php line 43

Open in your IDE?
  1. <?php
  2. namespace Webkul\UVDesk\MailboxBundle\Services;
  3. use PhpMimeMailParser\Parser as EmailParser;
  4. use Symfony\Component\Yaml\Yaml;
  5. use Doctrine\ORM\EntityManagerInterface;
  6. use Symfony\Component\HttpFoundation\Request;
  7. use Symfony\Component\HttpFoundation\Response;
  8. use Symfony\Component\HttpFoundation\RequestStack;
  9. use Webkul\UVDesk\CoreFrameworkBundle\Entity\User;
  10. use Webkul\UVDesk\CoreFrameworkBundle\Entity\Ticket;
  11. use Webkul\UVDesk\CoreFrameworkBundle\Entity\Thread;
  12. use Webkul\UVDesk\CoreFrameworkBundle\Entity\Website;
  13. use Webkul\UVDesk\MailboxBundle\Utils\Mailbox\Mailbox;
  14. use Webkul\UVDesk\CoreFrameworkBundle\Utils\HTMLFilter;
  15. use Webkul\UVDesk\CoreFrameworkBundle\Entity\SupportRole;
  16. use Webkul\UVDesk\CoreFrameworkBundle\Utils\TokenGenerator;
  17. use Webkul\UVDesk\MailboxBundle\Utils\MailboxConfiguration;
  18. use Symfony\Component\DependencyInjection\ContainerInterface;
  19. use Webkul\UVDesk\CoreFrameworkBundle\Workflow\Events as CoreWorkflowEvents;
  20. use Webkul\UVDesk\MailboxBundle\Utils\IMAP;
  21. use Webkul\UVDesk\MailboxBundle\Utils\SMTP;
  22. use Webkul\UVDesk\MailboxBundle\Utils\Imap\Configuration as ImapConfiguration;
  23. use Webkul\UVDesk\CoreFrameworkBundle\SwiftMailer\SwiftMailer as SwiftMailerService;
  24. use Webkul\UVDesk\MailboxBundle\Workflow\Events as MaibloxWorkflowEvents;
  25. class MailboxService
  26. {
  27.     const PATH_TO_CONFIG '/config/packages/uvdesk_mailbox.yaml';
  28.     private $parser;
  29.     private $container;
  30.     private $requestStack;
  31.     private $entityManager;
  32.     private $mailboxCollection = [];
  33.     public function __construct(ContainerInterface $containerRequestStack $requestStackEntityManagerInterface $entityManagerSwiftMailerService $swiftMailer)
  34.     {
  35.         $this->container $container;
  36.         $this->requestStack $requestStack;
  37.         $this->entityManager $entityManager;
  38.         $this->swiftMailer $swiftMailer;
  39.     }
  40.     public function getPathToConfigurationFile()
  41.     {
  42.         return $this->container->get('kernel')->getProjectDir() . self::PATH_TO_CONFIG;
  43.     }
  44.     public function createConfiguration($params)
  45.     {
  46.         $configuration = new MailboxConfigurations\MailboxConfiguration($params);
  47.         return $configuration ?? null;
  48.     }
  49.     public function parseMailboxConfigurations(bool $ignoreInvalidAttributes false
  50.     {
  51.         $path $this->getPathToConfigurationFile();
  52.         if (! file_exists($path)) {
  53.             throw new \Exception("File '$path' not found.");
  54.         }
  55.         // Read configurations from package config.
  56.         $mailboxConfiguration = new MailboxConfiguration();
  57.         foreach (Yaml::parse(file_get_contents($path))['uvdesk_mailbox']['mailboxes'] ?? [] as $id => $params) {
  58.             // Swiftmailer Configuration
  59.             
  60.             $swiftMailerConfigurations $this->swiftMailer->parseSwiftMailerConfigurations() ?? null;
  61.             if (isset($params['smtp_swift_mailer_server'])) {
  62.                 foreach ($swiftMailerConfigurations as $configuration) {
  63.                     if ($configuration->getId() == $params['smtp_swift_mailer_server']['mailer_id']) {
  64.                         $swiftMailerConfiguration $configuration;
  65.                         break;
  66.                     }
  67.                 }
  68.             }
  69.             // IMAP Configuration
  70.             $imapConfiguration null;
  71.             if (! empty($params['imap_server'])) {
  72.                 $imapConfiguration IMAP\Configuration::guessTransportDefinition($params['imap_server']);
  73.     
  74.                 if ($imapConfiguration instanceof IMAP\Transport\AppTransportConfigurationInterface) {
  75.                     $imapConfiguration
  76.                         ->setClient($params['imap_server']['client'])
  77.                         ->setUsername($params['imap_server']['username'])
  78.                     ;
  79.                 } else if ($imapConfiguration instanceof IMAP\Transport\SimpleTransportConfigurationInterface) {
  80.                     $imapConfiguration
  81.                         ->setUsername($params['imap_server']['username'])
  82.                     ;
  83.                 } else {
  84.                     $imapConfiguration
  85.                         ->setUsername($params['imap_server']['username'])
  86.                         ->setPassword($params['imap_server']['password'])
  87.                     ;
  88.                 }
  89.             }
  90.             // SMTP Configuration
  91.             $smtpConfiguration null
  92.             if (
  93.                 ! empty($params['smtp_server']) 
  94.                 && !isset($params['smtp_server']['mailer_id'])
  95.             ) {
  96.                 $smtpConfiguration SMTP\Configuration::guessTransportDefinition($params['smtp_server']);
  97.     
  98.                 if ($smtpConfiguration instanceof SMTP\Transport\AppTransportConfigurationInterface) {
  99.                     $smtpConfiguration
  100.                         ->setClient($params['smtp_server']['client'])
  101.                         ->setUsername($params['smtp_server']['username'])
  102.                     ;
  103.                 } else if ($smtpConfiguration instanceof SMTP\Transport\ResolvedTransportConfigurationInterface) {
  104.                     $smtpConfiguration
  105.                         ->setUsername($params['smtp_server']['username'])
  106.                         ->setPassword($params['smtp_server']['password'])
  107.                     ;
  108.                 }  else {
  109.                     $smtpConfiguration
  110.                         ->setHost($params['smtp_server']['host'])
  111.                         ->setPort($params['smtp_server']['port'])
  112.                         ->setUsername($params['smtp_server']['username'])
  113.                         ->setPassword($params['smtp_server']['password'])
  114.                     ;
  115.                     if (! empty($params['smtp_server']['sender_address'])) {
  116.                         $smtpConfiguration
  117.                             ->setSenderAddress($params['smtp_server']['sender_address'])
  118.                         ;
  119.                     }
  120.                 }
  121.             }
  122.             // Mailbox Configuration
  123.             ($mailbox = new Mailbox($id))
  124.                 ->setName($params['name'])
  125.                 ->setIsEnabled($params['enabled']);
  126.             if (! empty($imapConfiguration)) {
  127.                 $mailbox
  128.                     ->setImapConfiguration($imapConfiguration)
  129.                 ;
  130.             }
  131.             if (! empty($smtpConfiguration)) {
  132.                 $mailbox
  133.                     ->setSmtpConfiguration($smtpConfiguration)
  134.                 ;
  135.             }
  136.             
  137.             if (! empty($swiftMailerConfiguration)) {
  138.                 $mailbox->setSwiftMailerConfiguration($swiftMailerConfiguration);
  139.             } else if (! empty($params['smtp_server']['mailer_id']) && true === $ignoreInvalidAttributes) {
  140.                 $mailbox->setSwiftMailerConfiguration($swiftmailerService->createConfiguration('smtp'$params['smtp_server']['mailer_id']));
  141.             }
  142.             $mailboxConfiguration->addMailbox($mailbox);
  143.         }
  144.         return $mailboxConfiguration;
  145.     }
  146.     private function getParser()
  147.     {
  148.         if (empty($this->parser)) {
  149.             $this->parser = new EmailParser();
  150.         }
  151.         return $this->parser;
  152.     }
  153.     private function getLoadedEmailContentParser($emailContents null$cacheContent true): ?EmailParser
  154.     {
  155.         if (empty($emailContents)) {
  156.             return $this->emailParser ?? null;
  157.         }
  158.         $emailParser = new EmailParser();
  159.         $emailParser
  160.             ->setText($emailContents)
  161.         ;
  162.         if ($cacheContent) {
  163.             $this->emailParser $emailParser;
  164.         }
  165.         return $emailParser;
  166.     }
  167.     private function getRegisteredMailboxes()
  168.     {
  169.         if (empty($this->mailboxCollection)) {
  170.             $this->mailboxCollection array_map(function ($mailboxId) {
  171.                 return $this->container->getParameter("uvdesk.mailboxes.$mailboxId");
  172.             }, $this->container->getParameter('uvdesk.mailboxes'));
  173.         }
  174.         return $this->mailboxCollection;
  175.     }
  176.     public function getRegisteredMailboxesById()
  177.     {
  178.         // Fetch existing content in file
  179.         $filePath $this->getPathToConfigurationFile();
  180.         $file_content file_get_contents($filePath);
  181.         // Convert yaml file content into array and merge existing mailbox and new mailbox
  182.         $file_content_array Yaml::parse($file_content6);
  183.         if ($file_content_array['uvdesk_mailbox']['mailboxes']) {
  184.             foreach ($file_content_array['uvdesk_mailbox']['mailboxes'] as $key => $value) {
  185.                 $value['mailbox_id'] = $key;
  186.                 $mailboxCollection[] = $value;
  187.             }
  188.         }
  189.         
  190.         return $mailboxCollection ?? [];
  191.     }
  192.     public function getEmailAddresses($collection)
  193.     {
  194.         $formattedCollection array_map(function ($emailAddress) {
  195.             if (filter_var($emailAddress['address'], FILTER_VALIDATE_EMAIL)) {
  196.                 return $emailAddress['address'];
  197.             }
  198.             return null;
  199.         }, (array) $collection);
  200.         $filteredCollection array_values(array_filter($formattedCollection));
  201.         return count($filteredCollection) == $filteredCollection[0] : $filteredCollection;
  202.     }
  203.     public function parseAddress($type)
  204.     {
  205.         $addresses mailparse_rfc822_parse_addresses($this->getParser()->getHeader($type));
  206.         return $addresses ?: false;
  207.     }
  208.     public function getEmailAddress($addresses)
  209.     {
  210.         foreach ((array) $addresses as $address) {
  211.             if (filter_var($address['address'], FILTER_VALIDATE_EMAIL)) {
  212.                 return $address['address'];
  213.             }
  214.         }
  215.         return null;
  216.     }
  217.     public function getMailboxByEmail($email)
  218.     {
  219.         foreach ($this->getRegisteredMailboxes() as $registeredMailbox) {
  220.             if (strtolower($email) === strtolower($registeredMailbox['imap_server']['username'])) {
  221.                 return $registeredMailbox;
  222.             }
  223.         }
  224.         throw new \Exception("No mailbox found for email '$email'");
  225.     }
  226.     
  227.     public function getMailboxByToEmail($email)
  228.     {
  229.         foreach ($this->getRegisteredMailboxes() as $registeredMailbox) {
  230.             if (strtolower($email) === strtolower($registeredMailbox['imap_server']['username'])) {
  231.                 return true;
  232.             }
  233.         }
  234.         return false;
  235.     }
  236.     private function searchTicketSubjectReference($senderEmail$messageSubject) {
  237.         
  238.         // Search Criteria: Find ticket based on subject
  239.         if (
  240.             ! empty($senderEmail)
  241.             && ! empty($messageSubject)
  242.         ) {
  243.             $threadRepository $this->entityManager->getRepository(Thread::class);
  244.             $ticket $threadRepository->findTicketBySubject($senderEmail$messageSubject);
  245.             if ($ticket  != null) {
  246.                 return $ticket;
  247.             }
  248.         }
  249.         return null;
  250.     }
  251.     private function searchExistingTickets(array $criterias = [])
  252.     {
  253.         if (empty($criterias)) {
  254.             return null;
  255.         }
  256.         $ticketRepository $this->entityManager->getRepository(Ticket::class);
  257.         $threadRepository $this->entityManager->getRepository(Thread::class);
  258.         foreach ($criterias as $criteria => $criteriaValue) {
  259.             if (empty($criteriaValue)) {
  260.                 continue;
  261.             }
  262.             switch ($criteria) {
  263.                 case 'messageId':
  264.                     // Search Criteria 1: Find ticket by unique message id
  265.                     $ticket $ticketRepository->findOneByReferenceIds($criteriaValue);
  266.                     if (! empty($ticket)) {
  267.                         return $ticket;
  268.                     } else {
  269.                         $thread $threadRepository->findOneByMessageId($criteriaValue);
  270.         
  271.                         if (! empty($thread)) {
  272.                             return $thread->getTicket();
  273.                         }
  274.                     }
  275.                     
  276.                     break;
  277.                 case 'outlookConversationId':
  278.                     // Search Criteria 1: Find ticket by unique message id
  279.                     $ticket $ticketRepository->findOneByOutlookConversationId($criteriaValue);
  280.                     if (! empty($ticket)) {
  281.                         return $ticket;
  282.                     }
  283.                     
  284.                     break;
  285.                 case 'inReplyTo':
  286.                     // Search Criteria 2: Find ticket based on in-reply-to reference id
  287.                     $ticket $this->entityManager->getRepository(Thread::class)->findThreadByRefrenceId($criteriaValue);
  288.                     if (! empty($ticket)) {
  289.                         return $ticket;
  290.                     } else {
  291.                         $thread $threadRepository->findOneByMessageId($criteriaValue);
  292.         
  293.                         if (! empty($thread)) {
  294.                             return $thread->getTicket();
  295.                         }
  296.                     }
  297.                     break;
  298.                 case 'referenceIds':
  299.                     // Search Criteria 3: Find ticket based on reference id
  300.                     // Break references into ind. message id collection, and iteratively 
  301.                     // search for existing threads for these message ids.
  302.                     $referenceIds explode(' '$criteriaValue);
  303.                     foreach ($referenceIds as $messageId) {
  304.                         $thread $threadRepository->findOneByMessageId($messageId);
  305.                         if (! empty($thread)) {
  306.                             return $thread->getTicket();
  307.                         }
  308.                     }
  309.                     break;
  310.                 default:
  311.                     break;
  312.             }
  313.         }
  314.         return null;
  315.     }
  316.     
  317.     public function processMail($rawEmail)
  318.     {
  319.         $mailData = [];
  320.         $parser $this->getParser();
  321.         $parser->setText($rawEmail);
  322.         $from $this->parseAddress('from') ?: $this->parseAddress('sender');
  323.         $addresses = [
  324.             'from'         => $this->getEmailAddress($from),
  325.             'to'           => empty($this->parseAddress('X-Forwarded-To')) ? $this->parseAddress('to') : $this->parseAddress('X-Forwarded-To'),
  326.             'cc'           => $this->parseAddress('cc'),
  327.             'delivered-to' => $this->parseAddress('delivered-to'),
  328.         ];
  329.         if (empty($addresses['from'])) {
  330.             return [
  331.                 'message' => "No 'from' email address was found while processing contents of email."
  332.                 'content' => [], 
  333.             ];
  334.         } else {
  335.             if (! empty($addresses['delivered-to'])) {
  336.                 $addresses['to'] = array_map(function($address) {
  337.                     return $address['address'];
  338.                 }, $addresses['delivered-to']);
  339.             } else if (! empty($addresses['to'])) {
  340.                 $addresses['to'] = array_map(function($address) {
  341.                     return $address['address'];
  342.                 }, $addresses['to']);
  343.             } else if (! empty($addresses['cc'])) {
  344.                 $addresses['to'] = array_map(function($address) {
  345.                     return $address['address'];
  346.                 }, $addresses['cc']);
  347.             }
  348.             
  349.             // Skip email processing if no to-emails are specified
  350.             if (empty($addresses['to'])) {
  351.                 return [
  352.                     'message' => "No 'to' email addresses were found in the email."
  353.                     'content' => [
  354.                         'from' => ! empty($addresses['from']) ? $addresses['from'] : null
  355.                     ], 
  356.                 ];
  357.             }
  358.             // Skip email processing if email is an auto-forwarded message to prevent infinite loop.
  359.             if ($parser->getHeader('precedence') || $parser->getHeader('x-autoreply') || $parser->getHeader('x-autorespond') || 'auto-replied' == $parser->getHeader('auto-submitted')) {
  360.                 return [
  361.                     'message' => "Received an auto-forwarded email which can lead to possible infinite loop of email exchanges. Skipping email from further processing."
  362.                     'content' => [
  363.                         'from' => ! empty($addresses['from']) ? $addresses['from'] : null
  364.                     ], 
  365.                 ];
  366.             }
  367.             // Check for self-referencing. Skip email processing if a mailbox is configured by the sender's address.
  368.             try {
  369.                 $this->getMailboxByEmail($addresses['from']);
  370.                 return [
  371.                     'message' => "Received a self-referencing email where the sender email address matches one of the configured mailbox address. Skipping email from further processing."
  372.                     'content' => [
  373.                         'from' => !empty($addresses['from']) ? $addresses['from'] : null
  374.                     ], 
  375.                 ];
  376.             } catch (\Exception $e) {
  377.                 // An exception being thrown means no mailboxes were found from the recipient's address. Continue processing.
  378.             }
  379.         }
  380.         $mailData['replyTo'] = '';
  381.         
  382.         foreach ($addresses['to'] as $mailboxEmail){
  383.             if ($this->getMailboxByToEmail(strtolower($mailboxEmail))) {
  384.                 $mailData['replyTo'] = $mailboxEmail;
  385.             }
  386.         }
  387.         // Process Mail - References
  388.         $addresses['to'][0] = isset($mailData['replyTo']) ? strtolower($mailData['replyTo']) : strtolower($addresses['to'][0]);
  389.         $mailData['replyTo'] = $addresses['to'];
  390.         $mailData['messageId'] = $parser->getHeader('message-id') ?: null;
  391.         $mailData['inReplyTo'] = htmlspecialchars_decode($parser->getHeader('in-reply-to'));
  392.         $mailData['referenceIds'] = htmlspecialchars_decode($parser->getHeader('references'));
  393.         $mailData['cc'] = array_filter(explode(','$parser->getHeader('cc'))) ?: [];
  394.         $mailData['bcc'] = array_filter(explode(','$parser->getHeader('bcc'))) ?: [];
  395.         // Process Mail - User Details
  396.         $mailData['source'] = 'email';
  397.         $mailData['createdBy'] = 'customer';
  398.         $mailData['role'] = 'ROLE_CUSTOMER';
  399.         $mailData['from'] = $addresses['from'];
  400.         $mailData['name'] = trim(current(explode('@'$from[0]['display'])));
  401.         // Process Mail - Content
  402.         try {
  403.             $htmlFilter = new HTMLFilter();
  404.             $mailData['subject'] = $parser->getHeader('subject');
  405.             $mailData['message'] = autolink($htmlFilter->addClassEmailReplyQuote($parser->getMessageBody('htmlEmbedded')));
  406.             $mailData['attachments'] = $parser->getAttachments();
  407.         } catch(\Exception $e) {
  408.             return [
  409.                 'error'   => true,
  410.                 'message' => $e->getMessage(),
  411.             ];
  412.         }
  413.         
  414.         if (! $mailData['message']) {
  415.             $mailData['message'] = autolink($htmlFilter->addClassEmailReplyQuote($parser->getMessageBody('text')));
  416.         }
  417.         $website $this->entityManager->getRepository(Website::class)->findOneByCode('knowledgebase');
  418.         
  419.         if (! empty($mailData['from']) && $this->container->get('ticket.service')->isEmailBlocked($mailData['from'], $website)) {
  420.             return [
  421.                 'message' => "Received email where the sender email address is present in the block list. Skipping this email from further processing."
  422.                 'content' => [
  423.                     'from' => !empty($mailData['from']) ? $mailData['from'] : null
  424.                 ], 
  425.             ];
  426.         }
  427.         // Search for any existing tickets
  428.         $ticket $this->searchExistingTickets([
  429.             'messageId'    => $mailData['messageId'],
  430.             'inReplyTo'    => $mailData['inReplyTo'],
  431.             'referenceIds' => $mailData['referenceIds'],
  432.             'from'         => $mailData['from'],
  433.             'subject'      => $mailData['subject'],
  434.         ]);
  435.         if (empty($ticket)) {
  436.             $mailData['threadType'] = 'create';
  437.             $mailData['referenceIds'] = $mailData['messageId'];
  438.             // @Todo For same subject with same customer check
  439.             // $ticketSubjectReferenceExist = $this->searchTicketSubjectReference($mailData['from'], $mailData['subject']);
  440.             // if (!empty($ticketSubjectReferenceExist)) {
  441.             //     return;
  442.             // }
  443.             $thread $this->container->get('ticket.service')->createTicket($mailData);
  444.             // Trigger ticket created event
  445.             $event = new CoreWorkflowEvents\Ticket\Create();
  446.             $event
  447.                 ->setTicket($thread->getTicket())
  448.             ;
  449.             $this->container->get('event_dispatcher')->dispatch($event'uvdesk.automation.workflow.execute');
  450.         } else if (false === $ticket->getIsTrashed() && strtolower($ticket->getStatus()->getCode()) != 'spam' && !empty($mailData['inReplyTo'])) {
  451.             $mailData['threadType'] = 'reply';
  452.             $thread $this->entityManager->getRepository(Thread::class)->findOneByMessageId($mailData['messageId']);
  453.             $ticketRef $this->entityManager->getRepository(Ticket::class)->findById($ticket->getId());
  454.             $referenceIds explode(' '$ticketRef[0]->getReferenceIds());
  455.             if (!empty($thread)) {
  456.                 // Thread with the same message id exists skip process.
  457.                 return [
  458.                     'message' => "The contents of this email has already been processed."
  459.                     'content' => [
  460.                         'from'   => ! empty($mailData['from']) ? $mailData['from'] : null,
  461.                         'thread' => $thread->getId(),
  462.                         'ticket' => $ticket->getId(),
  463.                     ], 
  464.                 ];
  465.             }
  466.             if (in_array($mailData['messageId'], $referenceIds)) {
  467.                 // Thread with the same message id exists skip process.
  468.                 return [
  469.                     'message' => "The contents of this email has already been processed."
  470.                     'content' => [
  471.                         'from' => !empty($mailData['from']) ? $mailData['from'] : null
  472.                     ], 
  473.                 ];
  474.             }
  475.             if (
  476.                 $ticket->getCustomer() 
  477.                 && $ticket->getCustomer()->getEmail() == $mailData['from']
  478.             ) {
  479.                 // Reply from customer
  480.                 $user $ticket->getCustomer();
  481.                 $mailData['user'] = $user;
  482.                 $userDetails $user->getCustomerInstance()->getPartialDetails();
  483.             } else if ($this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket$mailData['from'])) {
  484.                 // Reply from collaborator
  485.                 $user $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
  486.                 $mailData['user'] = $user;
  487.                 $mailData['createdBy'] = 'collaborator';
  488.                 $userDetails $user->getCustomerInstance()->getPartialDetails();
  489.             } else {
  490.                 $user $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
  491.                 
  492.                 if (
  493.                     ! empty($user
  494.                     && null != $user->getAgentInstance()
  495.                 ) {
  496.                     $mailData['user'] = $user;
  497.                     $mailData['createdBy'] = 'agent';
  498.                     $userDetails $user->getAgentInstance()->getPartialDetails();
  499.                 } else {
  500.                     // Add user as a ticket collaborator
  501.                     if (empty($user)) {
  502.                         // Create a new user instance with customer support role
  503.                         $role $this->entityManager->getRepository(SupportRole::class)->findOneByCode('ROLE_CUSTOMER');
  504.                         $user $this->container->get('user.service')->createUserInstance($mailData['from'], $mailData['name'], $role, [
  505.                             'source' => 'email',
  506.                             'active' => true
  507.                         ]);
  508.                     }
  509.                     $mailData['user'] = $user;
  510.                     $userDetails $user->getCustomerInstance()->getPartialDetails();
  511.                     if (false == $this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket$mailData['from'])) {
  512.                         $ticket->addCollaborator($user);
  513.                         $this->entityManager->persist($ticket);
  514.                         $this->entityManager->flush();
  515.                         $ticket->lastCollaborator $user;
  516.                                
  517.                         $event = new CoreWorkflowEvents\Ticket\Collaborator();
  518.                         $event
  519.                             ->setTicket($ticket)
  520.                         ;
  521.                         $this->container->get('event_dispatcher')->dispatch($event'uvdesk.automation.workflow.execute');
  522.                     }
  523.                 }
  524.             }
  525.             $mailData['fullname'] = $userDetails['name'];
  526.             
  527.             $thread $this->container->get('ticket.service')->createThread($ticket$mailData);
  528.             
  529.             if ($thread->getThreadType() == 'reply') {
  530.                 if ($thread->getCreatedBy() == 'customer') {
  531.                     $event = new CoreWorkflowEvents\Ticket\CustomerReply();
  532.                     $event
  533.                         ->setTicket($ticket)
  534.                     ;
  535.                 }  else if ($thread->getCreatedBy() == 'collaborator') {
  536.                     $event = new CoreWorkflowEvents\Ticket\CollaboratorReply();
  537.                     $event
  538.                         ->setTicket($ticket)
  539.                     ;
  540.                 } else {
  541.                     $event = new CoreWorkflowEvents\Ticket\AgentReply();
  542.                     $event
  543.                         ->setTicket($ticket)
  544.                     ;
  545.                 }
  546.             }
  547.             // Trigger thread reply event
  548.             $this->container->get('event_dispatcher')->dispatch($event'uvdesk.automation.workflow.execute');
  549.         } else if (false === $ticket->getIsTrashed() && strtolower($ticket->getStatus()->getCode()) != 'spam' && empty($mailData['inReplyTo'])) {
  550.             return [
  551.                 'message' => "The contents of this email has already been processed."
  552.                 'content' => [
  553.                     'from'   => ! empty($mailData['from']) ? $mailData['from'] : null
  554.                     'thread' => ! empty($thread) ? $thread->getId() : null
  555.                     'ticket' => ! empty($ticket) ? $ticket->getId() : null
  556.                 ], 
  557.             ];
  558.         }
  559.         return [
  560.             'message' => "Inbound email processed successfully."
  561.             'content' => [
  562.                 'from'   => ! empty($mailData['from']) ? $mailData['from'] : null
  563.                 'thread' => ! empty($thread) ? $thread->getId() : null
  564.                 'ticket' => ! empty($ticket) ? $ticket->getId() : null
  565.             ], 
  566.         ];
  567.     }
  568.     public function processOutlookMail(array $outlookEmail)
  569.     {
  570.         $mailData = [];
  571.         $senderName null;
  572.         $senderAddress null;
  573.         if (! empty($outlookEmail['from']['emailAddress']['address'])) {
  574.             $senderName $outlookEmail['from']['emailAddress']['name'];
  575.             $senderAddress $outlookEmail['from']['emailAddress']['address'];
  576.         } else if (! empty($outlookEmail['sender']['emailAddress']['address'])) {
  577.             $senderName $outlookEmail['sender']['emailAddress']['name'];
  578.             $senderAddress $outlookEmail['sender']['emailAddress']['address'];
  579.         } else {
  580.             return [
  581.                 'message' => "No 'from' email address was found while processing contents of email."
  582.                 'content' => [], 
  583.             ];
  584.         }
  585.         $toRecipients array_map(function ($recipient) { return $recipient['emailAddress']['address']; }, $outlookEmail['toRecipients']);
  586.         $ccRecipients array_map(function ($recipient) { return $recipient['emailAddress']['address']; }, $outlookEmail['ccRecipients'] ?? []);
  587.         $bccRecipients array_map(function ($recipient) { return $recipient['emailAddress']['address']; }, $outlookEmail['bccRecipients'] ?? []);
  588.         $addresses = [
  589.             'from' => $senderAddress
  590.             'to'   => $toRecipients
  591.             'cc'   => $ccRecipients
  592.         ];
  593.         
  594.         // Skip email processing if no to-emails are specified
  595.         if (empty($addresses['to'])) {
  596.             return [
  597.                 'message' => "No 'to' email addresses were found in the email."
  598.                 'content' => [
  599.                     'from' => $senderAddress ?? null
  600.                 ], 
  601.             ];
  602.         }
  603.         // Check for self-referencing. Skip email processing if a mailbox is configured by the sender's address.
  604.         try {
  605.             $this->getMailboxByEmail($senderAddress);
  606.             return [
  607.                 'message' => "Received a self-referencing email where the sender email address matches one of the configured mailbox address. Skipping email from further processing."
  608.                 'content' => [
  609.                     'from' => $senderAddress ?? null
  610.                 ], 
  611.             ];
  612.         } catch (\Exception $e) {
  613.             // An exception being thrown means no mailboxes were found from the recipient's address. Continue processing.
  614.         }
  615.         // Process Mail - References
  616.         // $addresses['to'][0] = isset($mailData['replyTo']) ? strtolower($mailData['replyTo']) : strtolower($addresses['to'][0]);
  617.         $mailData['replyTo'] = $addresses['to'];
  618.         $mailData['messageId'] = $outlookEmail['internetMessageId'];
  619.         $mailData['outlookConversationId'] = $outlookEmail['conversationId'];
  620.         $mailData['inReplyTo'] = $outlookEmail['conversationId'];
  621.         // $mailData['inReplyTo'] = htmlspecialchars_decode($parser->getHeader('in-reply-to'));
  622.         $mailData['referenceIds'] = '';
  623.         // $mailData['referenceIds'] = htmlspecialchars_decode($parser->getHeader('references'));
  624.         $mailData['cc'] = $ccRecipients;
  625.         $mailData['bcc'] = $bccRecipients;
  626.         // Process Mail - User Details
  627.         $mailData['source'] = 'email';
  628.         $mailData['createdBy'] = 'customer';
  629.         $mailData['role'] = 'ROLE_CUSTOMER';
  630.         $mailData['from'] = $senderAddress;
  631.         $mailData['name'] = trim($senderName);
  632.         // Process Mail - Content
  633.         $htmlFilter = new HTMLFilter();
  634.         $mailData['subject'] = $outlookEmail['subject'];
  635.         $mailData['message'] = autolink($htmlFilter->addClassEmailReplyQuote($outlookEmail['body']['content']));
  636.         $mailData['attachments'] = [];
  637.         $mailData['attachmentContent'] = isset($outlookEmail['outlookAttachments']) ? $outlookEmail['outlookAttachments'] : [];
  638.         $website $this->entityManager->getRepository(Website::class)->findOneByCode('knowledgebase');
  639.         
  640.         if (
  641.             ! empty($mailData['from'])
  642.             && $this->container->get('ticket.service')->isEmailBlocked($mailData['from'], $website)
  643.         ) {
  644.             return [
  645.                 'message' => "Received email where the sender email address is present in the block list. Skipping this email from further processing."
  646.                 'content' => [
  647.                     'from' => !empty($mailData['from']) ? $mailData['from'] : null
  648.                 ], 
  649.             ];
  650.         }
  651.         // return [
  652.         //     'outlookConversationId' => $mailData['outlookConversationId'],
  653.         //     'message' => "No 'to' email addresses were found in the email.", 
  654.         //     'content' => [
  655.         //         'outlookConversationId' => $mailData['outlookConversationId'],
  656.         //     ], 
  657.         // ];
  658.         // Search for any existing tickets
  659.         $ticket $this->searchExistingTickets([
  660.             'messageId'             => $mailData['messageId'],
  661.             'inReplyTo'             => $mailData['inReplyTo'],
  662.             'referenceIds'          => $mailData['referenceIds'],
  663.             'from'                  => $mailData['from'],
  664.             'subject'               => $mailData['subject'], 
  665.             'outlookConversationId' => $mailData['outlookConversationId'],
  666.         ]);
  667.         if (empty($ticket)) {
  668.             $mailData['threadType'] = 'create';
  669.             $mailData['referenceIds'] = $mailData['messageId'];
  670.             // @Todo For same subject with same customer check
  671.             // $ticketSubjectReferenceExist = $this->searchTicketSubjectReference($mailData['from'], $mailData['subject']);
  672.             // if(!empty($ticketSubjectReferenceExist)) {
  673.             //     return;
  674.             // }
  675.             $thread $this->container->get('ticket.service')->createTicket($mailData);
  676.             // Trigger ticket created event
  677.             $event = new CoreWorkflowEvents\Ticket\Create();
  678.             $event
  679.                 ->setTicket($thread->getTicket())
  680.             ;
  681.             $this->container->get('event_dispatcher')->dispatch($event'uvdesk.automation.workflow.execute');
  682.         } else if (
  683.             false === $ticket->getIsTrashed()
  684.             && strtolower($ticket->getStatus()->getCode()) != 'spam'
  685.             && ! empty($mailData['inReplyTo'])
  686.         ) {
  687.             $mailData['threadType'] = 'reply';
  688.             $thread $this->entityManager->getRepository(Thread::class)->findOneByMessageId($mailData['messageId']);
  689.             $ticketRef $this->entityManager->getRepository(Ticket::class)->findById($ticket->getId());
  690.             $referenceIds explode(' '$ticketRef[0]->getReferenceIds());
  691.             if (! empty($thread)) {
  692.                 // Thread with the same message id exists skip process.
  693.                 return [
  694.                     'message' => "The contents of this email has already been processed 1."
  695.                     'content' => [
  696.                         'from'   => ! empty($mailData['from']) ? $mailData['from'] : null
  697.                         'thread' => $thread->getId(), 
  698.                         'ticket' => $ticket->getId(), 
  699.                     ], 
  700.                 ];
  701.             }
  702.             if (in_array($mailData['messageId'], $referenceIds)) {
  703.                 // Thread with the same message id exists skip process.
  704.                 return [
  705.                     'message' => "The contents of this email has already been processed 2."
  706.                     'content' => [
  707.                         'from' => !empty($mailData['from']) ? $mailData['from'] : null
  708.                     ], 
  709.                 ];
  710.             }
  711.             if ($ticket->getCustomer() && $ticket->getCustomer()->getEmail() == $mailData['from']) {
  712.                 // Reply from customer
  713.                 $user $ticket->getCustomer();
  714.                 $mailData['user'] = $user;
  715.                 $userDetails $user->getCustomerInstance()->getPartialDetails();
  716.             } else if ($this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket$mailData['from'])){
  717.                 // Reply from collaborator
  718.                 $user $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
  719.                 $mailData['user'] = $user;
  720.                 $mailData['createdBy'] = 'collaborator';
  721.                 $userDetails $user->getCustomerInstance()->getPartialDetails();
  722.             } else {
  723.                 $user $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
  724.                 if (! empty($user) && null != $user->getAgentInstance()) {
  725.                     $mailData['user'] = $user;
  726.                     $mailData['createdBy'] = 'agent';
  727.                     $userDetails $user->getAgentInstance()->getPartialDetails();
  728.                 } else {
  729.                     // Add user as a ticket collaborator
  730.                     if (empty($user)) {
  731.                         // Create a new user instance with customer support role
  732.                         $role $this->entityManager->getRepository(SupportRole::class)->findOneByCode('ROLE_CUSTOMER');
  733.                         $user $this->container->get('user.service')->createUserInstance($mailData['from'], $mailData['name'], $role, [
  734.                             'source' => 'email',
  735.                             'active' => true
  736.                         ]);
  737.                     }
  738.                     $mailData['user'] = $user;
  739.                     $userDetails $user->getCustomerInstance()->getPartialDetails();
  740.                     if (false == $this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket$mailData['from'])) {
  741.                         $ticket->addCollaborator($user);
  742.                         $this->entityManager->persist($ticket);
  743.                         $this->entityManager->flush();
  744.                         $ticket->lastCollaborator $user;
  745.                         
  746.                         $event = new CoreWorkflowEvents\Ticket\Collaborator();
  747.                         $event
  748.                             ->setTicket($ticket)
  749.                         ;
  750.                         $this->container->get('event_dispatcher')->dispatch($event'uvdesk.automation.workflow.execute');
  751.                     }
  752.                 }
  753.             }
  754.             $mailData['fullname'] = $userDetails['name'];
  755.             
  756.             $thread $this->container->get('ticket.service')->createThread($ticket$mailData);
  757.             
  758.             if ($thread->getThreadType() == 'reply') {
  759.                 if ($thread->getCreatedBy() == 'customer') {
  760.                     $event = new CoreWorkflowEvents\Ticket\CustomerReply();
  761.                     $event
  762.                         ->setTicket($ticket)
  763.                     ;
  764.                 }  else if ($thread->getCreatedBy() == 'collaborator') {
  765.                     $event = new CoreWorkflowEvents\Ticket\CollaboratorReply();
  766.                     $event
  767.                         ->setTicket($ticket)
  768.                     ;
  769.                 } else {
  770.                     $event = new CoreWorkflowEvents\Ticket\AgentReply();
  771.                     $event
  772.                         ->setTicket($ticket)
  773.                     ;
  774.                 }
  775.             }
  776.             // Trigger thread reply event
  777.             $this->container->get('event_dispatcher')->dispatch($event'uvdesk.automation.workflow.execute');
  778.         } else if (false === $ticket->getIsTrashed() && strtolower($ticket->getStatus()->getCode()) != 'spam' && empty($mailData['inReplyTo'])) {
  779.             return [
  780.                 'message' => "The contents of this email has already been processed 3."
  781.                 'content' => [
  782.                     'from'   => ! empty($mailData['from']) ? $mailData['from'] : null
  783.                     'thread' => ! empty($thread) ? $thread->getId() : null
  784.                     'ticket' => ! empty($ticket) ? $ticket->getId() : null
  785.                 ], 
  786.             ];
  787.         }
  788.         return [
  789.             'message' => "Inbound email processed successfully."
  790.             'content' => [
  791.                 'from'   => ! empty($mailData['from']) ? $mailData['from'] : null
  792.                 'thread' => ! empty($thread) ? $thread->getId() : null
  793.                 'ticket' => ! empty($ticket) ? $ticket->getId() : null
  794.             ],
  795.         ];
  796.     }
  797. }