mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-11-04 08:37:32 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			237 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			237 lines
		
	
	
		
			7.6 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\EmailBounce;
 | 
						|
use App\DataMapper\Analytics\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 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'))
 | 
						|
        {
 | 
						|
            // nlog($request->all());
 | 
						|
 | 
						|
            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)->batch();
 | 
						|
 | 
						|
        SystemLogger::dispatch($request->all(), SystemLog::CATEGORY_MAIL, SystemLog::EVENT_MAIL_BOUNCED, SystemLog::TYPE_WEBHOOK_RESPONSE, $this->invitation->contact->client, $this->invitation->company);
 | 
						|
    }
 | 
						|
 | 
						|
// {
 | 
						|
//   "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($bounce)->batch();
 | 
						|
 | 
						|
        SystemLogger::dispatch($request->all(), SystemLog::CATEGORY_MAIL, SystemLog::EVENT_MAIL_SPAM_COMPLAINT, SystemLog::TYPE_WEBHOOK_RESPONSE, $this->invitation->contact->client, $this->invitation->company);
 | 
						|
    }
 | 
						|
 | 
						|
    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;
 | 
						|
    }
 | 
						|
}
 |