oliverdavies.uk/source/_posts/drupal-8-commerce-fixing-error-on-user-checkout.md

96 lines
2.9 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

---
title: "Drupal 8 Commerce: Fixing 'No Such Customer' error on checkout"
date: 2018-08-15
excerpt: Fixing a Drupal Commerce error when a user tries to complete a checkout.
tags:
- drupal
- drupal-8
- drupal-commerce
- stripe
---
Recently I was experiencing an issue on the Drupal 8 website Im working on,
where a small number of users were not able to complete the checkout process and
instead got a generic `The site has encountered an unexpected error` message.
Looking at the log, I was able to see the error being thrown (the customer ID
has been redacted):
> Stripe\Error\InvalidRequest: No such customer: cus_xxxxxxxxxxxxxx in
> Stripe\ApiRequestor::\_specificAPIError() (line 124 of
> /var/www/vendor/stripe/stripe-php/lib/ApiRequestor.php).
Logging in to the Stripe account, I was able to confirm that the specified
customer ID did not exist. So where was it coming from, and why was Drupal
trying to retrieve a non-existent customer?
## Investigation
After some investigation, I found a table in the database named
`user__commerce_remote_id` which stores the remote customer ID for each payment
method (again, the customer ID has been redacted).
![A screenshot of a row in the user__commerce_remote_id table](/images/blog/commerce-stripe-error/remote-id-table.png){.border.p-1}
The `entity_id` and `revision_id` values in this case refer to the user that the
Stripe customer has been associated with.
As there was no customer in Stripe with this ID, I think that this must be a
customer ID from the test environment (the data from which was deleted before
the site went live).
### Drupal code
This I believe is the Drupal code where the error was being triggered:
```php
// modules/contrib/commerce_stripe/src/Plugin/Commerce/PaymentGateway/Stripe.php
public function createPayment(PaymentInterface $payment, $capture = TRUE) {
...
$owner = $payment_method->getOwner();
if ($owner && $owner->isAuthenticated()) {
$transaction_data['customer'] = $this->getRemoteCustomerId($owner);
}
try {
$result = \Stripe\Charge::create($transaction_data);
ErrorHelper::handleErrors($result);
}
catch (\Stripe\Error\Base $e) {
ErrorHelper::handleException($e);
}
...
}
```
### Stripe code
I can also see in the Stripe library where the original error is generated.
```php
private static function _specificAPIError($rbody, $rcode, $rheaders, $resp, $errorData)
{
$msg = isset($errorData['message']) ? $errorData['message'] : null;
$param = isset($errorData['param']) ? $errorData['param'] : null;
$code = isset($errorData['code']) ? $errorData['code'] : null;
switch ($rcode) {
...
case 404:
return new Error\InvalidRequest($msg, $param, $rcode, $rbody, $resp, $rheaders);
...
}
}
```
## Solution
After confirming that it was the correct user ID, simply removing that row from
the database allowed the new Stripe customer to be created and for the user to
check out successfully.