Автоматически отменять заказ через X дней, если в WooCommerce нет оплаты - PullRequest
2 голосов
/ 22 апреля 2019

Мне удалось собрать это после поиска в Интернете, и все же, он не работает. Моя цель - автоматически отменить ВСЕ заказы со статусом «в ожидании», независимо от того, какой платежный шлюз был выполнен, если заказ не был оплачен в течение трех дней.

Код явно неполный, и я прошу помощи, чтобы завершить его. Я проверял это с -1 minute, чтобы увидеть, если что-нибудь случилось. Это не так.

function get_unpaid_orders() {
    global $wpdb;

    $unpaid_orders = $wpdb->get_col( $wpdb->prepare( "
        SELECT posts.ID
        FROM {$wpdb->posts} AS posts
        WHERE posts.post_status = 'wc-on-hold'
        AND posts.post_date < %s
    ", date( 'Y-m-d H:i:s', strtotime('-1 minute') ) ) );

    return $unpaid_orders;
}

add_action( 'woocommerce_cancel_unpaid_submitted', 'cancel_unpaid_orders' );
function cancel_unpaid_orders() {
    $unpaid_orders = get_unpaid_orders();

    if ( $unpaid_orders ) {
        foreach ( $unpaid_orders as $unpaid_order ) {
            $order = wc_get_order( $unpaid_order );
            $cancel_order = true;

            foreach  ( $order->get_items() as $item_key => $item_values) {
                $manage_stock = get_post_meta( $item_values, '_manage_stock', true );
                if ( $manage_stock == "yes" ) {
                    $payment_method = $order->get_payment_method();
                    if ( $payment_method == "bacs" ) {
                        $cancel_order = false;
                    }
                }
            }
            if ( $cancel_order == true ) {
                $order -> update_status( 'cancelled', __( 'The order was cancelled due to no payment from customer.', 'woocommerce') );
            }
        }
    }
}

Ответы [ 2 ]

2 голосов
/ 22 апреля 2019

Обновление 4

Примечание: в WooCommerce, уже есть функция , подключенная к woocommerce_cancel_unpaid_orders ловушке действия, которая отменяет неоплаченные заказы через 7 дней.

Я не нашел woocommerce_cancel_unpaid_submitted ловушку действия, поэтому я не знаю, существует ли она и запущена ли она.

Теперь в вашем коде есть некоторые ошибки, и вы можете лучше использовать wc_get_orders () , которые дают вам правильный правильный массив WC_Order объектов вместо ...

Вот несколько способов сделать это (последние не проверены) :

1) Это последнее решение протестировано и работает Когда роли пользователей магазина или администратора просматривают список заказов администратора (выполняется только один раз в день) :

add_action( 'restrict_manage_posts', 'cancel_unpaid_orders' );
function cancel_unpaid_orders() {
    global $pagenow, $post_type;

    // Enable the process to be executed daily when browsing Admin order list 
    if( 'shop_order' === $post_type && 'edit.php' === $pagenow 
        && get_option( 'unpaid_orders_daily_process' ) < time() ) :

    $days_delay = 5; // <=== SET the delay (number of days to wait before cancelation)

    $one_day    = 24 * 60 * 60;
    $today      = strtotime( date('Y-m-d') );

    // Get unpaid orders (5 days old)
    $unpaid_orders = (array) wc_get_orders( array(
        'limit'        => -1,
        'status'       => 'on-hold',
        'date_created' => '<' . ( $today - ($days_delay * $one_day) ),
    ) );

    if ( sizeof($unpaid_orders) > 0 ) {
        $cancelled_text = __("The order was cancelled due to no payment from customer.", "woocommerce");

        // Loop through orders
        foreach ( $unpaid_orders as $order ) {
            $order->update_status( 'cancelled', $cancelled_text );
        }
    }
    // Schedule the process to the next day (executed once restriction)
    update_option( 'unpaid_orders_daily_process', $today + $one_day );

    endif;
}

Код помещается в файл function.php вашей активной дочерней темы (или активной темы).


2) Это третье решение протестировано и работает : функция срабатывает, когда любое изменение ордера в состояние "обработка" или "завершено" (выполняется только один раз день)

// Triggered on orders status change to "processing" or "completed"
add_action( 'woocommerce_order_status_changed', 'daily_cancel_unpaid_orders', 10, 4 );
function daily_cancel_unpaid_orders( $order_id, $old_status, $new_status, $order ) {
    // Enable the process to be executed daily
    if( in_array( $new_status, array('processing', 'completed') ) 
        && get_option( 'unpaid_orders_daily_process' ) < time() ) :

    $days_delay = 5; // <=== SET the delay (number of days to wait before cancelation)

    $one_day    = 24 * 60 * 60;
    $today      = strtotime( date('Y-m-d') );

    // Get unpaid orders (5 days old)
    $unpaid_orders = (array) wc_get_orders( array(
        'limit'        => -1,
        'status'       => 'on-hold',
        'date_created' => '<' . ( $today - ($days_delay * $one_day) ),
    ) );

    if ( sizeof($unpaid_orders) > 0 ) {
        $cancelled_text = __("The order was cancelled due to no payment from customer.", "woocommerce");

        // Loop through WC_Order Objects
        foreach ( $unpaid_orders as $unpaid_order ) {
            $order->update_status( 'cancelled', $cancelled_text );
        }
    }
    // Schedule the process to the next day (executed once restriction)
    update_option( 'unpaid_orders_daily_process', $today + $one_day );

    endif;
}

Код помещается в файл function.php вашей активной дочерней темы (или активной темы).


3) Так что вы можете попробовать с woocommerce_cancel_unpaid_submitted action hook:

add_action( 'woocommerce_cancel_unpaid_submitted', 'cancel_unpaid_orders' );
function cancel_unpaid_orders() {
    $days_delay = 5; // <=== SET the delay (number of days to wait before cancelation)

    $one_day    = 24 * 60 * 60;
    $today      = strtotime( date('Y-m-d') );

    // Get unpaid orders (5 days old here)
    $unpaid_orders = (array) wc_get_orders( array(
        'limit'        => -1,
        'status'       => 'on-hold',
        'date_created' => '<' . ( $today - ($days_delay * $one_day) ),
    ) );

    if ( sizeof($unpaid_orders) > 0 ) {
        $cancelled_text = __("The order was cancelled due to no payment from customer.", "woocommerce");

        // Loop through orders
        foreach ( $unpaid_orders as $order ) {
            $order->update_status( 'cancelled', $cancelled_text );
        }
    }
}

Код помещается в файл function.php вашей активной дочерней темы (или активной темы).

Код функции должен работать лучше. Для крючка я действительно не знаю.


4) Вместо этого вы можете попробовать woocommerce_cancel_unpaid_orders action hook.

0 голосов
/ 21 мая 2019

Мне нужно отменить отложенные ордера в определенном сценарии через 4 часа после создания ордера.

Пример: 2 клиента

1-ый клиент делает заказ в 17:00, поэтому его статус приостановлен (я хочу, чтобы он был отменен через 4 часа с даты создания заказа, если он не оплатил, поэтому он должен быть отменен 9: 00:00)

2-ой заказчик делает заказ в 5:37, поэтому он находится в режиме ожидания (я хочу, чтобы он был отменен через 4 часа после даты создания заказа, если он не оплатил, поэтому он должен быть отменен 9: 37)

Я хочу, чтобы он отменял заказы сразу после истечения каждого заказа по истечении 4 часов с даты создания заказа отдельно, а не как задание cron, чтобы работать каждые 4 часа, как в предыдущем примере, если событие cron начинается в 6: 00:00 и будет выполняться каждые 4 часа, поэтому первый заказ клиента истекает в 22:00, а второй - также в 22:00.

Я пробовал решения выше

, но он не мог истечь через 4 часа, как это определено только в днях, также не знал, какой из них является лучшим сценарием для меня для каждого ордера, который будет отменен отдельно через 4 часа после создания дата не как событие cron,

Я использовал многие из решений, упомянутых в соответствующем вопросе выше, но только в том, что первый ответ работает, насколько я помню, когда администратор / менеджер магазина посещает страницу заказов. Мне нужно, чтобы это было сделано автоматически без каких-либо действий со стороны людей. .

Это третье решение протестировано и работает нормально: НО я хочу, чтобы он был отменен, когда заказ находится в состоянии удержания, не завершен / не обработан, и отменен, если его прошло через 4 часа после даты создания заказа, и это планироваться и запускаться каждые 30 минут, а не ежедневно

...