spryker / app-webhook
AppWebhook module
Installs: 2 692
Dependents: 2
Suggesters: 0
Security: 0
Stars: 0
Watchers: 21
Forks: 0
Open Issues: 1
Requires
- php: >=8.2
- spryker/glue-application-extension: ^1.0.0
- spryker/kernel: ^3.30.0
- spryker/log: ^3.0.0
- spryker/symfony: ^3.0.0
- spryker/transfer: ^3.33.0
Requires (Dev)
- codeception/codeception: ^5.0
- codeception/module-asserts: ^3.0
- codeception/module-cli: ^2.0.0
- codeception/module-filesystem: ^3.0.0
- codeception/module-phpbrowser: ^3.0.0
- codeception/module-rest: ^3.0.0
- codeception/module-webdriver: ^3.0.0
- infection/codeception-adapter: ^0.4.3
- infection/infection: ^0.27.8
- phpstan/phpdoc-parser: 1.25.0
- phpstan/phpstan: 1.10.66
- rector/rector: ^0.19.0
- spryker/app-kernel: ^1.0.0
- spryker/code-sniffer: *
- spryker/development: ^3.34.0
- spryker/glue-application: ^1.64.0
- spryker/message-broker-aws: ^1.7.0
- spryker/propel: *
- spryker/testify: *
- spryker/testify-async-api: ^0.1.4
This package is auto-updated.
Last update: 2025-01-13 11:11:00 UTC
README
Provides SyncAPI and AsyncAPI schema files and the needed code to make the Mini-Framework an App with Webhook capabilities.
Installation
composer require spryker/app-webhook
Configure
App Identifier
config/Shared/config_default.php
use Spryker\Shared\AppWebhook\AppConstants;
$config[AppConstants::APP_IDENTIFIER] = getenv('APP_IDENTIFIER') ?: 'hello-world';
Testing the AppWebhook
You can test the AppWebhook as usual with Codeception. Before that you need to run some commands:
composer setup
With these commands you've set up the AppWebhook and can start the tests
vendor/bin/codecept build
vendor/bin/codecept run
Documentation
Webhook handling
This package is responsible to receive and handle webhooks. The package provides a controller that can be used to handle incoming webhooks.
The API endpoint is /webhooks
and the controller is WebhooksController
inside the Glue Application. This package is not handling webhooks on its own, you must implement the logic to handle the webhooks via the provided \Spryker\Zed\AppWebhook\Dependency\Plugin\WebhookHandlerPluginInterface
, see the description down below.
Process in a Nutshell
- A webhook is received by the
WebhooksController
and theWebhookRequestTransfer
is created. - The webhook content is mapped to a
WebhookRequestTransfer
. - The
WebhookRequestTransfer
is passed together with aWebhookResponseTransfer
to the\Spryker\Glue\AppWebhookBackendApi\Dependency\Facade\AppWebhookBackendApiToAppWebhookFacadeInterface::handleWebhook()
method. - The
handleWebhook
method does:- Creates an identifier for this specific webhook.
- When it is not a retried webhook the webhook will be persisted in the database.
- In case of a retried webhook the webhook will be fetched from the database.
- When the number of retries exceeds the configurable allowed number of retries the webhook will be removed from the database and an exception will be thrown.
- Find the correct handler for the webhook and call the
handle
method of the handler. - When the handler returns a failed
WebhookResponseTransfer
the response will be persisted in the database. - When the handler returns a not handled
WebhookResponseTransfer
the response will be persisted in the database with a message that was provided by the implementation of the plugin. - When the handler throws an exception the exception message will be persisted in the database.
- When the handler returns a successful
WebhookResponseTransfer
the webhook will be removed from the database. - It returns the
WebhookResponseTransfer
to the controller.
- The controller formats the
WebhookResponseTransfer
into a Glue response which will be either:- 200 OK in case everything went well.
- 400 BAD REQUEST in case of a failed response.
Retry Mechanism
In a case when a webhook can not be handled it is persisted in the database and will be retried with the next incoming webhook. The number of retries is configurable and can be set in the AppWebhookConfig::getAllowedNumberOfRetries()
method.
There are numerous reasons why a webhook may fail. An exception is thrown, the plugin implementation returns a failed response or the plugin implementation returns a not handled response.
Another case could be an event is sent to the application before it is ready to handle it. For example, in the PreOrder payment of a PSP the order has not persisted yet and has no order-reference, but the PSP sends a webhook request about a payment state, in this case, the system has to wait until it can process the webhook.
Future improvements for the Retry mechanism
It may be helpful in the future to provide a console command that can be used to retry failed webhooks. This command can be used to retry all failed webhooks or only a specific webhook.
Configuration
Currently only the number of allowed retries can be configured. The configuration can be found in the AppWebhookConfig
class.
Plugins
GlueApplication
\Spryker\Glue\AppWebhookBackendApi\Plugin\GlueApplication\AppWebhookBackendApiRouteProviderPlugin
This plugin provides the routes for the AppWebhookBackendApi module.
Extensions
\Spryker\Zed\AppWebhook\Dependency\Plugin\WebhookHandlerPluginInterface
This plugin can be implemented by any other module and has two methods:
\Spryker\Zed\AppWebhook\Dependency\Plugin\WebhookHandlerPluginInterface::canHandle(WebhookRequestTransfer $webhookRequestTransfer): bool
\Spryker\Zed\AppWebhook\Dependency\Plugin\WebhookHandlerPluginInterface::handle(WebhookRequestTransfer $webhookRequestTransfer): WebhookResponseTransfer
The canHandle()
method is used to check if a webhook can be handled by a specific module. F.e. you have two handlers one for order.created
and one for order.updated
you can check in the canHandle()
method if the webhook can be handled by the module and return true or false.
The handle()
method is used to handle the webhook. The method is called if the canHandle()
method returns true. The method should return a WebhookResponseTransfer
with the status of the webhook handling.