<?php
namespace Webkul\UVDesk\MailboxBundle\Services;
use PhpMimeMailParser\Parser as EmailParser;
use Symfony\Component\Yaml\Yaml;
use Doctrine\ORM\EntityManagerInterface;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpFoundation\RequestStack;
use Webkul\UVDesk\CoreFrameworkBundle\Entity\User;
use Webkul\UVDesk\CoreFrameworkBundle\Entity\Ticket;
use Webkul\UVDesk\CoreFrameworkBundle\Entity\Thread;
use Webkul\UVDesk\CoreFrameworkBundle\Entity\Website;
use Webkul\UVDesk\MailboxBundle\Utils\Mailbox\Mailbox;
use Webkul\UVDesk\CoreFrameworkBundle\Utils\HTMLFilter;
use Webkul\UVDesk\CoreFrameworkBundle\Entity\SupportRole;
use Webkul\UVDesk\CoreFrameworkBundle\Utils\TokenGenerator;
use Webkul\UVDesk\MailboxBundle\Utils\MailboxConfiguration;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Webkul\UVDesk\CoreFrameworkBundle\Workflow\Events as CoreWorkflowEvents;
use Webkul\UVDesk\MailboxBundle\Utils\IMAP;
use Webkul\UVDesk\MailboxBundle\Utils\SMTP;
use Webkul\UVDesk\MailboxBundle\Utils\Imap\Configuration as ImapConfiguration;
use Webkul\UVDesk\CoreFrameworkBundle\SwiftMailer\SwiftMailer as SwiftMailerService;
use Webkul\UVDesk\MailboxBundle\Workflow\Events as MaibloxWorkflowEvents;
class MailboxService
{
const PATH_TO_CONFIG = '/config/packages/uvdesk_mailbox.yaml';
private $parser;
private $container;
private $requestStack;
private $entityManager;
private $mailboxCollection = [];
public function __construct(ContainerInterface $container, RequestStack $requestStack, EntityManagerInterface $entityManager, SwiftMailerService $swiftMailer)
{
$this->container = $container;
$this->requestStack = $requestStack;
$this->entityManager = $entityManager;
$this->swiftMailer = $swiftMailer;
}
public function getPathToConfigurationFile()
{
return $this->container->get('kernel')->getProjectDir() . self::PATH_TO_CONFIG;
}
public function createConfiguration($params)
{
$configuration = new MailboxConfigurations\MailboxConfiguration($params);
return $configuration ?? null;
}
public function parseMailboxConfigurations(bool $ignoreInvalidAttributes = false)
{
$path = $this->getPathToConfigurationFile();
if (! file_exists($path)) {
throw new \Exception("File '$path' not found.");
}
// Read configurations from package config.
$mailboxConfiguration = new MailboxConfiguration();
foreach (Yaml::parse(file_get_contents($path))['uvdesk_mailbox']['mailboxes'] ?? [] as $id => $params) {
// Swiftmailer Configuration
$swiftMailerConfigurations = $this->swiftMailer->parseSwiftMailerConfigurations() ?? null;
if (isset($params['smtp_swift_mailer_server'])) {
foreach ($swiftMailerConfigurations as $configuration) {
if ($configuration->getId() == $params['smtp_swift_mailer_server']['mailer_id']) {
$swiftMailerConfiguration = $configuration;
break;
}
}
}
// IMAP Configuration
$imapConfiguration = null;
if (! empty($params['imap_server'])) {
$imapConfiguration = IMAP\Configuration::guessTransportDefinition($params['imap_server']);
if ($imapConfiguration instanceof IMAP\Transport\AppTransportConfigurationInterface) {
$imapConfiguration
->setClient($params['imap_server']['client'])
->setUsername($params['imap_server']['username'])
;
} else if ($imapConfiguration instanceof IMAP\Transport\SimpleTransportConfigurationInterface) {
$imapConfiguration
->setUsername($params['imap_server']['username'])
;
} else {
$imapConfiguration
->setUsername($params['imap_server']['username'])
->setPassword($params['imap_server']['password'])
;
}
}
// SMTP Configuration
$smtpConfiguration = null;
if (
! empty($params['smtp_server'])
&& !isset($params['smtp_server']['mailer_id'])
) {
$smtpConfiguration = SMTP\Configuration::guessTransportDefinition($params['smtp_server']);
if ($smtpConfiguration instanceof SMTP\Transport\AppTransportConfigurationInterface) {
$smtpConfiguration
->setClient($params['smtp_server']['client'])
->setUsername($params['smtp_server']['username'])
;
} else if ($smtpConfiguration instanceof SMTP\Transport\ResolvedTransportConfigurationInterface) {
$smtpConfiguration
->setUsername($params['smtp_server']['username'])
->setPassword($params['smtp_server']['password'])
;
} else {
$smtpConfiguration
->setHost($params['smtp_server']['host'])
->setPort($params['smtp_server']['port'])
->setUsername($params['smtp_server']['username'])
->setPassword($params['smtp_server']['password'])
;
if (! empty($params['smtp_server']['sender_address'])) {
$smtpConfiguration
->setSenderAddress($params['smtp_server']['sender_address'])
;
}
}
}
// Mailbox Configuration
($mailbox = new Mailbox($id))
->setName($params['name'])
->setIsEnabled($params['enabled']);
if (! empty($imapConfiguration)) {
$mailbox
->setImapConfiguration($imapConfiguration)
;
}
if (! empty($smtpConfiguration)) {
$mailbox
->setSmtpConfiguration($smtpConfiguration)
;
}
if (! empty($swiftMailerConfiguration)) {
$mailbox->setSwiftMailerConfiguration($swiftMailerConfiguration);
} else if (! empty($params['smtp_server']['mailer_id']) && true === $ignoreInvalidAttributes) {
$mailbox->setSwiftMailerConfiguration($swiftmailerService->createConfiguration('smtp', $params['smtp_server']['mailer_id']));
}
$mailboxConfiguration->addMailbox($mailbox);
}
return $mailboxConfiguration;
}
private function getParser()
{
if (empty($this->parser)) {
$this->parser = new EmailParser();
}
return $this->parser;
}
private function getLoadedEmailContentParser($emailContents = null, $cacheContent = true): ?EmailParser
{
if (empty($emailContents)) {
return $this->emailParser ?? null;
}
$emailParser = new EmailParser();
$emailParser
->setText($emailContents)
;
if ($cacheContent) {
$this->emailParser = $emailParser;
}
return $emailParser;
}
private function getRegisteredMailboxes()
{
if (empty($this->mailboxCollection)) {
$this->mailboxCollection = array_map(function ($mailboxId) {
return $this->container->getParameter("uvdesk.mailboxes.$mailboxId");
}, $this->container->getParameter('uvdesk.mailboxes'));
}
return $this->mailboxCollection;
}
public function getRegisteredMailboxesById()
{
// Fetch existing content in file
$filePath = $this->getPathToConfigurationFile();
$file_content = file_get_contents($filePath);
// Convert yaml file content into array and merge existing mailbox and new mailbox
$file_content_array = Yaml::parse($file_content, 6);
if ($file_content_array['uvdesk_mailbox']['mailboxes']) {
foreach ($file_content_array['uvdesk_mailbox']['mailboxes'] as $key => $value) {
$value['mailbox_id'] = $key;
$mailboxCollection[] = $value;
}
}
return $mailboxCollection ?? [];
}
public function getEmailAddresses($collection)
{
$formattedCollection = array_map(function ($emailAddress) {
if (filter_var($emailAddress['address'], FILTER_VALIDATE_EMAIL)) {
return $emailAddress['address'];
}
return null;
}, (array) $collection);
$filteredCollection = array_values(array_filter($formattedCollection));
return count($filteredCollection) == 1 ? $filteredCollection[0] : $filteredCollection;
}
public function parseAddress($type)
{
$addresses = mailparse_rfc822_parse_addresses($this->getParser()->getHeader($type));
return $addresses ?: false;
}
public function getEmailAddress($addresses)
{
foreach ((array) $addresses as $address) {
if (filter_var($address['address'], FILTER_VALIDATE_EMAIL)) {
return $address['address'];
}
}
return null;
}
public function getMailboxByEmail($email)
{
foreach ($this->getRegisteredMailboxes() as $registeredMailbox) {
if (strtolower($email) === strtolower($registeredMailbox['imap_server']['username'])) {
return $registeredMailbox;
}
}
throw new \Exception("No mailbox found for email '$email'");
}
public function getMailboxByToEmail($email)
{
foreach ($this->getRegisteredMailboxes() as $registeredMailbox) {
if (strtolower($email) === strtolower($registeredMailbox['imap_server']['username'])) {
return true;
}
}
return false;
}
private function searchTicketSubjectReference($senderEmail, $messageSubject) {
// Search Criteria: Find ticket based on subject
if (
! empty($senderEmail)
&& ! empty($messageSubject)
) {
$threadRepository = $this->entityManager->getRepository(Thread::class);
$ticket = $threadRepository->findTicketBySubject($senderEmail, $messageSubject);
if ($ticket != null) {
return $ticket;
}
}
return null;
}
private function searchExistingTickets(array $criterias = [])
{
if (empty($criterias)) {
return null;
}
$ticketRepository = $this->entityManager->getRepository(Ticket::class);
$threadRepository = $this->entityManager->getRepository(Thread::class);
foreach ($criterias as $criteria => $criteriaValue) {
if (empty($criteriaValue)) {
continue;
}
switch ($criteria) {
case 'messageId':
// Search Criteria 1: Find ticket by unique message id
$ticket = $ticketRepository->findOneByReferenceIds($criteriaValue);
if (! empty($ticket)) {
return $ticket;
} else {
$thread = $threadRepository->findOneByMessageId($criteriaValue);
if (! empty($thread)) {
return $thread->getTicket();
}
}
break;
case 'outlookConversationId':
// Search Criteria 1: Find ticket by unique message id
$ticket = $ticketRepository->findOneByOutlookConversationId($criteriaValue);
if (! empty($ticket)) {
return $ticket;
}
break;
case 'inReplyTo':
// Search Criteria 2: Find ticket based on in-reply-to reference id
$ticket = $this->entityManager->getRepository(Thread::class)->findThreadByRefrenceId($criteriaValue);
if (! empty($ticket)) {
return $ticket;
} else {
$thread = $threadRepository->findOneByMessageId($criteriaValue);
if (! empty($thread)) {
return $thread->getTicket();
}
}
break;
case 'referenceIds':
// Search Criteria 3: Find ticket based on reference id
// Break references into ind. message id collection, and iteratively
// search for existing threads for these message ids.
$referenceIds = explode(' ', $criteriaValue);
foreach ($referenceIds as $messageId) {
$thread = $threadRepository->findOneByMessageId($messageId);
if (! empty($thread)) {
return $thread->getTicket();
}
}
break;
default:
break;
}
}
return null;
}
public function processMail($rawEmail)
{
$mailData = [];
$parser = $this->getParser();
$parser->setText($rawEmail);
$from = $this->parseAddress('from') ?: $this->parseAddress('sender');
$addresses = [
'from' => $this->getEmailAddress($from),
'to' => empty($this->parseAddress('X-Forwarded-To')) ? $this->parseAddress('to') : $this->parseAddress('X-Forwarded-To'),
'cc' => $this->parseAddress('cc'),
'delivered-to' => $this->parseAddress('delivered-to'),
];
if (empty($addresses['from'])) {
return [
'message' => "No 'from' email address was found while processing contents of email.",
'content' => [],
];
} else {
if (! empty($addresses['delivered-to'])) {
$addresses['to'] = array_map(function($address) {
return $address['address'];
}, $addresses['delivered-to']);
} else if (! empty($addresses['to'])) {
$addresses['to'] = array_map(function($address) {
return $address['address'];
}, $addresses['to']);
} else if (! empty($addresses['cc'])) {
$addresses['to'] = array_map(function($address) {
return $address['address'];
}, $addresses['cc']);
}
// Skip email processing if no to-emails are specified
if (empty($addresses['to'])) {
return [
'message' => "No 'to' email addresses were found in the email.",
'content' => [
'from' => ! empty($addresses['from']) ? $addresses['from'] : null,
],
];
}
// Skip email processing if email is an auto-forwarded message to prevent infinite loop.
if ($parser->getHeader('precedence') || $parser->getHeader('x-autoreply') || $parser->getHeader('x-autorespond') || 'auto-replied' == $parser->getHeader('auto-submitted')) {
return [
'message' => "Received an auto-forwarded email which can lead to possible infinite loop of email exchanges. Skipping email from further processing.",
'content' => [
'from' => ! empty($addresses['from']) ? $addresses['from'] : null,
],
];
}
// Check for self-referencing. Skip email processing if a mailbox is configured by the sender's address.
try {
$this->getMailboxByEmail($addresses['from']);
return [
'message' => "Received a self-referencing email where the sender email address matches one of the configured mailbox address. Skipping email from further processing.",
'content' => [
'from' => !empty($addresses['from']) ? $addresses['from'] : null,
],
];
} catch (\Exception $e) {
// An exception being thrown means no mailboxes were found from the recipient's address. Continue processing.
}
}
$mailData['replyTo'] = '';
foreach ($addresses['to'] as $mailboxEmail){
if ($this->getMailboxByToEmail(strtolower($mailboxEmail))) {
$mailData['replyTo'] = $mailboxEmail;
}
}
// Process Mail - References
$addresses['to'][0] = isset($mailData['replyTo']) ? strtolower($mailData['replyTo']) : strtolower($addresses['to'][0]);
$mailData['replyTo'] = $addresses['to'];
$mailData['messageId'] = $parser->getHeader('message-id') ?: null;
$mailData['inReplyTo'] = htmlspecialchars_decode($parser->getHeader('in-reply-to'));
$mailData['referenceIds'] = htmlspecialchars_decode($parser->getHeader('references'));
$mailData['cc'] = array_filter(explode(',', $parser->getHeader('cc'))) ?: [];
$mailData['bcc'] = array_filter(explode(',', $parser->getHeader('bcc'))) ?: [];
// Process Mail - User Details
$mailData['source'] = 'email';
$mailData['createdBy'] = 'customer';
$mailData['role'] = 'ROLE_CUSTOMER';
$mailData['from'] = $addresses['from'];
$mailData['name'] = trim(current(explode('@', $from[0]['display'])));
// Process Mail - Content
try {
$htmlFilter = new HTMLFilter();
$mailData['subject'] = $parser->getHeader('subject');
$mailData['message'] = autolink($htmlFilter->addClassEmailReplyQuote($parser->getMessageBody('htmlEmbedded')));
$mailData['attachments'] = $parser->getAttachments();
} catch(\Exception $e) {
return [
'error' => true,
'message' => $e->getMessage(),
];
}
if (! $mailData['message']) {
$mailData['message'] = autolink($htmlFilter->addClassEmailReplyQuote($parser->getMessageBody('text')));
}
$website = $this->entityManager->getRepository(Website::class)->findOneByCode('knowledgebase');
if (! empty($mailData['from']) && $this->container->get('ticket.service')->isEmailBlocked($mailData['from'], $website)) {
return [
'message' => "Received email where the sender email address is present in the block list. Skipping this email from further processing.",
'content' => [
'from' => !empty($mailData['from']) ? $mailData['from'] : null,
],
];
}
// Search for any existing tickets
$ticket = $this->searchExistingTickets([
'messageId' => $mailData['messageId'],
'inReplyTo' => $mailData['inReplyTo'],
'referenceIds' => $mailData['referenceIds'],
'from' => $mailData['from'],
'subject' => $mailData['subject'],
]);
if (empty($ticket)) {
$mailData['threadType'] = 'create';
$mailData['referenceIds'] = $mailData['messageId'];
// @Todo For same subject with same customer check
// $ticketSubjectReferenceExist = $this->searchTicketSubjectReference($mailData['from'], $mailData['subject']);
// if (!empty($ticketSubjectReferenceExist)) {
// return;
// }
$thread = $this->container->get('ticket.service')->createTicket($mailData);
// Trigger ticket created event
$event = new CoreWorkflowEvents\Ticket\Create();
$event
->setTicket($thread->getTicket())
;
$this->container->get('event_dispatcher')->dispatch($event, 'uvdesk.automation.workflow.execute');
} else if (false === $ticket->getIsTrashed() && strtolower($ticket->getStatus()->getCode()) != 'spam' && !empty($mailData['inReplyTo'])) {
$mailData['threadType'] = 'reply';
$thread = $this->entityManager->getRepository(Thread::class)->findOneByMessageId($mailData['messageId']);
$ticketRef = $this->entityManager->getRepository(Ticket::class)->findById($ticket->getId());
$referenceIds = explode(' ', $ticketRef[0]->getReferenceIds());
if (!empty($thread)) {
// Thread with the same message id exists skip process.
return [
'message' => "The contents of this email has already been processed.",
'content' => [
'from' => ! empty($mailData['from']) ? $mailData['from'] : null,
'thread' => $thread->getId(),
'ticket' => $ticket->getId(),
],
];
}
if (in_array($mailData['messageId'], $referenceIds)) {
// Thread with the same message id exists skip process.
return [
'message' => "The contents of this email has already been processed.",
'content' => [
'from' => !empty($mailData['from']) ? $mailData['from'] : null,
],
];
}
if (
$ticket->getCustomer()
&& $ticket->getCustomer()->getEmail() == $mailData['from']
) {
// Reply from customer
$user = $ticket->getCustomer();
$mailData['user'] = $user;
$userDetails = $user->getCustomerInstance()->getPartialDetails();
} else if ($this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket, $mailData['from'])) {
// Reply from collaborator
$user = $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
$mailData['user'] = $user;
$mailData['createdBy'] = 'collaborator';
$userDetails = $user->getCustomerInstance()->getPartialDetails();
} else {
$user = $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
if (
! empty($user)
&& null != $user->getAgentInstance()
) {
$mailData['user'] = $user;
$mailData['createdBy'] = 'agent';
$userDetails = $user->getAgentInstance()->getPartialDetails();
} else {
// Add user as a ticket collaborator
if (empty($user)) {
// Create a new user instance with customer support role
$role = $this->entityManager->getRepository(SupportRole::class)->findOneByCode('ROLE_CUSTOMER');
$user = $this->container->get('user.service')->createUserInstance($mailData['from'], $mailData['name'], $role, [
'source' => 'email',
'active' => true
]);
}
$mailData['user'] = $user;
$userDetails = $user->getCustomerInstance()->getPartialDetails();
if (false == $this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket, $mailData['from'])) {
$ticket->addCollaborator($user);
$this->entityManager->persist($ticket);
$this->entityManager->flush();
$ticket->lastCollaborator = $user;
$event = new CoreWorkflowEvents\Ticket\Collaborator();
$event
->setTicket($ticket)
;
$this->container->get('event_dispatcher')->dispatch($event, 'uvdesk.automation.workflow.execute');
}
}
}
$mailData['fullname'] = $userDetails['name'];
$thread = $this->container->get('ticket.service')->createThread($ticket, $mailData);
if ($thread->getThreadType() == 'reply') {
if ($thread->getCreatedBy() == 'customer') {
$event = new CoreWorkflowEvents\Ticket\CustomerReply();
$event
->setTicket($ticket)
;
} else if ($thread->getCreatedBy() == 'collaborator') {
$event = new CoreWorkflowEvents\Ticket\CollaboratorReply();
$event
->setTicket($ticket)
;
} else {
$event = new CoreWorkflowEvents\Ticket\AgentReply();
$event
->setTicket($ticket)
;
}
}
// Trigger thread reply event
$this->container->get('event_dispatcher')->dispatch($event, 'uvdesk.automation.workflow.execute');
} else if (false === $ticket->getIsTrashed() && strtolower($ticket->getStatus()->getCode()) != 'spam' && empty($mailData['inReplyTo'])) {
return [
'message' => "The contents of this email has already been processed.",
'content' => [
'from' => ! empty($mailData['from']) ? $mailData['from'] : null,
'thread' => ! empty($thread) ? $thread->getId() : null,
'ticket' => ! empty($ticket) ? $ticket->getId() : null,
],
];
}
return [
'message' => "Inbound email processed successfully.",
'content' => [
'from' => ! empty($mailData['from']) ? $mailData['from'] : null,
'thread' => ! empty($thread) ? $thread->getId() : null,
'ticket' => ! empty($ticket) ? $ticket->getId() : null,
],
];
}
public function processOutlookMail(array $outlookEmail)
{
$mailData = [];
$senderName = null;
$senderAddress = null;
if (! empty($outlookEmail['from']['emailAddress']['address'])) {
$senderName = $outlookEmail['from']['emailAddress']['name'];
$senderAddress = $outlookEmail['from']['emailAddress']['address'];
} else if (! empty($outlookEmail['sender']['emailAddress']['address'])) {
$senderName = $outlookEmail['sender']['emailAddress']['name'];
$senderAddress = $outlookEmail['sender']['emailAddress']['address'];
} else {
return [
'message' => "No 'from' email address was found while processing contents of email.",
'content' => [],
];
}
$toRecipients = array_map(function ($recipient) { return $recipient['emailAddress']['address']; }, $outlookEmail['toRecipients']);
$ccRecipients = array_map(function ($recipient) { return $recipient['emailAddress']['address']; }, $outlookEmail['ccRecipients'] ?? []);
$bccRecipients = array_map(function ($recipient) { return $recipient['emailAddress']['address']; }, $outlookEmail['bccRecipients'] ?? []);
$addresses = [
'from' => $senderAddress,
'to' => $toRecipients,
'cc' => $ccRecipients,
];
// Skip email processing if no to-emails are specified
if (empty($addresses['to'])) {
return [
'message' => "No 'to' email addresses were found in the email.",
'content' => [
'from' => $senderAddress ?? null,
],
];
}
// Check for self-referencing. Skip email processing if a mailbox is configured by the sender's address.
try {
$this->getMailboxByEmail($senderAddress);
return [
'message' => "Received a self-referencing email where the sender email address matches one of the configured mailbox address. Skipping email from further processing.",
'content' => [
'from' => $senderAddress ?? null,
],
];
} catch (\Exception $e) {
// An exception being thrown means no mailboxes were found from the recipient's address. Continue processing.
}
// Process Mail - References
// $addresses['to'][0] = isset($mailData['replyTo']) ? strtolower($mailData['replyTo']) : strtolower($addresses['to'][0]);
$mailData['replyTo'] = $addresses['to'];
$mailData['messageId'] = $outlookEmail['internetMessageId'];
$mailData['outlookConversationId'] = $outlookEmail['conversationId'];
$mailData['inReplyTo'] = $outlookEmail['conversationId'];
// $mailData['inReplyTo'] = htmlspecialchars_decode($parser->getHeader('in-reply-to'));
$mailData['referenceIds'] = '';
// $mailData['referenceIds'] = htmlspecialchars_decode($parser->getHeader('references'));
$mailData['cc'] = $ccRecipients;
$mailData['bcc'] = $bccRecipients;
// Process Mail - User Details
$mailData['source'] = 'email';
$mailData['createdBy'] = 'customer';
$mailData['role'] = 'ROLE_CUSTOMER';
$mailData['from'] = $senderAddress;
$mailData['name'] = trim($senderName);
// Process Mail - Content
$htmlFilter = new HTMLFilter();
$mailData['subject'] = $outlookEmail['subject'];
$mailData['message'] = autolink($htmlFilter->addClassEmailReplyQuote($outlookEmail['body']['content']));
$mailData['attachments'] = [];
$mailData['attachmentContent'] = isset($outlookEmail['outlookAttachments']) ? $outlookEmail['outlookAttachments'] : [];
$website = $this->entityManager->getRepository(Website::class)->findOneByCode('knowledgebase');
if (
! empty($mailData['from'])
&& $this->container->get('ticket.service')->isEmailBlocked($mailData['from'], $website)
) {
return [
'message' => "Received email where the sender email address is present in the block list. Skipping this email from further processing.",
'content' => [
'from' => !empty($mailData['from']) ? $mailData['from'] : null,
],
];
}
// return [
// 'outlookConversationId' => $mailData['outlookConversationId'],
// 'message' => "No 'to' email addresses were found in the email.",
// 'content' => [
// 'outlookConversationId' => $mailData['outlookConversationId'],
// ],
// ];
// Search for any existing tickets
$ticket = $this->searchExistingTickets([
'messageId' => $mailData['messageId'],
'inReplyTo' => $mailData['inReplyTo'],
'referenceIds' => $mailData['referenceIds'],
'from' => $mailData['from'],
'subject' => $mailData['subject'],
'outlookConversationId' => $mailData['outlookConversationId'],
]);
if (empty($ticket)) {
$mailData['threadType'] = 'create';
$mailData['referenceIds'] = $mailData['messageId'];
// @Todo For same subject with same customer check
// $ticketSubjectReferenceExist = $this->searchTicketSubjectReference($mailData['from'], $mailData['subject']);
// if(!empty($ticketSubjectReferenceExist)) {
// return;
// }
$thread = $this->container->get('ticket.service')->createTicket($mailData);
// Trigger ticket created event
$event = new CoreWorkflowEvents\Ticket\Create();
$event
->setTicket($thread->getTicket())
;
$this->container->get('event_dispatcher')->dispatch($event, 'uvdesk.automation.workflow.execute');
} else if (
false === $ticket->getIsTrashed()
&& strtolower($ticket->getStatus()->getCode()) != 'spam'
&& ! empty($mailData['inReplyTo'])
) {
$mailData['threadType'] = 'reply';
$thread = $this->entityManager->getRepository(Thread::class)->findOneByMessageId($mailData['messageId']);
$ticketRef = $this->entityManager->getRepository(Ticket::class)->findById($ticket->getId());
$referenceIds = explode(' ', $ticketRef[0]->getReferenceIds());
if (! empty($thread)) {
// Thread with the same message id exists skip process.
return [
'message' => "The contents of this email has already been processed 1.",
'content' => [
'from' => ! empty($mailData['from']) ? $mailData['from'] : null,
'thread' => $thread->getId(),
'ticket' => $ticket->getId(),
],
];
}
if (in_array($mailData['messageId'], $referenceIds)) {
// Thread with the same message id exists skip process.
return [
'message' => "The contents of this email has already been processed 2.",
'content' => [
'from' => !empty($mailData['from']) ? $mailData['from'] : null,
],
];
}
if ($ticket->getCustomer() && $ticket->getCustomer()->getEmail() == $mailData['from']) {
// Reply from customer
$user = $ticket->getCustomer();
$mailData['user'] = $user;
$userDetails = $user->getCustomerInstance()->getPartialDetails();
} else if ($this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket, $mailData['from'])){
// Reply from collaborator
$user = $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
$mailData['user'] = $user;
$mailData['createdBy'] = 'collaborator';
$userDetails = $user->getCustomerInstance()->getPartialDetails();
} else {
$user = $this->entityManager->getRepository(User::class)->findOneByEmail($mailData['from']);
if (! empty($user) && null != $user->getAgentInstance()) {
$mailData['user'] = $user;
$mailData['createdBy'] = 'agent';
$userDetails = $user->getAgentInstance()->getPartialDetails();
} else {
// Add user as a ticket collaborator
if (empty($user)) {
// Create a new user instance with customer support role
$role = $this->entityManager->getRepository(SupportRole::class)->findOneByCode('ROLE_CUSTOMER');
$user = $this->container->get('user.service')->createUserInstance($mailData['from'], $mailData['name'], $role, [
'source' => 'email',
'active' => true
]);
}
$mailData['user'] = $user;
$userDetails = $user->getCustomerInstance()->getPartialDetails();
if (false == $this->entityManager->getRepository(Ticket::class)->isTicketCollaborator($ticket, $mailData['from'])) {
$ticket->addCollaborator($user);
$this->entityManager->persist($ticket);
$this->entityManager->flush();
$ticket->lastCollaborator = $user;
$event = new CoreWorkflowEvents\Ticket\Collaborator();
$event
->setTicket($ticket)
;
$this->container->get('event_dispatcher')->dispatch($event, 'uvdesk.automation.workflow.execute');
}
}
}
$mailData['fullname'] = $userDetails['name'];
$thread = $this->container->get('ticket.service')->createThread($ticket, $mailData);
if ($thread->getThreadType() == 'reply') {
if ($thread->getCreatedBy() == 'customer') {
$event = new CoreWorkflowEvents\Ticket\CustomerReply();
$event
->setTicket($ticket)
;
} else if ($thread->getCreatedBy() == 'collaborator') {
$event = new CoreWorkflowEvents\Ticket\CollaboratorReply();
$event
->setTicket($ticket)
;
} else {
$event = new CoreWorkflowEvents\Ticket\AgentReply();
$event
->setTicket($ticket)
;
}
}
// Trigger thread reply event
$this->container->get('event_dispatcher')->dispatch($event, 'uvdesk.automation.workflow.execute');
} else if (false === $ticket->getIsTrashed() && strtolower($ticket->getStatus()->getCode()) != 'spam' && empty($mailData['inReplyTo'])) {
return [
'message' => "The contents of this email has already been processed 3.",
'content' => [
'from' => ! empty($mailData['from']) ? $mailData['from'] : null,
'thread' => ! empty($thread) ? $thread->getId() : null,
'ticket' => ! empty($ticket) ? $ticket->getId() : null,
],
];
}
return [
'message' => "Inbound email processed successfully.",
'content' => [
'from' => ! empty($mailData['from']) ? $mailData['from'] : null,
'thread' => ! empty($thread) ? $thread->getId() : null,
'ticket' => ! empty($ticket) ? $ticket->getId() : null,
],
];
}
}