mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2025-10-24 16:59:19 -04:00
246 lines
8.0 KiB
PHP
246 lines
8.0 KiB
PHP
<?php
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace App\Http\Controllers;
|
|
|
|
use App\DataMapper\Analytics\Mail\EmailBounce;
|
|
use App\DataMapper\Analytics\Mail\EmailSpam;
|
|
use App\Jobs\Util\SystemLogger;
|
|
use App\Libraries\MultiDB;
|
|
use App\Models\CreditInvitation;
|
|
use App\Models\InvoiceInvitation;
|
|
use App\Models\QuoteInvitation;
|
|
use App\Models\RecurringInvoiceInvitation;
|
|
use App\Models\SystemLog;
|
|
use App\Notifications\Ninja\EmailBounceNotification;
|
|
use App\Notifications\Ninja\EmailSpamNotification;
|
|
use Illuminate\Http\Request;
|
|
use Turbo124\Beacon\Facades\LightLogs;
|
|
|
|
/**
|
|
* Class PostMarkController.
|
|
*/
|
|
class PostMarkController extends BaseController
|
|
{
|
|
private $invitation;
|
|
|
|
public function __construct()
|
|
{
|
|
}
|
|
|
|
/**
|
|
* Process Postmark Webhook.
|
|
*
|
|
*
|
|
* @OA\Post(
|
|
* path="/api/v1/postmark_webhook",
|
|
* operationId="postmarkWebhook",
|
|
* tags={"postmark"},
|
|
* summary="Processing webhooks from PostMark",
|
|
* description="Adds an credit to the system",
|
|
* @OA\Parameter(ref="#/components/parameters/X-Api-Secret"),
|
|
* @OA\Parameter(ref="#/components/parameters/X-Api-Token"),
|
|
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
|
|
* @OA\Parameter(ref="#/components/parameters/include"),
|
|
* @OA\Response(
|
|
* response=200,
|
|
* description="Returns the saved credit object",
|
|
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
|
|
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
|
|
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
|
|
* @OA\JsonContent(ref="#/components/schemas/Credit"),
|
|
* ),
|
|
* @OA\Response(
|
|
* response=422,
|
|
* description="Validation error",
|
|
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
|
|
*
|
|
* ),
|
|
* @OA\Response(
|
|
* response="default",
|
|
* description="Unexpected Error",
|
|
* @OA\JsonContent(ref="#/components/schemas/Error"),
|
|
* ),
|
|
* )
|
|
*/
|
|
public function webhook(Request $request)
|
|
{
|
|
|
|
if($request->header('X-API-SECURITY') && $request->header('X-API-SECURITY') == config('postmark.secret'))
|
|
{
|
|
|
|
MultiDB::findAndSetDbByCompanyKey($request->input('Tag'));
|
|
|
|
$this->invitation = $this->discoverInvitation($request->input('MessageID'));
|
|
|
|
if($this->invitation)
|
|
$this->invitation->email_error = $request->input('Details');
|
|
else
|
|
return response()->json(['message' => 'Message not found']);
|
|
|
|
switch ($request->input('RecordType'))
|
|
{
|
|
case 'Delivery':
|
|
return $this->processDelivery($request);
|
|
case 'Bounce':
|
|
return $this->processBounce($request);
|
|
case 'SpamComplaint':
|
|
return $this->processSpamComplaint($request);
|
|
default:
|
|
# code...
|
|
break;
|
|
}
|
|
|
|
return response()->json(['message' => 'Success'], 200);
|
|
|
|
}
|
|
|
|
return response()->json(['message' => 'Unauthorized'], 403);
|
|
|
|
}
|
|
|
|
// {
|
|
// "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($request)
|
|
{
|
|
$this->invitation->email_status = 'delivered';
|
|
$this->invitation->save();
|
|
|
|
SystemLogger::dispatch($request->all(),
|
|
SystemLog::CATEGORY_MAIL,
|
|
SystemLog::EVENT_MAIL_DELIVERY,
|
|
SystemLog::TYPE_WEBHOOK_RESPONSE,
|
|
$this->invitation->contact->client,
|
|
$this->invitation->company
|
|
);
|
|
}
|
|
|
|
// {
|
|
// "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($request)
|
|
{
|
|
$this->invitation->email_status = 'bounced';
|
|
$this->invitation->save();
|
|
|
|
$bounce = new EmailBounce(
|
|
$request->input('Tag'),
|
|
$request->input('From'),
|
|
$request->input('MessageID')
|
|
);
|
|
|
|
LightLogs::create($bounce)->queue();
|
|
|
|
SystemLogger::dispatch($request->all(), SystemLog::CATEGORY_MAIL, SystemLog::EVENT_MAIL_BOUNCED, SystemLog::TYPE_WEBHOOK_RESPONSE, $this->invitation->contact->client, $this->invitation->company);
|
|
|
|
if(config('ninja.notification.slack'))
|
|
$this->invitation->company->notification(new EmailBounceNotification($this->invitation->company->account))->ninja();
|
|
|
|
}
|
|
|
|
// {
|
|
// "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($request)
|
|
{
|
|
|
|
$this->invitation->email_status = 'spam';
|
|
$this->invitation->save();
|
|
|
|
$spam = new EmailSpam(
|
|
$request->input('Tag'),
|
|
$request->input('From'),
|
|
$request->input('MessageID')
|
|
);
|
|
|
|
LightLogs::create($spam)->queue();
|
|
|
|
SystemLogger::dispatch($request->all(), SystemLog::CATEGORY_MAIL, SystemLog::EVENT_MAIL_SPAM_COMPLAINT, SystemLog::TYPE_WEBHOOK_RESPONSE, $this->invitation->contact->client, $this->invitation->company);
|
|
|
|
if(config('ninja.notification.slack'))
|
|
$this->invitation->company->notification(new EmailSpamNotification($this->invitation->company->account))->ninja();
|
|
|
|
}
|
|
|
|
private function discoverInvitation($message_id)
|
|
{
|
|
$invitation = false;
|
|
|
|
if($invitation = InvoiceInvitation::whereRaw('BINARY `message_id`= ?', [$message_id])->first())
|
|
return $invitation;
|
|
elseif($invitation = QuoteInvitation::whereRaw('BINARY `message_id`= ?', [$message_id])->first())
|
|
return $invitation;
|
|
elseif($invitation = RecurringInvoiceInvitation::whereRaw('BINARY `message_id`= ?', [$message_id])->first())
|
|
return $invitation;
|
|
elseif($invitation = CreditInvitation::whereRaw('BINARY `message_id`= ?', [$message_id])->first())
|
|
return $invitation;
|
|
else
|
|
return $invitation;
|
|
}
|
|
}
|