zarulizham / laravel-fpx
A Laravel implementation for FPX payment service
Requires
- php: ^8.0
- illuminate/support: ^8.0|^9.0|^10.0|^11.0|^12.0
- nesbot/carbon: ^2.48.0|^3.0
Requires (Dev)
- orchestra/testbench: ^8.5
- phpunit/phpunit: ^10.1
README
This package provides Laravel implementations for Paynet FPX services.
Shipped with transaction viewer (available in dark mode)
Simple query FPX transaction and view status including request and response payload.
Installation
You can install the package via composer:
composer require zarulizham/laravel-fpx
Then run the publish command to publish the config files and support controller
php artisan fpx:publish
This will generate the following files
- The config file with default setup for you to override
fpx.php - The controller that will receive payment response and any host-to-host events
Http/Controllers/FPX/Controller.php - The assets in public directory.
- The view file with default html for you to override
payment.blade.php. Note do not change form action URLfpx.payment.auth.request.
Setups
- Add your redirect urls and your Seller and Exchange Id to the
.envfile.
FPX_INDIRECT_URL=https://app.test/payments/fpx/callback FPX_INDIRECT_PATH=payments/fpx/callback FPX_DIRECT_URL=https://app.test/payments/fpx/direct-callback FPX_DIRECT_PATH=payments/fpx/direct-callback FPX_EXCHANGE_ID= FPX_SELLER_ID=
-
You can skip this steps, if you have already generated CSR. Visit
fpx/csr/requestpath in browser to generate CSR.http://app.test/fpx/csr/requestFill the form and click on
GENERATE. On right side textarea will be generated with openSSL code. Download openSSL fromhttps://www.openssl.org/if you don't have installed it. Run openssl code to generate CSR. Submit this CSR to FPX service provider to get the Exchange Certificates. -
After generating your certificates add them to your app. By default, we look for the certificates inside the following directives. Create
fpx/uatandfpx/proddirectories instorage/app/publicdirectory and paste your certificates there. You can find UAT certificate inuat certificate/fpxuat_current.currename it with your Exchange ID and place it in mentioned UAT directory.
'certificates' => [ 'uat' => [ 'disk' => 'local', // S3 or Local. Don't put your certificate in public disk 'dir' => '/paynet/fpx/uat', ], 'production' => [ 'disk' => 'local', // S3 or Local. Don't put your certificate in public disk 'dir' => '/paynet/fpx/prod', ] ],
You can override the defaults by updating the config file.
- Publish package migrations to your Laravel app
php artisan vendor:publish --provider="ZarulIzham\\Fpx\\FpxServiceProvider" --tag="fpx-migrations"
- Run migration to add the banks and fpx_transactions table
php artisan migrate
- Laravel 12: Exclude validate CSRF Token for FPX direct callback URL in
bootstrap/app.php.
use Illuminate\Foundation\Configuration\Middleware; ->withMiddleware(function (Middleware $middleware): void { $middleware->validateCsrfTokens(except: [ 'fpx/direct', ]); })
Usage
- First run the following command to seed the banks list.
php artisan fpx:banks
you should schedule the fpx:banks Artisan command to run daily:
$schedule->command('fpx:banks')->daily();
- Add one the
x-fpx-paymentcomponent with the following attributes
<x-fpx-payment :reference-id="$invoice->id" :datetime="$invoice->created_at->format('Ymdhms')" :amount="$invoice->total" :customer-name="$company->name" :customer-email="$company->owner->email" :product-description="Salary Invoice" :class="css class name for styling button">
During testing, you can use the test-mode attribute to override the provided amount to 'MYR 1.00'
<x-fpx-payment :reference-id="$invoice->id" :datetime="$invoice->created_at->format('Ymdhms')" :amount="$invoice->total" :customer-name="$company->name" :customer-email="$company->owner->email" :product-description="Salary Invoice" :class="css class name for styling button" test-mode>
- Handle the payment response in
Http/Controllers/FPX/Controller.php
/** * This will be called after the user approve the payment * on the bank side */ public function direct(Request $request) { $response = $request->handle(); // Update your order status return 'OK'; } /** * This will handle any browser redirect from FPX */ public function indirect(Request $request) { $response = $request->handle(); // Update your order status return redirect('some/url'); }
- Check Status of all pending transactions using command
php artisan fpx:payment-status
- Check Status of specific transaction using command pass comma separated order reference ids.
php artisan fpx:payment-status reference_id1,reference_id2,reference_id3
- Check transaction status and Bank list from Controller
use ZarulIzham/Fpx/Fpx; /** * Returns status of transaction * * @param string $reference_id reference order id * @return array */ $status = Fpx::getTransactionStatus($reference_id); /** * returns collection of bank_id and name * * @param bool $getLatest (optional) pass true to get latest banks * @return \Illuminate\Support\Collection */ $banks = Fpx::getBankList(true);
- Restrict access to
fpx/transactionsroutes in yourAppServiceProvider.
use ZarulIzham\Fpx\Fpx; public function boot(): void { Fpx::auth(function ($request) { return $request->user() && $request->user()->role('Programmer'); }); }
Web Integration
You can visit http://app.test/fpx/initiate/payment for the payment flow demo of web integration.
B2B Approval / Checker
In order to approve Pending Authorization transaction, navigate here and enter transaction ID. https://simulator.fpx.uat.inet.paynet.my/UatBuyerBankSim1.7/B2BAuth.jsp
Mobile App Integration
- Append
appparameter in the URL to check the demo. http://app.test/fpx/initiate/payment/app - This will print JSON response after completion of transaction to integrate with mobile app.
Follow these steps to integrate in mobile application.
Request Details
Open http://app.test/fpx/initiate/payment/app in web view with POST method and POST below parameters.
response_format = "JSON"
reference_id = unique order reference id
customer_name = name of the buyer/customer
amount = amount to be charged
customer_email = email id of customer
remark = remarks for the transaction
additional_params = any additional parameters you want to pass
Response
You must use response field to display receipt. fpx_response is added if you need any extra details.
response.status will be succeeded, failed or pending.
{
"response": {
"status": "succeeded/failed/pending",
"message": "Payment is successful",
"transaction_id": "",
"reference_id": "",
"amount": "",
"transaction_timestamp": "",
"buyer_bank_name": "",
"response_format": "JSON",
"additional_params": "type=123"
},
"fpx_response": {
"fpx_debitAuthCode": "",
"fpx_debitAuthNo": "",
"fpx_sellerExId": "",
"fpx_creditAuthNo": "",
"fpx_buyerName": "",
"fpx_buyerId": null,
"fpx_sellerTxnTime": "",
"fpx_sellerExOrderNo": "",
"fpx_makerName": "",
"fpx_buyerBankBranch": "",
"fpx_buyerBankId": "",
"fpx_msgToken": "",
"fpx_creditAuthCode": "",
"fpx_sellerId": "",
"fpx_fpxTxnTime": "",
"fpx_buyerIban": null,
"fpx_sellerOrderNo": "",
"fpx_txnAmount": "",
"fpx_fpxTxnId": "",
"fpx_checkSum": "",
"fpx_msgType": "",
"fpx_txnCurrency": "",
}
}
You can also override payment.blade.php with your custom design to integrate with your layout. but do not change name attribute of html controls and action URL of form.
Changelog
Please see CHANGELOG for more information what has changed recently.
Contributing
Please see CONTRIBUTING for details.
Security
If you discover any security related issues, please email zarul.izham@gmail.com instead of using the issue tracker.
Credits
License
The MIT License (MIT). Please see License File for more information.
