namespace WPForms\Integrations\Square\Api;
use BadMethodCallException;
use WPForms\Integrations\Square\Helpers;
use WPForms\Integrations\Square\WebhooksHealthCheck;
* Webhooks Rest Route handler.
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 Square Webhooks are still sending requests to disabled CURL endpoint.
// phpcs:disable WordPress.Security.NonceVerification.Recommended
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' ] );
// Check if Square is configured.
if ( ! Helpers::is_square_configured() ) {
// phpcs:enable WordPress.Security.NonceVerification.Recommended
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_square_webhooks_payload' ],
'show_in_index' => false,
'permission_callback' => '__return_true',
* Dispatch Square 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_square_webhooks_payload();
* Dispatch Square 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 Square PHP Listener method handler but the site is not configured to use it.', 'wpforms-lite' );
$this->response_code = 500;
* Dispatch Square webhooks' payload.
* @throws RuntimeException When Square signature is not set.
public function dispatch_square_webhooks_payload() {
if ( $this->is_rest_verification() ) {
// Get raw payload and signature.
$this->payload = file_get_contents( 'php://input' );
$event = WebhookEvent::construct_event(
$this->get_webhook_signature(),
$this->get_webhook_signing_secret()
$event_whitelist = self::get_webhooks_events_list();
if ( ! in_array( $event->type, $event_whitelist, true ) ) {
throw new RuntimeException( 'Square event type is not whitelisted.' );
// Update webhook site health status.
WebhooksHealthCheck::save_status( WebhooksHealthCheck::ENDPOINT_OPTION, WebhooksHealthCheck::STATUS_OK );
$this->event_type = $event->type;
$this->response = 'WPForms Square: ' . $this->event_type . ' event received.';
$processed = $this->process_event( $event );
$this->response_code = $processed ? 200 : 202;
} catch ( Exception $e ) {
$this->response = $e->getMessage();
$this->response_code = $e instanceof BadMethodCallException ? 501 : 500;
* @throws RuntimeException When Square signature is not set.
private function get_webhook_signature(): string {
if ( ! isset( $_SERVER['HTTP_X_SQUARE_HMACSHA256_SIGNATURE'] ) ) {
throw new RuntimeException( 'Square signature is not set.' );
return sanitize_text_field( wp_unslash( $_SERVER['HTTP_X_SQUARE_HMACSHA256_SIGNATURE'] ) );
* Get webhook signing secret.
* @throws RuntimeException When webhook signing secret is not set.
private function get_webhook_signing_secret(): string {
$secret = wpforms_setting( 'square-webhooks-secret-' . Helpers::get_mode() );
if ( empty( $secret ) ) {
throw new RuntimeException( 'Webhook signing secret is not set.' );
* @param object $event Square event.
* @return bool True if event has handling class, false otherwise.
private function process_event( $event ): bool {
$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(): array {
'refund.updated' => Webhooks\RefundUpdated::class,
'payment.updated' => Webhooks\PaymentUpdated::class,
'payment.created' => Webhooks\PaymentCreated::class,
'subscription.created' => Webhooks\SubscriptionCreated::class,
'subscription.updated' => Webhooks\SubscriptionUpdated::class,
* Check if rest verification is requested.
private function is_rest_verification(): bool {
// 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 explicitly display logs on output, return: this would make response to Square 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 a webhook events list.
public static function get_webhooks_events_list(): array {
return array_keys( self::get_event_whitelist() );