<?php namespace WPForms\Integrations\Square; use WPForms\Admin\Notice; /** * Webhooks Health Check class. * * @since 1.9.5 */ class WebhooksHealthCheck { /** * Endpoint status option name. * * @since 1.9.5 */ public const ENDPOINT_OPTION = 'wpforms_square_webhooks_endpoint_status'; /** * Signature verified key. * * @since 1.9.5 */ public const STATUS_OK = 'ok'; /** * Signature error key. * * @since 1.9.5 */ private const STATUS_ERROR = 'error'; /** * AS task name. * * @since 1.9.5 */ private const ACTION = 'wpforms_square_webhooks_health_check'; /** * Admin notice ID. * * @since 1.9.5 */ private const NOTICE_ID = 'wpforms_square_webhooks_site_health'; /** * Initialization. * * @since 1.9.5 */ public function init() { $this->hooks(); } /** * Register hooks. * * @since 1.9.5 */ private function hooks() { add_action( 'admin_notices', [ $this, 'admin_notice' ] ); add_action( self::ACTION, [ $this, 'process_webhooks_status_action' ] ); add_action( 'action_scheduler/migration_complete', [ $this, 'maybe_schedule_task' ] ); } /** * Schedule webhook health check. * * @since 1.9.5 */ public function maybe_schedule_task() { /** * Allow customers to disable a webhook health check task. * * @since 1.9.5 * * @param bool $cancel True if a task needs to be canceled. */ $is_canceled = (bool) apply_filters( 'wpforms_integrations_square_webhooks_health_check_cancel', false ); $tasks = wpforms()->obj( 'tasks' ); // Bail early in some instances. if ( $is_canceled || $tasks === null || ! Helpers::is_square_configured() || $tasks->is_scheduled( self::ACTION ) ) { return; } /** * Filters the webhook health check interval. * * @since 1.9.5 * * @param int $interval Interval in seconds. */ $interval = (int) apply_filters( 'wpforms_integrations_square_webhooks_health_check_interval', HOUR_IN_SECONDS ); $tasks->create( self::ACTION ) ->recurring( time(), $interval ) ->register(); } /** * Process webhook status. * * @since 1.9.5 */ public function process_webhooks_status_action() { // Bail out if user unchecked option to enable webhooks. if ( ! Helpers::is_webhook_enabled() ) { return; } $last_payment = $this->get_last_square_payment(); // Bail out if there is no Square payment, // and remove options for reason to avoid any edge cases. if ( ! $last_payment ) { delete_option( self::ENDPOINT_OPTION ); return; } // If a last Square payment has processed status and webhooks are not valid, // most likely there is an issue with webhooks. if ( $last_payment['status'] === 'processed' && time() > ( strtotime( $last_payment['date_created_gmt'] ) + ( 15 * MINUTE_IN_SECONDS ) ) ) { Helpers::reset_webhook_configuration(); self::save_status( self::ENDPOINT_OPTION, self::STATUS_ERROR ); return; } self::save_status( self::ENDPOINT_OPTION, self::STATUS_OK ); } /** * Determine whether there is Square payment. * * @since 1.9.5 * * @return array */ private function get_last_square_payment(): array { $payment = wpforms()->obj( 'payment' )->get_payments( [ 'gateway' => 'square', 'mode' => 'any', 'number' => 1, ] ); return ! empty( $payment[0] ) ? $payment[0] : []; } /** * Display notice about issues with webhooks. * * @since 1.9.5 */ public function admin_notice() { // Bail out if a Square account is not connected. if ( ! Helpers::is_square_configured() ) { return; } // Bail out if webhooks are not enabled. if ( ! Helpers::is_webhook_enabled() ) { return; } // Bail out if webhooks are configured and active. if ( Helpers::is_webhook_configured() ) { return; } // Show notice only in case if ENDPOINT_OPTION has error status. if ( get_option( self::ENDPOINT_OPTION, self::STATUS_OK ) === self::STATUS_OK ) { return; } // Bail out if there are no Square payments. if ( ! $this->get_last_square_payment() ) { return; } $notice = sprintf( wp_kses( /* translators: %s - WPForms.com URL for Square webhooks documentation. */ __( 'Looks like you have a problem with your webhooks configuration. Please check and confirm that you\'ve configured the WPForms webhooks in your Square account. This notice will disappear automatically when a new Square request comes in. See our <a href="%1$s" rel="nofollow noopener" target="_blank">documentation</a> for more information.', 'wpforms-lite' ), [ 'a' => [ 'href' => [], 'target' => [], 'rel' => [], ], ] ), esc_url( wpforms_utm_link( 'https://wpforms.com/docs/setting-up-square-webhooks/', 'Admin', 'Square Webhooks not active' ) ) ); Notice::error( $notice, [ 'dismiss' => true, 'slug' => self::NOTICE_ID, ] ); } /** * Save webhooks status. * * @since 1.9.5 * * @param string $option Option name. * @param string $value Status value. */ public static function save_status( string $option, string $value ) { if ( ! in_array( $value, [ self::STATUS_OK, self::STATUS_ERROR ], true ) ) { return; } update_option( $option, $value ); } }