448 lines
14 KiB
PHP
448 lines
14 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* Payment Actions
|
||
|
*
|
||
|
* @package EDD
|
||
|
* @subpackage Payments
|
||
|
* @copyright Copyright (c) 2018, Easy Digital Downloads, LLC
|
||
|
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
|
||
|
* @since 1.0
|
||
|
*/
|
||
|
|
||
|
// Exit if accessed directly
|
||
|
if ( !defined( 'ABSPATH' ) ) exit;
|
||
|
|
||
|
/**
|
||
|
* Complete a purchase
|
||
|
*
|
||
|
* Performs all necessary actions to complete a purchase.
|
||
|
* Triggered by the edd_update_payment_status() function.
|
||
|
*
|
||
|
* @since 1.0.8.3
|
||
|
* @since 3.0 Updated to use new order methods.
|
||
|
*
|
||
|
* @param int $order_id Order ID.
|
||
|
* @param string $new_status New order status.
|
||
|
* @param string $old_status Old order status.
|
||
|
*/
|
||
|
function edd_complete_purchase( $order_id, $new_status, $old_status ) {
|
||
|
|
||
|
// This specifically does not use edd_get_complete_order_statuses().
|
||
|
$completed_statuses = array( 'publish', 'complete', 'completed' );
|
||
|
// Make sure that payments are only completed once.
|
||
|
if ( in_array( $old_status, $completed_statuses, true ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Make sure the payment completion is only processed when new status is complete.
|
||
|
if ( ! in_array( $new_status, $completed_statuses, true ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$order = edd_get_order( $order_id );
|
||
|
|
||
|
if ( ! $order || 'sale' !== $order->type ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$completed_date = empty( $order->date_completed )
|
||
|
? null
|
||
|
: $order->date_completed;
|
||
|
|
||
|
$customer_id = $order->customer_id;
|
||
|
$amount = $order->total;
|
||
|
$order_items = $order->items;
|
||
|
|
||
|
do_action( 'edd_pre_complete_purchase', $order_id );
|
||
|
|
||
|
if ( is_array( $order_items ) ) {
|
||
|
|
||
|
// Increase purchase count and earnings.
|
||
|
foreach ( $order_items as $item ) {
|
||
|
|
||
|
// "bundle" or "default"
|
||
|
$download_type = edd_get_download_type( $item->product_id );
|
||
|
|
||
|
// Increase earnings and fire actions once per quantity number.
|
||
|
for ( $i = 0; $i < $item->quantity; $i++ ) {
|
||
|
|
||
|
// Ensure these actions only run once, ever.
|
||
|
if ( empty( $completed_date ) ) {
|
||
|
|
||
|
// For backwards compatibility purposes, we need to construct an array and pass it
|
||
|
// to edd_complete_download_purchase.
|
||
|
$item_fees = array();
|
||
|
|
||
|
foreach ( $item->get_fees() as $key => $item_fee ) {
|
||
|
/** @var EDD\Orders\Order_Adjustment $item_fee */
|
||
|
|
||
|
$download_id = $item->product_id;
|
||
|
$price_id = $item->price_id;
|
||
|
$no_tax = (bool) 0.00 === $item_fee->tax;
|
||
|
$id = is_null( $item_fee->type_key ) ? $item_fee->id : $item_fee->type_key;
|
||
|
if ( array_key_exists( $id, $item_fees ) ) {
|
||
|
$id .= '_2';
|
||
|
}
|
||
|
|
||
|
$item_fees[ $id ] = array(
|
||
|
'amount' => $item_fee->amount,
|
||
|
'label' => $item_fee->description,
|
||
|
'no_tax' => $no_tax ? $no_tax : false,
|
||
|
'type' => 'fee',
|
||
|
'download_id' => $download_id,
|
||
|
'price_id' => $price_id ? $price_id : null,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$item_options = array(
|
||
|
'quantity' => $item->quantity,
|
||
|
'price_id' => $item->price_id,
|
||
|
);
|
||
|
|
||
|
/*
|
||
|
* For backwards compatibility from pre-3.0: add in order item meta prefixed with `_option_`.
|
||
|
* While saving, we've migrated these values to order item meta, but people may still be looking
|
||
|
* for them in this cart details array, so we need to fill them back in.
|
||
|
*/
|
||
|
$order_item_meta = edd_get_order_item_meta( $item->id );
|
||
|
if ( ! empty( $order_item_meta ) ) {
|
||
|
foreach ( $order_item_meta as $item_meta_key => $item_meta_value ) {
|
||
|
if ( '_option_' === substr( $item_meta_key, 0, 8 ) && isset( $item_meta_value[0] ) ) {
|
||
|
$item_options[ str_replace( '_option_', '', $item_meta_key ) ] = $item_meta_value[0];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$cart_details = array(
|
||
|
'name' => $item->product_name,
|
||
|
'id' => $item->product_id,
|
||
|
'item_number' => array(
|
||
|
'id' => $item->product_id,
|
||
|
'quantity' => $item->quantity,
|
||
|
'options' => $item_options,
|
||
|
),
|
||
|
'item_price' => $item->amount,
|
||
|
'quantity' => $item->quantity,
|
||
|
'discount' => $item->discount,
|
||
|
'subtotal' => $item->subtotal,
|
||
|
'tax' => $item->tax,
|
||
|
'fees' => $item_fees,
|
||
|
'price' => $item->amount,
|
||
|
);
|
||
|
|
||
|
do_action( 'edd_complete_download_purchase', $item->product_id, $order_id, $download_type, $cart_details, $item->cart_index );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Clear the total earnings cache
|
||
|
delete_transient( 'edd_earnings_total' );
|
||
|
delete_transient( 'edd_earnings_total_without_tax' );
|
||
|
|
||
|
// Clear the This Month earnings (this_monththis_month is NOT a typo)
|
||
|
delete_transient( md5( 'edd_earnings_this_monththis_month' ) );
|
||
|
delete_transient( md5( 'edd_earnings_todaytoday' ) );
|
||
|
}
|
||
|
|
||
|
// Increase the customer's purchase stats
|
||
|
$customer = new EDD_Customer( $customer_id );
|
||
|
$customer->recalculate_stats();
|
||
|
|
||
|
edd_increase_total_earnings( $amount );
|
||
|
|
||
|
// Check for discount codes and increment their use counts
|
||
|
$discounts = $order->get_discounts();
|
||
|
foreach ( $discounts as $adjustment ) {
|
||
|
/** @var EDD\Orders\Order_Adjustment $adjustment */
|
||
|
|
||
|
edd_increase_discount_usage( $adjustment->description );
|
||
|
}
|
||
|
|
||
|
// Ensure this action only runs once ever
|
||
|
if ( empty( $completed_date ) ) {
|
||
|
$date = EDD()->utils->date()->format( 'mysql' );
|
||
|
|
||
|
$date_refundable = edd_get_refund_date( $date );
|
||
|
$date_refundable = false === $date_refundable
|
||
|
? ''
|
||
|
: $date_refundable;
|
||
|
|
||
|
// Save the completed date
|
||
|
edd_update_order( $order_id, array(
|
||
|
'date_completed' => $date,
|
||
|
'date_refundable' => $date_refundable,
|
||
|
) );
|
||
|
|
||
|
// Required for backwards compatibility.
|
||
|
$payment = edd_get_payment( $order_id );
|
||
|
|
||
|
/**
|
||
|
* Runs **when** a purchase is marked as "complete".
|
||
|
*
|
||
|
* @since 2.8 Added EDD_Payment and EDD_Customer object to action.
|
||
|
*
|
||
|
* @param int $order_id Payment ID.
|
||
|
* @param EDD_Payment $payment EDD_Payment object containing all payment data.
|
||
|
* @param EDD_Customer $customer EDD_Customer object containing all customer data.
|
||
|
*/
|
||
|
do_action( 'edd_complete_purchase', $order_id, $payment, $customer );
|
||
|
|
||
|
// If cron doesn't work on a site, allow the filter to use __return_false and run the events immediately.
|
||
|
$use_cron = apply_filters( 'edd_use_after_payment_actions', true, $order_id );
|
||
|
if ( false === $use_cron ) {
|
||
|
/**
|
||
|
* Runs **after** a purchase is marked as "complete".
|
||
|
*
|
||
|
* @see edd_process_after_payment_actions()
|
||
|
*
|
||
|
* @since 2.8 - Added EDD_Payment and EDD_Customer object to action.
|
||
|
*
|
||
|
* @param int $order_id Payment ID.
|
||
|
* @param EDD_Payment $payment EDD_Payment object containing all payment data.
|
||
|
* @param EDD_Customer $customer EDD_Customer object containing all customer data.
|
||
|
*/
|
||
|
do_action( 'edd_after_payment_actions', $order_id, $payment, $customer );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Empty the shopping cart
|
||
|
edd_empty_cart();
|
||
|
}
|
||
|
add_action( 'edd_update_payment_status', 'edd_complete_purchase', 100, 3 );
|
||
|
|
||
|
/**
|
||
|
* Schedules the one time event via WP_Cron to fire after purchase actions.
|
||
|
*
|
||
|
* Is run on the edd_complete_purchase action.
|
||
|
*
|
||
|
* @since 2.8
|
||
|
* @param $payment_id
|
||
|
*/
|
||
|
function edd_schedule_after_payment_action( $payment_id ) {
|
||
|
$use_cron = apply_filters( 'edd_use_after_payment_actions', true, $payment_id );
|
||
|
if ( $use_cron ) {
|
||
|
$after_payment_delay = apply_filters( 'edd_after_payment_actions_delay', 30, $payment_id );
|
||
|
|
||
|
// Use time() instead of current_time( 'timestamp' ) to avoid scheduling the event in the past when server time
|
||
|
// and WordPress timezone are different.
|
||
|
wp_schedule_single_event( time() + $after_payment_delay, 'edd_after_payment_scheduled_actions', array( $payment_id, false ) );
|
||
|
}
|
||
|
}
|
||
|
add_action( 'edd_complete_purchase', 'edd_schedule_after_payment_action', 10, 1 );
|
||
|
|
||
|
/**
|
||
|
* Executes the one time event used for after purchase actions.
|
||
|
*
|
||
|
* @since 2.8
|
||
|
* @param $payment_id
|
||
|
* @param $force
|
||
|
*/
|
||
|
function edd_process_after_payment_actions( $payment_id = 0, $force = false ) {
|
||
|
if ( empty( $payment_id ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$payment = new EDD_Payment( $payment_id );
|
||
|
$has_fired = $payment->get_meta( '_edd_complete_actions_run' );
|
||
|
if ( ! empty( $has_fired ) && false === $force ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$payment->add_note( __( 'After payment actions processed.', 'easy-digital-downloads' ) );
|
||
|
$payment->update_meta( '_edd_complete_actions_run', time() ); // This is in GMT
|
||
|
|
||
|
do_action( 'edd_after_payment_actions', $payment_id, $payment, new EDD_Customer( $payment->customer_id ) );
|
||
|
}
|
||
|
add_action( 'edd_after_payment_scheduled_actions', 'edd_process_after_payment_actions', 10, 1 );
|
||
|
|
||
|
/**
|
||
|
* Updates week-old+ 'pending' orders to 'abandoned'
|
||
|
*
|
||
|
* This function is only intended to be used by WordPress cron.
|
||
|
*
|
||
|
* @since 1.6
|
||
|
* @return void
|
||
|
*/
|
||
|
function edd_mark_abandoned_orders() {
|
||
|
|
||
|
// Bail if not in WordPress cron
|
||
|
if ( ! edd_doing_cron() ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$args = array(
|
||
|
'status' => 'pending',
|
||
|
'number' => 9999999,
|
||
|
'output' => 'edd_payments',
|
||
|
);
|
||
|
|
||
|
add_filter( 'posts_where', 'edd_filter_where_older_than_week' );
|
||
|
|
||
|
$payments = edd_get_payments( $args );
|
||
|
|
||
|
remove_filter( 'posts_where', 'edd_filter_where_older_than_week' );
|
||
|
|
||
|
if( $payments ) {
|
||
|
foreach( $payments as $payment ) {
|
||
|
if( 'pending' === $payment->post_status ) {
|
||
|
$payment->status = 'abandoned';
|
||
|
$payment->save();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
add_action( 'edd_weekly_scheduled_events', 'edd_mark_abandoned_orders' );
|
||
|
|
||
|
/**
|
||
|
* Process an attempt to complete a recoverable payment.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
* @return void
|
||
|
*/
|
||
|
function edd_recover_payment() {
|
||
|
if ( empty( $_GET['payment_id'] ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$payment = new EDD_Payment( $_GET['payment_id'] );
|
||
|
if ( $payment->ID !== (int) $_GET['payment_id'] ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ( ! $payment->is_recoverable() ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (
|
||
|
// Logged in, but wrong user ID
|
||
|
( is_user_logged_in() && $payment->user_id != get_current_user_id() )
|
||
|
|
||
|
// ...OR...
|
||
|
||
|
||
|
|
||
|
// Logged out, but payment is for a user
|
||
|
( ! is_user_logged_in() && ! empty( $payment->user_id ) )
|
||
|
) {
|
||
|
$redirect = get_permalink( edd_get_option( 'purchase_history_page' ) );
|
||
|
edd_set_error( 'edd-payment-recovery-user-mismatch', __( 'Error resuming payment.', 'easy-digital-downloads' ) );
|
||
|
edd_redirect( $redirect );
|
||
|
}
|
||
|
|
||
|
$payment->add_note( __( 'Payment recovery triggered URL', 'easy-digital-downloads' ) );
|
||
|
|
||
|
// Empty out the cart.
|
||
|
EDD()->cart->empty_cart();
|
||
|
|
||
|
// Recover any downloads.
|
||
|
foreach ( $payment->cart_details as $download ) {
|
||
|
edd_add_to_cart( $download['id'], $download['item_number']['options'] );
|
||
|
|
||
|
// Recover any item specific fees.
|
||
|
if ( ! empty( $download['fees'] ) ) {
|
||
|
foreach ( $download['fees'] as $key => $fee ) {
|
||
|
$fee['id'] = ! empty( $fee['id'] ) ? $fee['id'] : $key;
|
||
|
EDD()->fees->add_fee( $fee );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Recover any global fees.
|
||
|
foreach ( $payment->fees as $key => $fee ) {
|
||
|
$fee['id'] = ! empty( $fee['id'] ) ? $fee['id'] : $key;
|
||
|
EDD()->fees->add_fee( $fee );
|
||
|
}
|
||
|
|
||
|
// Recover any discounts.
|
||
|
if ( 'none' !== $payment->discounts && ! empty( $payment->discounts ) ){
|
||
|
$discounts = ! is_array( $payment->discounts ) ? explode( ',', $payment->discounts ) : $payment->discounts;
|
||
|
|
||
|
foreach ( $discounts as $discount ) {
|
||
|
edd_set_cart_discount( $discount );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EDD()->session->set( 'edd_resume_payment', $payment->ID );
|
||
|
|
||
|
$redirect_args = array( 'payment-mode' => urlencode( $payment->gateway ) );
|
||
|
$redirect = add_query_arg( $redirect_args, edd_get_checkout_uri() );
|
||
|
edd_redirect( $redirect );
|
||
|
}
|
||
|
add_action( 'edd_recover_payment', 'edd_recover_payment' );
|
||
|
|
||
|
/**
|
||
|
* If the payment trying to be recovered has a User ID associated with it, be sure it's the same user.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
* @return void
|
||
|
*/
|
||
|
function edd_recovery_user_mismatch() {
|
||
|
if ( ! edd_is_checkout() ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$resuming_payment = EDD()->session->get( 'edd_resume_payment' );
|
||
|
if ( $resuming_payment ) {
|
||
|
$payment = new EDD_Payment( $resuming_payment );
|
||
|
if ( is_user_logged_in() && $payment->user_id != get_current_user_id() ) {
|
||
|
edd_empty_cart();
|
||
|
edd_set_error( 'edd-payment-recovery-user-mismatch', __( 'Error resuming payment.', 'easy-digital-downloads' ) );
|
||
|
edd_redirect( get_permalink( edd_get_option( 'purchase_page' ) ) );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
add_action( 'template_redirect', 'edd_recovery_user_mismatch' );
|
||
|
|
||
|
/**
|
||
|
* If the payment trying to be recovered has a User ID associated with it, we need them to log in.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
* @return void
|
||
|
*/
|
||
|
function edd_recovery_force_login_fields() {
|
||
|
$resuming_payment = EDD()->session->get( 'edd_resume_payment' );
|
||
|
if ( $resuming_payment ) {
|
||
|
$payment = new EDD_Payment( $resuming_payment );
|
||
|
$requires_login = edd_no_guest_checkout();
|
||
|
if ( ( $requires_login && ! is_user_logged_in() ) && ( $payment->user_id > 0 && ( ! is_user_logged_in() ) ) ) {
|
||
|
?>
|
||
|
<div class="edd-alert edd-alert-info">
|
||
|
<p><?php _e( 'To complete this payment, please login to your account.', 'easy-digital-downloads' ); ?></p>
|
||
|
<p>
|
||
|
<a href="<?php echo esc_url( edd_get_lostpassword_url() ); ?>" title="<?php esc_attr_e( 'Lost Password', 'easy-digital-downloads' ); ?>">
|
||
|
<?php _e( 'Lost Password?', 'easy-digital-downloads' ); ?>
|
||
|
</a>
|
||
|
</p>
|
||
|
</div>
|
||
|
<?php
|
||
|
$show_register_form = edd_get_option( 'show_register_form', 'none' );
|
||
|
|
||
|
if ( 'both' === $show_register_form || 'login' === $show_register_form ) {
|
||
|
return;
|
||
|
}
|
||
|
do_action( 'edd_purchase_form_login_fields' );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
add_action( 'edd_purchase_form_before_register_login', 'edd_recovery_force_login_fields' );
|
||
|
|
||
|
/**
|
||
|
* When processing the payment, check if the resuming payment has a user id and that it matches the logged in user.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
* @param $verified_data
|
||
|
* @param $post_data
|
||
|
*/
|
||
|
function edd_recovery_verify_logged_in( $verified_data, $post_data ) {
|
||
|
$resuming_payment = EDD()->session->get( 'edd_resume_payment' );
|
||
|
if ( $resuming_payment ) {
|
||
|
$payment = new EDD_Payment( $resuming_payment );
|
||
|
$same_user = ! empty( $payment->user_id ) && ( is_user_logged_in() && $payment->user_id == get_current_user_id() );
|
||
|
$same_email = strtolower( $payment->email ) === strtolower( $post_data['edd_email'] );
|
||
|
|
||
|
if ( ( is_user_logged_in() && ! $same_user ) || ( ! is_user_logged_in() && (int) $payment->user_id > 0 && ! $same_email ) ) {
|
||
|
edd_set_error( 'recovery_requires_login', __( 'To complete this payment, please login to your account.', 'easy-digital-downloads' ) );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
add_action( 'edd_checkout_error_checks', 'edd_recovery_verify_logged_in', 10, 2 );
|