For subscriptions that aren't standalone, the initial order triggers a sequence of creating a draft order, and later closing it and creating a new draft - the renewal.
It would be nice to allow subscriptions where the store does not need to have a stored payment method. Instead, when the current term of the subscription ends - or before, the user is directed to the checkout, to pay for the next term of the subscription. This is almost possible at the moment.
There are a few problems standing in the way of this:
Whenever the order is refreshed, as a recurring order, the payment method and payment gateway are reset using the subscription as the source, overriding any values set in the checkout.
src/RecurringOrderManager.php, in function refreshOrder, lines 105-112:
$subscriptions = $this->collectSubscriptions($order);
$payment_method = $this->selectPaymentMethod($subscriptions);
$billing_profile = $payment_method ? $payment_method->getBillingProfile() : NULL;
$payment_gateway_id = $payment_method ? $payment_method->getPaymentGatewayId() : NULL;
$order->set('billing_profile', $billing_profile);
$order->set('payment_method', $payment_method);
$order->set('payment_gateway', $payment_gateway_id);
Proposed solution: Introduce a test to distinguish between recurring order refresh during checkout and other times, and do not alter payment method or gateway during checkout. More below.
The code would become something like:
$subscriptions = $this->collectSubscriptions($order);
$payment_method = $this->selectPaymentMethod($subscriptions);
$billing_profile = $payment_method ? $payment_method->getBillingProfile() : NULL;
$payment_gateway_id = $payment_method ? $payment_method->getPaymentGatewayId() : NULL;
$order->set('billing_profile', $billing_profile);
// When in the checkout, do not override user input with info held on the subscription
if ($this->route_match->getRouteName() !== 'commerce_checkout.form') {
$order->set('payment_method', $payment_method);
$order->set('payment_gateway', $payment_gateway_id);
}
A refresh is never triggered because the order is not in draft state, and cron checks for draft before enqueing the order for close and refresh.
Proposed solution:
When a recurring order is paid, through the checkout, enqueue a renewal.
Alternatively, and perhaps better because it would match the architecture for ordinary renewals, a cron task to find active subscriptions where there is no draft. For these a renewal would be enqueued. This also has the advantage that if the draft is deleted for other reasons, then a new one is created.
None - the checkout is already available for the draft.
There is an Issue to make it not available to users, but that issue suggests it is available to all users on all orders. This is not the case for my testing.
TBD. Probably none.
None.
Needs review
1.0
Code
Not all content is available!
It's likely this issue predates Contrib.social: some issue and comment data are missing.