1
+ <?php
2
+
3
+ require 'vendor/autoload.php ' ;
4
+
5
+ use WeBirr \Payment ;
6
+
7
+ class Webhook
8
+ {
9
+ /**
10
+ * Handle incoming webhook POST requests.
11
+ * Validates request method (must be POST).
12
+ * Checks authentication using the authKey from the query string, otherwise system will not know if the request is coming from WeBirr(authorized) or not
13
+ */
14
+ public function handleRequest ()
15
+ {
16
+ // Validate request method is POST
17
+ if ($ _SERVER ['REQUEST_METHOD ' ] !== 'POST ' ) {
18
+ http_response_code (405 ); // Method Not Allowed
19
+ header ('Content-Type: application/json ' );
20
+ echo json_encode (["error " => "Method Not Allowed. POST required. " ]);
21
+ return ;
22
+ }
23
+
24
+ // Authenticate using authKey query string parameter, otherwise system will not know if the request is from WeBirr(authorized) or not
25
+ if (!$ this ->isAuthenticated ()) {
26
+ http_response_code (403 );
27
+ header ('Content-Type: application/json ' );
28
+ echo json_encode (["error " => "Unauthorized access. Invalid authKey. " ]);
29
+ return ;
30
+ }
31
+
32
+ // Read the raw JSON input
33
+ $ rawPayload = file_get_contents ('php://input ' );
34
+ if (empty ($ rawPayload )) {
35
+ http_response_code (400 ); // Bad Request
36
+ header ('Content-Type: application/json ' );
37
+ echo json_encode (["error " => "Empty request body. " ]);
38
+ return ;
39
+ }
40
+
41
+ // Decode the JSON payload into an associative array
42
+ $ data = json_decode ($ rawPayload , true );
43
+ if (json_last_error () !== JSON_ERROR_NONE ) {
44
+ http_response_code (400 );
45
+ header ('Content-Type: application/json ' );
46
+ echo json_encode (["error " => "Invalid JSON format. " ]);
47
+ return ;
48
+ }
49
+
50
+ try {
51
+ $ payment = new Payment ($ data );
52
+ } catch (Exception $ e ) {
53
+ http_response_code (400 );
54
+ header ('Content-Type: application/json ' );
55
+ echo json_encode (["error " => "Invalid payment data: " . $ e ->getMessage ()]);
56
+ return ;
57
+ }
58
+
59
+ // Process the payment asynchronously (or enqueue it for later processing)
60
+ // here instead of immediate processing, this should be handled asynchronously or enqueued to a background worker/job.
61
+ $ this ->processPayment ($ payment );
62
+
63
+ // Return a JSON success response (can also be just empty body with 200 OK status)
64
+ http_response_code (200 );
65
+ header ('Content-Type: application/json ' );
66
+ echo json_encode (["success " => true , "message " => "Payment received and queued for processing " ]);
67
+ }
68
+
69
+ /**
70
+ * Authenticate the request using authKey from the query string.
71
+ *
72
+ * The authKey is compared against the environment variable `WB_WEBHOOK_AUTH_KEY`.
73
+ *
74
+ * @return bool True if authentication succeeds, false otherwise.
75
+ */
76
+ private function isAuthenticated (): bool
77
+ {
78
+ // Retrieve the authKey from the query string
79
+ $ providedAuthKey = $ _GET ['authKey ' ] ?? '' ;
80
+
81
+ // TODO: Replace this with your own auth key (preferably from environment variable)
82
+ $ expectedAuthKey = "please-change-me-to-secure-key-5114831AFD5D4646901DCDAC58B92F8E " ;
83
+ //$expectedAuthKey = getenv('wb_webhook_authkey') !== false ? getenv('wb_webhook_authkey') : "";
84
+
85
+ // Compare the provided key with the expected key
86
+ return !empty ($ expectedAuthKey ) && hash_equals ($ expectedAuthKey , $ providedAuthKey );
87
+ }
88
+
89
+ /**
90
+ * Process Payment should be impleneted as idempotent operation for production use cases
91
+ * Prefered approach is: Payment should be processed asynchronously or enqueued to a background worker.
92
+ * This method and logic can be shared among all payment processing consumers: 1. bulk polling, 2. webhook, 3. single payment polling.
93
+ * @param Payment $payment The Payment object.
94
+ */
95
+ private function processPayment (Payment $ payment )
96
+ {
97
+ echo "\nPayment Status: " . $ payment ->status ;
98
+ if ($ payment ->IsPaid ()) {
99
+ echo "\nPayment Status Text: Paid. " ;
100
+ }
101
+ if ($ payment ->IsReversed ()) {
102
+ echo "\nPayment Status Text: Reversed. " ;
103
+ }
104
+ echo "\nBank: " . $ payment ->bankID ;
105
+ echo "\nBank Reference Number: " . $ payment ->paymentReference ;
106
+ echo "\nAmount Paid: " . $ payment ->amount ;
107
+ echo "\nPayment Date: " . $ payment ->paymentDate ;
108
+ echo "\nReversal/Cancel Date: " . $ payment ->canceledTime ;
109
+ echo "\nUpdate Timestamp: " . $ payment ->updateTimeStamp ;
110
+ }
111
+
112
+ }
113
+
114
+ // Instantiate and handle the webhook request. once hosted, the url needs to be shared with WeBirr for configuration
115
+ $ webhook = new Webhook ();
116
+ $ webhook ->handleRequest ();
0 commit comments