mirror of
https://github.com/pronamic/woocommerce-subscriptions.git
synced 2025-10-15 13:52:57 +00:00
201 lines
6.1 KiB
PHP
Executable File
201 lines
6.1 KiB
PHP
Executable File
<?php
|
|
/**
|
|
* Failed Scheduled Action Manager for subscription events
|
|
*
|
|
* @version 2.2.19
|
|
* @package WooCommerce Subscriptions
|
|
* @category Class
|
|
* @author Prospress
|
|
*/
|
|
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit; // Exit if accessed directly
|
|
}
|
|
|
|
class WCS_Failed_Scheduled_Action_Manager {
|
|
|
|
/**
|
|
* Action hooks we're interested in tracking.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $tracked_scheduled_actions = array(
|
|
'woocommerce_scheduled_subscription_trial_end' => 1,
|
|
'woocommerce_scheduled_subscription_payment' => 1,
|
|
'woocommerce_scheduled_subscription_payment_retry' => 1,
|
|
'woocommerce_scheduled_subscription_expiration' => 1,
|
|
'woocommerce_scheduled_subscription_end_of_prepaid_term' => 1,
|
|
);
|
|
|
|
/**
|
|
* WC Logger instance for logging messages.
|
|
*
|
|
* @var WC_Logger
|
|
*/
|
|
protected $logger;
|
|
|
|
/**
|
|
* Constructor.
|
|
*
|
|
* @param WC_Logger_Interface $logger The WC Logger instance.
|
|
*
|
|
* @since 2.2.19
|
|
*/
|
|
public function __construct( WC_Logger_Interface $logger ) {
|
|
$this->logger = $logger;
|
|
}
|
|
|
|
/**
|
|
* Attach callbacks.
|
|
*
|
|
* @since 2.2.19
|
|
*/
|
|
public function init() {
|
|
add_action( 'action_scheduler_failed_action', array( $this, 'log_action_scheduler_failure' ), 10, 2 );
|
|
add_action( 'admin_notices', array( $this, 'maybe_show_admin_notice' ) );
|
|
}
|
|
|
|
/**
|
|
* Log a message to the failed-scheduled-actions log.
|
|
*
|
|
* @param string $message the message to be written to the log.
|
|
* @since 2.2.19
|
|
*/
|
|
protected function log( $message ) {
|
|
$this->logger->add( 'failed-scheduled-actions', $message );
|
|
}
|
|
|
|
/**
|
|
* When a scheduled action failure is triggered, log information about the failed action to a WC logger.
|
|
*
|
|
* @param int $action_id the action which failed.
|
|
* @param int $timeout the number of seconds an action can run for before timing out.
|
|
* @since 2.2.19
|
|
*/
|
|
public function log_action_scheduler_failure( $action_id, $timeout ) {
|
|
$action = $this->get_action( $action_id );
|
|
|
|
if ( ! isset( $this->tracked_scheduled_actions[ $action->get_hook() ] ) ) {
|
|
return;
|
|
}
|
|
|
|
$subscription_action = $this->get_action_hook_label( $action->get_hook() );
|
|
|
|
$this->log( sprintf( 'scheduled action %s (%s) failed to finish processing after %s seconds', $action_id, $subscription_action , $timeout ) );
|
|
$this->log( sprintf( 'action args: %s', $this->get_action_args_string( $action->get_args() ) ) );
|
|
|
|
// Store information about the scheduled action for displaying an admin notice
|
|
$failed_scheduled_actions = get_option( WC_Subscriptions_Admin::$option_prefix . '_failed_scheduled_actions', array() );
|
|
|
|
$failed_scheduled_actions[ $action_id ] = array(
|
|
'args' => $action->get_args(),
|
|
'type' => $subscription_action,
|
|
);
|
|
|
|
update_option( WC_Subscriptions_Admin::$option_prefix . '_failed_scheduled_actions', $failed_scheduled_actions );
|
|
}
|
|
|
|
/**
|
|
* Display an admin notice when a scheduled action failure has occurred.
|
|
*
|
|
* @since 2.2.19
|
|
*/
|
|
public function maybe_show_admin_notice() {
|
|
$this->maybe_disable_admin_notice();
|
|
$failed_scheduled_actions = get_option( WC_Subscriptions_Admin::$option_prefix . '_failed_scheduled_actions', array() );
|
|
|
|
if ( empty( $failed_scheduled_actions ) ) {
|
|
return;
|
|
}
|
|
|
|
$affected_subscription_events = $separator = '';
|
|
|
|
foreach ( array_slice( $failed_scheduled_actions, -10, 10 ) as $action ) {
|
|
|
|
if ( isset( $action['args']['subscription_id'] ) ) {
|
|
$subject = '<a href="' . get_edit_post_link( $action['args']['subscription_id'] ) . '">#' . $action['args']['subscription_id'] . '</a>';
|
|
} elseif ( isset( $action['args']['order_id'] ) ) {
|
|
$subject = '<a href="' . get_edit_post_link( $action['args']['order_id'] ) . '">#' . $action['args']['order_id'] . '</a>';
|
|
} else {
|
|
$subject = 'unknown';
|
|
}
|
|
|
|
$affected_subscription_events .= $separator . $action['type'] . ' for ' . $subject;
|
|
$separator = "\n";
|
|
}
|
|
|
|
$notice = new WCS_Admin_Notice( 'error' );
|
|
$notice->set_content_template( 'html-failed-scheduled-action-notice.php', plugin_dir_path( WC_Subscriptions::$plugin_file ) . 'templates/admin/', array(
|
|
'failed_scheduled_actions' => $failed_scheduled_actions,
|
|
'affected_subscription_events' => $affected_subscription_events,
|
|
) );
|
|
$notice->set_actions( array(
|
|
array(
|
|
'name' => __( 'Ignore this error', 'woocommerce-subscriptions' ),
|
|
'url' => wp_nonce_url( add_query_arg( 'wcs_scheduled_action_timeout_error_notice', 'ignore' ), 'wcs_scheduled_action_timeout_error_notice', '_wcsnonce' ),
|
|
'class' => 'button',
|
|
),
|
|
array(
|
|
'name' => __( 'Learn more', 'woocommerce-subscriptions' ),
|
|
'url' => 'https://docs.woocommerce.com/document/subscriptions/scheduled-action-errors/',
|
|
'class' => 'button button-primary',
|
|
),
|
|
) );
|
|
$notice->display();
|
|
}
|
|
|
|
/**
|
|
* Handle requests to disable the failed scheduled actions admin notice.
|
|
*
|
|
* @since 2.2.19
|
|
*/
|
|
protected function maybe_disable_admin_notice() {
|
|
if ( isset( $_GET['_wcsnonce'] ) && wp_verify_nonce( $_GET['_wcsnonce'], 'wcs_scheduled_action_timeout_error_notice' ) && isset( $_GET['wcs_scheduled_action_timeout_error_notice'] ) ) {
|
|
delete_option( WC_Subscriptions_Admin::$option_prefix . '_failed_scheduled_actions' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieve a user friendly description of the scheduled action from the action hook.
|
|
*
|
|
* @param string $hook the scheduled action hook
|
|
* @return string
|
|
* @since 2.2.19
|
|
*/
|
|
protected function get_action_hook_label( $hook ) {
|
|
return str_replace( array( 'woocommerce_scheduled_', '_' ), array( '', ' ' ), $hook );
|
|
}
|
|
|
|
/**
|
|
* Retrieve a list of scheduled action args as a string.
|
|
*
|
|
* @param mixed $args the scheduled action args
|
|
* @return string
|
|
* @since 2.2.19
|
|
*/
|
|
protected function get_action_args_string( $args ) {
|
|
$args_string = $separator = '';
|
|
|
|
foreach ( $args as $key => $value ) {
|
|
if ( is_scalar( $value ) ) {
|
|
$args_string .= $separator . $key . ': ' . $value;
|
|
$separator = ', ';
|
|
}
|
|
}
|
|
|
|
return $args_string;
|
|
}
|
|
|
|
/**
|
|
* Get a scheduled action object
|
|
*
|
|
* @param int $action_id the scheduled action ID
|
|
* @return ActionScheduler_Action
|
|
* @since 2.2.19
|
|
*/
|
|
protected function get_action( $action_id ) {
|
|
$store = ActionScheduler_Store::instance();
|
|
return $store->fetch_action( $action_id );
|
|
}
|
|
}
|