A comprehensive Laravel package for integrating with the EuPago payment gateway API. Support for Multibanco, MB Way, Credit Card, Google Pay, Apple Pay, and Payouts management.
| Release | PHP | Laravel |
|---|---|---|
| 3.0.0 | >= 8.4 | 11 || 12 || 13 |
| 2.3.0 | 8.3, 8.4 | 11 || 12 |
| 2.2.0 | >= 8.3 | 11 |
| 2.1.0 | >= 8.1 | 10 |
composer require digitaldev-lx/laravel-eupagoPublish the migration
php artisan vendor:publish --provider=DigitaldevLx\\LaravelEupago\\Providers\\EuPagoServiceProvider --tag=migrationsRun the migration
php artisan migratePublish the configuration file (optional)
php artisan vendor:publish --provider=DigitaldevLx\\LaravelEupago\\Providers\\EuPagoServiceProvider --tag=configPublish the translations files (optional)
php artisan vendor:publish --provider=DigitaldevLx\\LaravelEupago\\Providers\\EuPagoServiceProvider --tag=translationsAdd the following to your .env file:
EUPAGO_ENV=test # or 'prod'
EUPAGO_API_KEY=your_api_key
EUPAGO_CHANNEL=your_channelThere are two environments available: "test" and "prod". Use the "test" environment during development and switch to "prod" when your application is ready for production.
use DigitaldevLx\LaravelEupago\MB\MB;
$order = Order::find(1);
$mb = new MB(
$order->value, // float: payment amount
$order->id, // string: external identifier
$order->date, // string: start date (Y-m-d)
$order->payment_limit_date, // string: end date (Y-m-d)
$order->value, // float: minimum value
$order->value, // float: maximum value
0 // bool: allow duplicated payments
);
try {
$mbReferenceData = $mb->create();
if ($mb->hasErrors()) {
// handle errors
}
$order->mbReferences()->create($mbReferenceData);
} catch (\Exception $e) {
// handle exception
}Response format:
[
'success' => true,
'state' => 0,
'response' => "OK",
'reference' => "000001236",
'value' => "3.00000",
'entity' => "11249",
]use DigitaldevLx\LaravelEupago\Traits\Mbable;
class Order extends Model
{
use Mbable;
}
// Retrieve references
$order = Order::find(1);
$mbReferences = $order->mbReferences;The package handles callbacks automatically, updating the payment state and triggering events.
Endpoint: GET /eupago/callback
Payment Method Code: PC:PT
use DigitaldevLx\LaravelEupago\MBWay\MBWay;
$order = Order::find(1);
$mbway = new MBWay(
$order->value, // float: payment amount
$order->id, // int: external identifier
$customer->phone, // string: phone number (alias)
'Order payment' // string|null: optional description
);
try {
$mbwayReferenceData = $mbway->create();
if ($mbway->hasErrors()) {
// handle errors
}
$order->mbwayReferences()->create($mbwayReferenceData);
} catch (\Exception $e) {
// handle exception
}use DigitaldevLx\LaravelEupago\Traits\Mbwayable;
class Order extends Model
{
use Mbwayable;
}
// Retrieve references
$order = Order::find(1);
$mbwayReferences = $order->mbwayReferences;Endpoint: GET /eupago/callback
Payment Method Code: MW:PT
use DigitaldevLx\LaravelEupago\CreditCard\CreditCard;
$creditCard = new CreditCard(
150.00, // float: amount
'ORDER-456', // string: identifier
'https://example.com/success', // string: success URL
'https://example.com/fail', // string: fail URL
'https://example.com/back', // string: back URL
'EN', // string: language (PT, EN, FR, ES)
'EUR', // string: currency
'customer@example.com', // string|null: customer email
60 // int|null: minutes form up
);
try {
$result = $creditCard->create();
if ($result['success']) {
// Redirect user to $result['redirect_url']
// Store $result['reference'] and $result['transaction_id']
}
} catch (\Exception $e) {
// handle exception
}Maximum: €3,999 per transaction
Callback Code: CC:PT
Step 1: Create Authorization
use DigitaldevLx\LaravelEupago\CreditCard\CreditCardRecurrence;
$recurrence = new CreditCardRecurrence(
'SUBSCRIPTION-123', // string: identifier
'https://example.com/success', // string: success URL
'https://example.com/fail', // string: fail URL
'https://example.com/back', // string: back URL
'PT' // string: language
);
try {
$result = $recurrence->create();
if ($result['success']) {
// Redirect user to authorize subscription
// Store $result['subscription_id']
}
} catch (\Exception $e) {
// handle exception
}Step 2: Execute Recurring Payment
use DigitaldevLx\LaravelEupago\CreditCard\CreditCardRecurringPayment;
$payment = new CreditCardRecurringPayment(
'subscription-id-from-authorization', // string: subscription ID
50.00, // float: amount
'customer@example.com', // string|null: customer email
true // bool: notify customer
);
try {
$result = $payment->create();
if ($result['success']) {
// Payment processed
}
} catch (\Exception $e) {
// handle exception
}use DigitaldevLx\LaravelEupago\Traits\Creditcardable;
use DigitaldevLx\LaravelEupago\Traits\Creditcardrecurrable;
class Order extends Model
{
use Creditcardable;
}
class Subscription extends Model
{
use Creditcardrecurrable;
}use DigitaldevLx\LaravelEupago\GooglePay\GooglePay;
$googlePay = new GooglePay(
150.00, // float: amount
'ORDER-456', // string: identifier
'https://example.com/success', // string: success URL
'https://example.com/fail', // string: fail URL
'https://example.com/back', // string: back URL
'EN', // string: language
'EUR', // string: currency
'customer@example.com', // string|null: email
'John', // string|null: first name
'Doe', // string|null: last name
'PT', // string|null: country code
true, // bool: notify customer
60 // int|null: minutes form up
);
try {
$result = $googlePay->create();
if ($result['success']) {
// Redirect to $result['redirect_url']
}
} catch (\Exception $e) {
// handle exception
}Maximum: €99,999 per transaction
Callback Code: GP:PT
use DigitaldevLx\LaravelEupago\Traits\Googlepayable;
class Order extends Model
{
use Googlepayable;
}use DigitaldevLx\LaravelEupago\ApplePay\ApplePay;
$applePay = new ApplePay(
150.00, // float: amount
'ORDER-456', // string: identifier
'https://example.com/success', // string: success URL
'https://example.com/fail', // string: fail URL
'https://example.com/back', // string: back URL
'EN', // string: language
'EUR', // string: currency
'customer@example.com', // string|null: email
'John', // string|null: first name
'Doe', // string|null: last name
'PT', // string|null: country code
true, // bool: notify customer
60 // int|null: minutes form up
);
try {
$result = $applePay->create();
if ($result['success']) {
// Redirect to $result['redirect_url']
}
} catch (\Exception $e) {
// handle exception
}Callback Code: AP:PT
use DigitaldevLx\LaravelEupago\Traits\Applepayable;
class Order extends Model
{
use Applepayable;
}Retrieve all payouts for a specific date range using OAuth Bearer Token authentication.
use DigitaldevLx\LaravelEupago\Payouts\Payout;
$payout = new Payout(
'2024-01-01', // string: start date (yyyy-mm-dd)
'2024-01-31', // string: end date (yyyy-mm-dd)
'your-bearer-token' // string: OAuth Bearer Token
);
try {
$result = $payout->list();
if ($result['success']) {
foreach ($result['payouts'] as $payout) {
// Process payout data
}
}
} catch (\Exception $e) {
// handle exception
}Retrieve all transaction details within a date range.
use DigitaldevLx\LaravelEupago\Payouts\PayoutTransaction;
$transactions = new PayoutTransaction(
'2024-01-01', // string: start date (yyyy-mm-dd)
'2024-01-31', // string: end date (yyyy-mm-dd)
'your-bearer-token' // string: OAuth Bearer Token
);
try {
$result = $transactions->list();
if ($result['success']) {
foreach ($result['transactions'] as $transaction) {
// Process transaction data
// Includes: trid, date, amount, payment_method, status
}
}
} catch (\Exception $e) {
// handle exception
}Note: For single-day queries, use the same date for both start_date and end_date.
All payment callbacks receive the following parameters:
| Name | Type | Description |
|---|---|---|
| valor | float | Payment amount |
| canal | string | Channel identifier |
| referencia | string | Payment reference |
| transacao | string | Transaction ID |
| identificador | integer | External identifier |
| mp | string | Payment method code |
| chave_api | string | API key |
| data | date:Y-m-d H:i:s | Transaction date |
| entidade | string | Entity code |
| comissao | float | Commission |
| local | string | Transaction location |
The package dispatches events throughout the payment lifecycle.
MBReferenceCreated/MBReferenceCreationFailedMBWayReferenceCreated/MBWayReferenceCreationFailedCreditCardReferenceCreated/CreditCardReferenceCreationFailedCreditCardRecurrenceAuthorizationCreated/CreditCardRecurrenceAuthorizationCreationFailedCreditCardRecurringPaymentCreated/CreditCardRecurringPaymentCreationFailedGooglePayReferenceCreated/GooglePayReferenceCreationFailedApplePayReferenceCreated/ApplePayReferenceCreationFailed
MBReferencePaidMBWayReferencePaidCreditCardReferencePaidGooglePayReferencePaidApplePayReferencePaid
MBReferenceExpiredMBWayReferenceExpired
CallbackReceived- Dispatched for all callbacksInvalidCallbackReceived- Dispatched when validation fails
Auto-discovery (recommended):
Create a listener class in app/Listeners/ — Laravel automatically discovers it via the type-hinted handle() method:
// app/Listeners/HandleMBPayment.php
namespace App\Listeners;
use DigitaldevLx\LaravelEupago\Events\MBReferencePaid;
use Illuminate\Support\Facades\Mail;
class HandleMBPayment
{
public function handle(MBReferencePaid $event): void
{
$reference = $event->reference;
$reference->mbable->update(['status' => 'paid']);
Mail::to($reference->mbable->user)->send(
new \App\Mail\PaymentConfirmed($reference)
);
}
}No manual registration needed. Laravel scans app/Listeners/ automatically.
Manual registration:
Register listeners in your app/Providers/AppServiceProvider.php:
use DigitaldevLx\LaravelEupago\Events\MBReferencePaid;
use Illuminate\Support\Facades\Event;
public function boot(): void
{
Event::listen(
MBReferencePaid::class,
SendPaymentConfirmationEmail::class,
);
}Closure listeners:
use DigitaldevLx\LaravelEupago\Events\MBReferencePaid;
use Illuminate\Support\Facades\Event;
Event::listen(function (MBReferencePaid $event) {
$reference = $event->reference;
$reference->mbable->update(['status' => 'paid']);
});Check for expired payment references and dispatch expiration events:
php artisan eupago:check-expiredThis command finds all MB references where end_date has passed and state is 0 (unpaid), then dispatches MBReferenceExpired event for each.
Scheduling:
Laravel 11 & 12:
Add to your routes/console.php:
use Illuminate\Support\Facades\Schedule;
Schedule::command('eupago:check-expired')->daily();Laravel 10:
Add to your app/Console/Kernel.php:
protected function schedule(Schedule $schedule)
{
$schedule->command('eupago:check-expired')->daily();
}The package includes comprehensive test coverage using Pest PHP.
# Run all tests
composer test
# Run with coverage (minimum 80% required)
composer test:coverage
# Run specific test file
vendor/bin/pest tests/Unit/MB/MBTest.php# Fix code style with Pint
composer lint
# Check code style without fixing
composer lint:check# Run Larastan Level 6 analysis
composer analyseuse DigitaldevLx\LaravelEupago\Models\MbReference;
use DigitaldevLx\LaravelEupago\Models\MbwayReference;
use DigitaldevLx\LaravelEupago\Models\CreditCardReference;
use DigitaldevLx\LaravelEupago\Models\GooglePayReference;
use DigitaldevLx\LaravelEupago\Models\ApplePayReference;
// Create unpaid reference
$reference = MbReference::factory()->create();
// Create paid reference
$paidReference = MbReference::factory()->paid()->create();
// Create expired reference
$expiredReference = MbReference::factory()->expired()->create();
// Other payment methods
$mbwayReference = MbwayReference::factory()->create();
$ccReference = CreditCardReference::factory()->paid()->create();
$googlePayReference = GooglePayReference::factory()->create();
$applePayReference = ApplePayReference::factory()->paid()->create();Please see CONTRIBUTING.md for details on how to contribute to this project.
Please see CHANGELOG.md for details on recent changes.
digitaldev-lx/laravel-eupago is open-sourced software licensed under the MIT license.
DigitalDev is a digital transformation agency based in Lisbon, Portugal. We specialize in transforming ideas into digital solutions that drive business growth online.
With expertise in custom web development, system integration, DevOps, AI implementation, and advanced search systems, we help businesses scale beyond generic templates with practical, data-driven solutions. Our tech stack includes Laravel, Livewire, React, and modern cloud infrastructure.
Contact: geral@digitaldev.pt | +351 961 546 227