namespace WPForms\Integrations\Stripe\Api;
use WPForms\Integrations\Stripe\Api\Webhooks\Exceptions\AmountMismatchException;
use WPForms\Vendor\Stripe\Webhook;
use BadMethodCallException;
use WPForms\Vendor\Stripe\Event as StripeEvent;
use WPForms\Vendor\Stripe\Exception\SignatureVerificationException;
use WPForms\Integrations\Stripe\Helpers;
use WPForms\Integrations\Stripe\WebhooksHealthCheck;
* Webhooks Rest Route handler.
class WebhookRoute extends Common {
private $event_type = 'unknown';
private $response_code = 200;
private function hooks() {
if ( $this->is_rest_verification() ) {
add_action( 'rest_api_init', [ $this, 'register_rest_routes' ] );
// Do not serve regular page when it seems Stripe Webhooks are still sending requests to disabled CURL endpoint.
// phpcs:ignore WordPress.Security.NonceVerification.Recommended
if ( isset( $_GET[ Helpers::get_webhook_endpoint_data()['fallback'] ] )
! Helpers::is_webhook_enabled()
|| Helpers::is_rest_api_set()
add_action( 'wp', [ $this, 'dispatch_with_error_500' ] );
if ( ! Helpers::is_webhook_enabled() || ! Helpers::is_webhook_configured() ) {
if ( Helpers::is_rest_api_set() ) {
add_action( 'rest_api_init', [ $this, 'register_rest_routes' ] );
add_action( 'wp', [ $this, 'dispatch_with_url_param' ] );
* Register webhook REST route.
public function register_rest_routes() {
if ( $this->is_rest_verification() ) {
Helpers::get_webhook_endpoint_data()['namespace'],
'/' . Helpers::get_webhook_endpoint_data()['route'],
'callback' => [ $this, 'dispatch_stripe_webhooks_payload' ],
'show_in_index' => false,
'permission_callback' => '__return_true',
* Dispatch Stripe webhooks payload for the url param.
public function dispatch_with_url_param() {
// phpcs:ignore WordPress.Security.NonceVerification.Recommended
if ( ! isset( $_GET[ Helpers::get_webhook_endpoint_data()['fallback'] ] ) ) {
$this->dispatch_stripe_webhooks_payload();
* Dispatch Stripe webhooks payload for the url param with error 500.
* Runs when url param is not configured or webhooks are not enabled at all.
public function dispatch_with_error_500() {
$this->response = esc_html__( 'It seems to be request to Stripe PHP Listener method handler but the site is not configured to use it.', 'wpforms-lite' );
$this->response_code = 500;
* Dispatch Stripe webhooks payload.
public function dispatch_stripe_webhooks_payload() {
if ( $this->is_rest_verification() ) {
$this->payload = file_get_contents( 'php://input' );
$event = Webhook::constructEvent(
$this->get_webhook_signature(),
$this->get_webhook_signing_secret()
// Update webhooks site health status.
WebhooksHealthCheck::save_status( WebhooksHealthCheck::ENDPOINT_OPTION, WebhooksHealthCheck::STATUS_OK );
WebhooksHealthCheck::save_status( WebhooksHealthCheck::SIGNATURE_OPTION,WebhooksHealthCheck::STATUS_OK );
$this->event_type = $event->type;
$this->response = 'WPForms Stripe: ' . $this->event_type . ' event received.';
$processed = $this->process_event( $event );
$this->response_code = $processed ? 200 : 202;
} catch ( AmountMismatchException $e ) {
$this->response_code = 202;
$this->response = $e->getMessage();
} catch ( SignatureVerificationException $e ) {
WebhooksHealthCheck::save_status( WebhooksHealthCheck::SIGNATURE_OPTION, WebhooksHealthCheck::STATUS_ERROR );
$this->response_code = 500;
$this->response = $e->getMessage();
} catch ( Exception $e ) {
$this->handle_exception( $e );
$this->response = $e->getMessage();
$this->response_code = $e instanceof BadMethodCallException ? 501 : 500;
* Get webhook stripe signature.
* @throws RuntimeException When Stripe signature is not set.
private function get_webhook_signature() {
if ( ! isset( $_SERVER['HTTP_STRIPE_SIGNATURE'] ) ) {
throw new RuntimeException( 'Stripe signature is not set.' );
return $_SERVER['HTTP_STRIPE_SIGNATURE']; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized, WordPress.Security.ValidatedSanitizedInput.MissingUnslash
* Get webhook signing secret.
* @throws RuntimeException When webhook signing secret is not set.
private function get_webhook_signing_secret() {
$secret = wpforms_setting( 'stripe-webhooks-secret-' . Helpers::get_stripe_mode() );
if ( empty( $secret ) ) {
throw new RuntimeException( 'Webhook signing secret is not set.' );
* @param StripeEvent $event Stripe event.
* @return bool True if event has handling class, false otherwise.
private function process_event( StripeEvent $event ) {
$webhooks = self::get_event_whitelist();
// Event can't be handled.
if ( ! isset( $webhooks[ $event->type ] ) || ! class_exists( $webhooks[ $event->type ] ) ) {
$handler = new $webhooks[ $event->type ]();
$handler->setup( $event );
return $handler->handle();
private static function get_event_whitelist() {
'charge.refunded' => Webhooks\ChargeRefunded::class,
'charge.refund.updated' => Webhooks\ChargeRefundUpdated::class,
'invoice.payment_succeeded' => Webhooks\InvoicePaymentSucceeded::class,
'invoice.created' => Webhooks\InvoiceCreated::class,
'charge.succeeded' => Webhooks\ChargeSucceeded::class,
'customer.subscription.created' => Webhooks\CustomerSubscriptionCreated::class,
'customer.subscription.updated' => Webhooks\CustomerSubscriptionUpdated::class,
'customer.subscription.deleted' => Webhooks\CustomerSubscriptionDeleted::class,
* Check if rest verification is requested.
private function is_rest_verification() {
// phpcs:ignore WordPress.Security.NonceVerification.Recommended
return isset( $_GET['verify'] ) && $_GET['verify'] === '1';
* Respond to the request.
private function respond() {
wp_die( esc_html( $this->response ), '', (int) $this->response_code );
private function log_webhook() {
// log only if WP_DEBUG_LOG and WPFORMS_WEBHOOKS_DEBUG are set to true.
! defined( 'WPFORMS_WEBHOOKS_DEBUG' ) ||
! WPFORMS_WEBHOOKS_DEBUG ||
! defined( 'WP_DEBUG_LOG' ) ||
// If it is set to explictly display logs on output, return: this would make response to Stripe malformed.
if ( defined( 'WP_DEBUG_DISPLAY' ) && WP_DEBUG_DISPLAY ) {
$webhook_log = maybe_serialize(
'event_type' => $this->event_type,
'response_code' => $this->response_code,
'response' => $this->response,
'payload' => $this->payload,
// phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_error_log
error_log( $webhook_log );
* Get webhooks events list.
public static function get_webhooks_events_list() {
return array_keys( self::get_event_whitelist() );