2022-01-16 04:11:48 +01:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
|
|
*
|
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
|
|
*
|
2023-01-28 23:21:40 +01:00
|
|
|
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
|
2022-01-16 04:11:48 +01:00
|
|
|
*
|
|
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Jobs\PostMark;
|
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
use App\Models\SystemLog;
|
2022-01-16 04:11:48 +01:00
|
|
|
use App\Libraries\MultiDB;
|
2023-08-22 12:42:01 +02:00
|
|
|
use Postmark\PostmarkClient;
|
|
|
|
use Illuminate\Bus\Queueable;
|
|
|
|
use App\Jobs\Util\SystemLogger;
|
|
|
|
use App\Models\QuoteInvitation;
|
2022-01-16 04:11:48 +01:00
|
|
|
use App\Models\CreditInvitation;
|
|
|
|
use App\Models\InvoiceInvitation;
|
2023-08-22 12:42:01 +02:00
|
|
|
use Illuminate\Queue\SerializesModels;
|
|
|
|
use Turbo124\Beacon\Facades\LightLogs;
|
2022-08-17 09:18:30 +02:00
|
|
|
use App\Models\PurchaseOrderInvitation;
|
2023-08-22 12:42:01 +02:00
|
|
|
use Illuminate\Queue\InteractsWithQueue;
|
2022-01-16 04:11:48 +01:00
|
|
|
use App\Models\RecurringInvoiceInvitation;
|
|
|
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
|
|
|
use Illuminate\Foundation\Bus\Dispatchable;
|
2023-08-22 12:42:01 +02:00
|
|
|
use App\DataMapper\Analytics\Mail\EmailSpam;
|
|
|
|
use App\DataMapper\Analytics\Mail\EmailBounce;
|
|
|
|
use App\Notifications\Ninja\EmailSpamNotification;
|
2022-01-16 04:11:48 +01:00
|
|
|
|
|
|
|
class ProcessPostmarkWebhook implements ShouldQueue
|
|
|
|
{
|
|
|
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
|
|
|
|
|
|
|
public $tries = 1;
|
|
|
|
|
|
|
|
public $invitation;
|
2023-08-07 07:30:34 +02:00
|
|
|
|
2023-08-23 06:36:55 +02:00
|
|
|
private $entity;
|
|
|
|
|
|
|
|
private array $default_response = [
|
|
|
|
'recipients' => '',
|
|
|
|
'subject' => 'Message not found.',
|
|
|
|
'entity' => '',
|
|
|
|
'entity_id' => '',
|
|
|
|
'events' => [],
|
|
|
|
];
|
2023-08-22 12:42:01 +02:00
|
|
|
|
2022-01-16 04:11:48 +01:00
|
|
|
/**
|
|
|
|
* Create a new job instance.
|
|
|
|
*
|
|
|
|
*/
|
2023-08-07 07:30:34 +02:00
|
|
|
public function __construct(private array $request)
|
2022-01-16 04:11:48 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the job.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function handle()
|
|
|
|
{
|
|
|
|
MultiDB::findAndSetDbByCompanyKey($this->request['Tag']);
|
|
|
|
|
|
|
|
$this->invitation = $this->discoverInvitation($this->request['MessageID']);
|
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
if (!$this->invitation) {
|
2022-01-16 04:11:48 +01:00
|
|
|
return;
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2022-01-16 04:11:48 +01:00
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
if (array_key_exists('Details', $this->request)) {
|
2022-02-13 11:06:40 +01:00
|
|
|
$this->invitation->email_error = $this->request['Details'];
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2022-01-16 04:11:48 +01:00
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
switch ($this->request['RecordType']) {
|
2022-01-16 04:11:48 +01:00
|
|
|
case 'Delivery':
|
|
|
|
return $this->processDelivery();
|
|
|
|
case 'Bounce':
|
|
|
|
return $this->processBounce();
|
|
|
|
case 'SpamComplaint':
|
|
|
|
return $this->processSpamComplaint();
|
2022-02-07 09:14:54 +01:00
|
|
|
case 'Open':
|
|
|
|
return $this->processOpen();
|
2022-01-16 04:11:48 +01:00
|
|
|
default:
|
|
|
|
# code...
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-07 09:14:54 +01:00
|
|
|
// {
|
|
|
|
// "Metadata": {
|
|
|
|
// "example": "value",
|
|
|
|
// "example_2": "value"
|
|
|
|
// },
|
|
|
|
// "RecordType": "Open",
|
|
|
|
// "FirstOpen": true,
|
|
|
|
// "Client": {
|
|
|
|
// "Name": "Chrome 35.0.1916.153",
|
|
|
|
// "Company": "Google",
|
|
|
|
// "Family": "Chrome"
|
|
|
|
// },
|
|
|
|
// "OS": {
|
|
|
|
// "Name": "OS X 10.7 Lion",
|
|
|
|
// "Company": "Apple Computer, Inc.",
|
|
|
|
// "Family": "OS X 10"
|
|
|
|
// },
|
|
|
|
// "Platform": "WebMail",
|
|
|
|
// "UserAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_7_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/35.0.1916.153 Safari/537.36",
|
|
|
|
// "ReadSeconds": 5,
|
|
|
|
// "Geo": {
|
|
|
|
// "CountryISOCode": "RS",
|
|
|
|
// "Country": "Serbia",
|
|
|
|
// "RegionISOCode": "VO",
|
|
|
|
// "Region": "Autonomna Pokrajina Vojvodina",
|
|
|
|
// "City": "Novi Sad",
|
|
|
|
// "Zip": "21000",
|
|
|
|
// "Coords": "45.2517,19.8369",
|
|
|
|
// "IP": "188.2.95.4"
|
|
|
|
// },
|
|
|
|
// "MessageID": "00000000-0000-0000-0000-000000000000",
|
|
|
|
// "MessageStream": "outbound",
|
|
|
|
// "ReceivedAt": "2022-02-06T06:37:48Z",
|
|
|
|
// "Tag": "welcome-email",
|
|
|
|
// "Recipient": "john@example.com"
|
2023-02-16 02:36:09 +01:00
|
|
|
// }
|
2022-02-07 09:14:54 +01:00
|
|
|
|
|
|
|
private function processOpen()
|
|
|
|
{
|
|
|
|
$this->invitation->opened_date = now();
|
|
|
|
$this->invitation->save();
|
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
$data = array_merge($this->request, ['history' => $this->fetchMessage()]);
|
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
(new SystemLogger(
|
2023-08-22 12:42:01 +02:00
|
|
|
$data,
|
2023-02-16 02:36:09 +01:00
|
|
|
SystemLog::CATEGORY_MAIL,
|
|
|
|
SystemLog::EVENT_MAIL_OPENED,
|
|
|
|
SystemLog::TYPE_WEBHOOK_RESPONSE,
|
2022-02-07 09:14:54 +01:00
|
|
|
$this->invitation->contact->client,
|
|
|
|
$this->invitation->company
|
2022-12-12 05:27:22 +01:00
|
|
|
))->handle();
|
2022-02-07 09:14:54 +01:00
|
|
|
}
|
|
|
|
|
2022-01-16 04:11:48 +01:00
|
|
|
// {
|
|
|
|
// "RecordType": "Delivery",
|
|
|
|
// "ServerID": 23,
|
|
|
|
// "MessageStream": "outbound",
|
|
|
|
// "MessageID": "00000000-0000-0000-0000-000000000000",
|
|
|
|
// "Recipient": "john@example.com",
|
|
|
|
// "Tag": "welcome-email",
|
|
|
|
// "DeliveredAt": "2021-02-21T16:34:52Z",
|
|
|
|
// "Details": "Test delivery webhook details",
|
|
|
|
// "Metadata": {
|
|
|
|
// "example": "value",
|
|
|
|
// "example_2": "value"
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
private function processDelivery()
|
|
|
|
{
|
|
|
|
$this->invitation->email_status = 'delivered';
|
|
|
|
$this->invitation->save();
|
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
$data = array_merge($this->request, ['history' => $this->fetchMessage()]);
|
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
(new SystemLogger(
|
2023-08-22 12:42:01 +02:00
|
|
|
$data,
|
2023-02-16 02:36:09 +01:00
|
|
|
SystemLog::CATEGORY_MAIL,
|
|
|
|
SystemLog::EVENT_MAIL_DELIVERY,
|
|
|
|
SystemLog::TYPE_WEBHOOK_RESPONSE,
|
2022-01-16 04:11:48 +01:00
|
|
|
$this->invitation->contact->client,
|
|
|
|
$this->invitation->company
|
2022-12-12 05:27:22 +01:00
|
|
|
))->handle();
|
2022-01-16 04:11:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// {
|
|
|
|
// "Metadata": {
|
|
|
|
// "example": "value",
|
|
|
|
// "example_2": "value"
|
|
|
|
// },
|
|
|
|
// "RecordType": "Bounce",
|
|
|
|
// "ID": 42,
|
|
|
|
// "Type": "HardBounce",
|
|
|
|
// "TypeCode": 1,
|
|
|
|
// "Name": "Hard bounce",
|
|
|
|
// "Tag": "Test",
|
|
|
|
// "MessageID": "00000000-0000-0000-0000-000000000000",
|
|
|
|
// "ServerID": 1234,
|
|
|
|
// "MessageStream": "outbound",
|
|
|
|
// "Description": "The server was unable to deliver your message (ex: unknown user, mailbox not found).",
|
|
|
|
// "Details": "Test bounce details",
|
|
|
|
// "Email": "john@example.com",
|
|
|
|
// "From": "sender@example.com",
|
|
|
|
// "BouncedAt": "2021-02-21T16:34:52Z",
|
|
|
|
// "DumpAvailable": true,
|
|
|
|
// "Inactive": true,
|
|
|
|
// "CanActivate": true,
|
|
|
|
// "Subject": "Test subject",
|
|
|
|
// "Content": "Test content"
|
|
|
|
// }
|
|
|
|
|
|
|
|
private function processBounce()
|
|
|
|
{
|
|
|
|
$this->invitation->email_status = 'bounced';
|
|
|
|
$this->invitation->save();
|
|
|
|
|
|
|
|
$bounce = new EmailBounce(
|
|
|
|
$this->request['Tag'],
|
|
|
|
$this->request['From'],
|
|
|
|
$this->request['MessageID']
|
|
|
|
);
|
|
|
|
|
2022-10-28 05:37:01 +02:00
|
|
|
LightLogs::create($bounce)->send();
|
2022-01-16 04:11:48 +01:00
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
$data = array_merge($this->request, ['history' => $this->fetchMessage()]);
|
|
|
|
|
|
|
|
(new SystemLogger($data, SystemLog::CATEGORY_MAIL, SystemLog::EVENT_MAIL_BOUNCED, SystemLog::TYPE_WEBHOOK_RESPONSE, $this->invitation->contact->client, $this->invitation->company))->handle();
|
2022-01-16 04:11:48 +01:00
|
|
|
|
2022-07-11 03:21:01 +02:00
|
|
|
// if(config('ninja.notification.slack'))
|
2023-02-16 02:36:09 +01:00
|
|
|
// $this->invitation->company->notification(new EmailBounceNotification($this->invitation->company->account))->ninja();
|
2022-01-16 04:11:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// {
|
|
|
|
// "Metadata": {
|
|
|
|
// "example": "value",
|
|
|
|
// "example_2": "value"
|
|
|
|
// },
|
|
|
|
// "RecordType": "SpamComplaint",
|
|
|
|
// "ID": 42,
|
|
|
|
// "Type": "SpamComplaint",
|
|
|
|
// "TypeCode": 100001,
|
|
|
|
// "Name": "Spam complaint",
|
|
|
|
// "Tag": "Test",
|
|
|
|
// "MessageID": "00000000-0000-0000-0000-000000000000",
|
|
|
|
// "ServerID": 1234,
|
|
|
|
// "MessageStream": "outbound",
|
|
|
|
// "Description": "The subscriber explicitly marked this message as spam.",
|
|
|
|
// "Details": "Test spam complaint details",
|
|
|
|
// "Email": "john@example.com",
|
|
|
|
// "From": "sender@example.com",
|
|
|
|
// "BouncedAt": "2021-02-21T16:34:52Z",
|
|
|
|
// "DumpAvailable": true,
|
|
|
|
// "Inactive": true,
|
|
|
|
// "CanActivate": false,
|
|
|
|
// "Subject": "Test subject",
|
|
|
|
// "Content": "Test content"
|
|
|
|
// }
|
|
|
|
private function processSpamComplaint()
|
|
|
|
{
|
|
|
|
$this->invitation->email_status = 'spam';
|
|
|
|
$this->invitation->save();
|
|
|
|
|
|
|
|
$spam = new EmailSpam(
|
|
|
|
$this->request['Tag'],
|
|
|
|
$this->request['From'],
|
|
|
|
$this->request['MessageID']
|
|
|
|
);
|
|
|
|
|
2022-10-28 05:37:01 +02:00
|
|
|
LightLogs::create($spam)->send();
|
2022-01-16 04:11:48 +01:00
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
$data = array_merge($this->request, ['history' => $this->fetchMessage()]);
|
|
|
|
|
|
|
|
(new SystemLogger($data, SystemLog::CATEGORY_MAIL, SystemLog::EVENT_MAIL_SPAM_COMPLAINT, SystemLog::TYPE_WEBHOOK_RESPONSE, $this->invitation->contact->client, $this->invitation->company))->handle();
|
2022-01-16 04:11:48 +01:00
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
if (config('ninja.notification.slack')) {
|
2022-01-16 04:11:48 +01:00
|
|
|
$this->invitation->company->notification(new EmailSpamNotification($this->invitation->company->account))->ninja();
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2022-01-16 04:11:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private function discoverInvitation($message_id)
|
|
|
|
{
|
|
|
|
$invitation = false;
|
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
if ($invitation = InvoiceInvitation::where('message_id', $message_id)->first()) {
|
2023-08-23 06:36:55 +02:00
|
|
|
$this->entity = 'invoice';
|
2022-01-16 04:11:48 +01:00
|
|
|
return $invitation;
|
2023-02-16 02:36:09 +01:00
|
|
|
} elseif ($invitation = QuoteInvitation::where('message_id', $message_id)->first()) {
|
2023-08-23 06:36:55 +02:00
|
|
|
$this->entity = 'quote';
|
2022-01-16 04:11:48 +01:00
|
|
|
return $invitation;
|
2023-02-16 02:36:09 +01:00
|
|
|
} elseif ($invitation = RecurringInvoiceInvitation::where('message_id', $message_id)->first()) {
|
2023-08-23 06:36:55 +02:00
|
|
|
$this->entity = 'recurring_invoice';
|
2022-01-16 04:11:48 +01:00
|
|
|
return $invitation;
|
2023-02-16 02:36:09 +01:00
|
|
|
} elseif ($invitation = CreditInvitation::where('message_id', $message_id)->first()) {
|
2023-08-23 06:36:55 +02:00
|
|
|
$this->entity = 'credit';
|
2022-01-16 04:11:48 +01:00
|
|
|
return $invitation;
|
2023-02-16 02:36:09 +01:00
|
|
|
} elseif ($invitation = PurchaseOrderInvitation::where('message_id', $message_id)->first()) {
|
2023-08-23 06:36:55 +02:00
|
|
|
$this->entity = 'purchase_order';
|
2022-08-17 09:18:30 +02:00
|
|
|
return $invitation;
|
2023-02-16 02:36:09 +01:00
|
|
|
} else {
|
2022-01-16 04:11:48 +01:00
|
|
|
return $invitation;
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2022-01-16 04:11:48 +01:00
|
|
|
}
|
2023-08-22 12:42:01 +02:00
|
|
|
|
|
|
|
private function fetchMessage(): array
|
|
|
|
{
|
|
|
|
if(strlen($this->request['MessageID']) < 1){
|
|
|
|
return $this->default_response;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
2023-08-23 06:36:55 +02:00
|
|
|
$postmark = new PostmarkClient(config('services.postmark.token'));
|
2023-08-22 12:42:01 +02:00
|
|
|
$messageDetail = $postmark->getOutboundMessageDetails($this->request['MessageID']);
|
2023-08-23 06:36:55 +02:00
|
|
|
|
|
|
|
$recipients = collect($messageDetail['recipients'])->flatten()->implode(',');
|
|
|
|
$subject = $messageDetail->subject ?? '';
|
|
|
|
|
|
|
|
$events = collect($messageDetail->messageevents)->map(function ($event) {
|
|
|
|
|
|
|
|
return [
|
|
|
|
'recipient' => $event->Recipient ?? '',
|
|
|
|
'status' => $event->Type ?? '',
|
|
|
|
'delivery_message' => $event->Details->DeliveryMessage ?? $event->Details->Summary ?? '',
|
|
|
|
'server' => $event->Details->DestinationServer ?? '',
|
|
|
|
'server_ip' => $event->Details->DestinationIP ?? '',
|
|
|
|
'date' => \Carbon\Carbon::parse($event->ReceivedAt)->format('Y-m-d H:m:s') ?? '',
|
|
|
|
];
|
|
|
|
|
|
|
|
})->toArray();
|
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
return [
|
2023-08-23 06:36:55 +02:00
|
|
|
'recipients' => $recipients,
|
|
|
|
'subject' => $subject,
|
|
|
|
'entity' => $this->entity ?? '',
|
|
|
|
'entity_id' => $this->invitation->{$this->entity}->hashed_id ?? '',
|
|
|
|
'events' => $events,
|
2023-08-22 12:42:01 +02:00
|
|
|
];
|
2023-08-23 06:36:55 +02:00
|
|
|
|
2023-08-22 12:42:01 +02:00
|
|
|
}
|
|
|
|
catch (\Exception $e) {
|
|
|
|
|
|
|
|
return $this->default_response;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|