mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-26 19:02:53 -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;
 | |
|     }
 | |
| }
 |