3neti / emi-core
Provider-agnostic EMI domain: contracts, enums, models, DTOs, events
v1.0.4
2026-04-01 15:03 UTC
Requires
- php: ^8.3
- illuminate/support: ^12.0 || ^13.0
- lorisleiva/laravel-actions: ^2.9
- spatie/laravel-data: ^4.0
Requires (Dev)
- orchestra/testbench: ^10.0 || ^11.0
- pestphp/pest: ^4.0
README
Provider-agnostic EMI (Electronic Money Issuer) domain package for Laravel. Defines the shared contracts, enums, models, and events that any EMI provider adapter can implement.
Requirements
- PHP 8.3+
- Laravel 12 or 13
Installation
composer require lbhurtado/emi-core
The package auto-discovers its service provider. Publish the config if needed:
php artisan vendor:publish --tag=emi-core-config
What's Included
Contracts
Provider adapters must implement these interfaces:
WalletProvider— wallet CRUD (add merchant/customer, get details/balance, edit)TransferProvider— staged transfers (pre-transfer, settle, cancel)CashInProvider— inbound fundingCashOutProvider— outbound disbursement + OTP verificationSignsProviderPayloads— request signature generationVerifiesProviderPostbacks— webhook/postback signature verification
Enums
ProviderCode— registered EMI providers (currently:paynamics_constellation)WalletType— merchant, customer, phantomWalletStatus— active, locked, suspended, closedComplianceLevel— KYC levels (-1 through 4)VerificationStatus— PENDING, FOR REVIEW, REJECTED, APPROVED, RECAPTURETransactionType— cash_in, cash_out, transfer, airtime_load, bills_paymentTransactionStatus— 12 states frominitiatedthroughreconcilingTransactionDirection— inbound, outbound, internal
Models
All models use enum casts and proper Eloquent relationships:
| Model | Purpose |
|---|---|
ProviderAccount |
API credentials per provider/tenant |
Wallet |
Local mirror of provider wallet |
WalletProfile |
Personal/business identity fields |
WalletLimitSnapshot |
Captured limit fields at a point in time |
Transaction |
Master transaction mirror (indexed by request_id) |
Transfer |
Pre-transfer/settle/cancel lifecycle detail |
CashIn |
Cash-in detail (payment method, channel, sender) |
CashOut |
Cash-out detail (bank account, OTP status) |
BankAccount |
Linked settlement/disbursement bank accounts |
OtpChallenge |
OTP verification trail for cash-out |
WebhookReceipt |
Raw postback payload + signature verification |
ReconciliationEntry |
Local vs provider status drift detection |
Key Design Decisions
- request_id as first-class key — all transactions are uniquely indexed by
request_idfor idempotency and reconciliation - Enum casts everywhere — wallet type, status, compliance level, transaction status all use PHP 8.1 backed enums
- Provider-agnostic models — models don't depend on any specific provider; the
provider_codefield identifies which adapter created them - Async-first mindset —
TransactionStatusincludes states likeawaiting_provider,otp_required, andreconcilingfor asynchronous provider flows
Testing
# From the monorepo host
composer test:emi-core
License
Proprietary — Lester Hurtado