sudiptpa / paypal-ipn
Modernized, framework-agnostic PayPal IPN verification package for legacy IPN workflows.
Requires
- php: >=8.2 <8.6
Requires (Dev)
- phpstan/phpstan: ^1.12 || ^2.0
- phpunit/phpunit: ^10.5 || ^11.0
- rector/rector: ^1.2
Suggests
- ext-curl: Enable the built-in cURL transport for zero-dependency IPN verification.
- guzzlehttp/guzzle: Use the optional Guzzle transport or inject your own preferred HTTP client adapter.
This package is auto-updated.
Last update: 2026-03-21 07:31:48 UTC
README
Framework-agnostic, modernized PayPal IPN verification package for legacy Instant Payment Notification workflows.
Why This Package
PayPal IPN is legacy, but thousands of projects still depend on it. This package now gives you a modern fluent API for new work while still preserving the familiar listener flow that existing integrations already use.
If you are integrating today, prefer the modern Ipn entry point. Keep the legacy handler flow when you want minimal application changes during upgrades.
Highlights
- preferred modern fluent usage with
Sujip\PayPal\Notification\Ipn - stable legacy-style usage with
ArrayHandlerandStreamHandler - zero hard runtime dependencies beyond PHP
- no hard Guzzle dependency
- no hard Symfony dependency
- built-in lightweight event dispatcher
- built-in cURL transport when
ext-curlis available - optional Guzzle transport support
- custom transport and custom dispatcher support
- PHP
8.2to<8.6
Installation
composer require sudiptpa/paypal-ipn
Optional Guzzle usage:
composer require guzzlehttp/guzzle
Documentation
- User Guide
- Migration Guide
- Architecture
- cURL Transport
- Guzzle Transport
- Custom Transport
- Contributing
- Security
Looking For A Modern Unified PayPal Package?
If you are starting a new integration or want one modern package for both legacy IPN and PayPal Webhooks, use sudiptpa/paypal-notifications.
Use this package when:
- you want a focused IPN-only package
- you need to modernize an existing IPN integration with minimal behavioral change
Use paypal-notifications when:
- you want support for both PayPal IPN and Webhooks
- you are building a newer integration around the modern PayPal notification model
- you want one package to handle legacy and newer notification flows together
Recommended Usage
For new integrations and most upgrades, prefer the modern fluent API:
use Sujip\PayPal\Notification\Ipn; $result = Ipn::fromArray($_POST) ->sandbox() ->verify();
You can still attach listeners, custom transports, and dispatchers as needed.
Quick Start
Modern Fluent Usage
use Sujip\PayPal\Notification\Events\Failure; use Sujip\PayPal\Notification\Events\Invalid; use Sujip\PayPal\Notification\Events\Verified; use Sujip\PayPal\Notification\Ipn; $result = Ipn::fromArray($_POST) ->sandbox() ->onVerified(function (Verified $event): void { $payload = $event->getPayload(); // Process the verified PayPal IPN here. }) ->onInvalid(function (Invalid $event): void { $payload = $event->getPayload(); // Log the invalid payload here. }) ->onError(function (Failure $event): void { $error = $event->error(); // Log transport or verification errors here. }) ->verify();
Legacy Listener Usage
use Sujip\PayPal\Notification\Events\Failure; use Sujip\PayPal\Notification\Events\Invalid; use Sujip\PayPal\Notification\Events\Verified; use Sujip\PayPal\Notification\Handler\ArrayHandler; $manager = (new ArrayHandler($_POST)) ->sandbox() ->handle(); $manager->onVerified(function (Verified $event): void { $payload = $event->getPayload(); // Process the verified PayPal IPN here. }); $manager->onInvalid(function (Invalid $event): void { $payload = $event->getPayload(); // Log the invalid payload here. }); $manager->onError(function (Failure $event): void { $error = $event->error(); // Log transport or verification errors here. }); $manager->fire();
Public API Stability
The package is intentionally conservative about the legacy integration shape. These areas should be treated as public API for consumers:
Sujip\PayPal\Notification\IpnSujip\PayPal\Notification\Handler\ArrayHandlerSujip\PayPal\Notification\Handler\StreamHandlerSujip\PayPal\Notification\Manager- verification events
Sujip\PayPal\Notification\Contracts\Service- listener methods and event names
Internal implementation classes may evolve over time, especially where compatibility wrappers exist to preserve user-facing behavior.
Transport Resolution
The package resolves verification transports in this order:
- a custom service passed via
->using() - a transport or Guzzle client passed via
->withTransport()or->withClient() - the built-in cURL transport when
ext-curlis available - the optional Guzzle transport when Guzzle is installed
If none of those are available, the verification cycle fails with a clear transport exception.
Backward Compatibility
The legacy listener-driven flow is intentionally preserved, but it is now the compatibility path rather than the recommended starting point:
$manager = (new ArrayHandler($payload))->sandbox()->handle(); $manager->onVerified(fn ($event) => null); $manager->onInvalid(fn ($event) => null); $manager->onError(fn ($event) => null); $manager->fire();
There is also a modern fluent entry point with the same verification engine underneath:
Ipn::fromArray($payload) ->sandbox() ->onVerified(fn ($event) => null) ->verify();
That means users can upgrade the package internals without needing a functionality rewrite in their applications, while newer integrations can adopt a cleaner API.
Extendability
Custom transport
Implement Sujip\PayPal\Notification\Contracts\Service and pass it to ->using().
Optional Guzzle transport
Install Guzzle in your application and pass a client into ->withClient().
External event dispatcher
Pass any compatible dispatcher object into ->withDispatcher() as long as it provides addListener() and dispatch() methods.
Support
- use the GitHub issue tracker for bugs and regressions
- report security issues privately using SECURITY.md
- use PayPal sandbox IPN verification in your own app before deploying changes
Development
composer lint
composer stan
composer rector:check
composer test
composer test:coverage
Testing Strategy
The test suite covers:
- endpoint switching between live and sandbox
- payload parsing and serialization
- verified, invalid, and failure event dispatching
- custom service injection
- local dispatcher behavior and external dispatcher interoperability
- legacy and modern public usage styles
- request transport validation behavior
License
MIT