Capture order payment upon fulfillment, with Mechanic.

Mechanic is a development and ecommerce automation platform for Shopify. :)

Capture order payment upon fulfillment

Upon the fulfillment of an authorized or partially paid order, this task attempts to capture all open authorized payments for that order. (Multiple authorizations can exist on edited orders or with post-purchase upsells.)

Runs Occurs whenever an order is fulfilled.

15-day free trial – unlimited tasks

Documentation

Upon the fulfillment of an authorized or partially paid order, this task attempts to capture all open authorized payments for that order. (Multiple authorizations can exist on edited orders or with post-purchase upsells.)

If the order is modified before capturing, due to applying discounts, changing shipping fees, and/or making item adjustments, then this task will only capture up to a maximum of the current order total. Refunds that are not associated with an item adjustment are not supported by this task.

For expected results, be sure to enable "Manually capture payment for orders" in Shopify, using this guide.

Please note: You are responsible for ensuring that fulfillment occurs within the order payment's authorization period.

Developer details

Mechanic is designed to benefit everybody: merchants, customers, developers, agencies, Shopifolks, everybody.

That’s why we make it easy to configure automation without code, why we make it easy to tweak the underlying code once tasks are installed, and why we publish it all here for everyone to learn from.

(By the way, have you seen our documentation? Have you joined the Slack community?)

Open source
View on GitHub to contribute to this task
Subscriptions
shopify/orders/fulfilled
Tasks use subscriptions to sign up for specific kinds of events. Learn more
Code
{% comment %}
  -- get GraphQL order and transaction data
{% endcomment %}

{% capture query %}
  query {
    order(id: {{ order.admin_graphql_api_id | json }}) {
      id
      name
      displayFinancialStatus
      currentTotalPriceSet {
        presentmentMoney {
          amount
        }
      }
      totalReceivedSet {
        presentmentMoney {
          amount
          currencyCode
        }
      }
      transactions(capturable: true) {
        id
        kind
        totalUnsettledSet {
          presentmentMoney {
            amount
            currencyCode
          }
        }
      }
    }
  }
{% endcapture %}

{% assign result = query | shopify %}

{% if event.preview %}
  {% capture result_json %}
    {
      "data": {
        "order": {
          "id": "gid://shopify/Order/1234567890",
          "name": "#SAMPLE",
          "displayFinancialStatus": "AUTHORIZED",
          "currentTotalPriceSet": {
            "presentmentMoney": {
              "amount": "23.45"
            }
          },
          "transactions": [
            {
              "id": "gid://shopify/OrderTransaction/1234567890",
              "kind": "AUTHORIZATION",
              "totalUnsettledSet": {
                "presentmentMoney": {
                  "amount": "20.00",
                  "currencyCode": "USD"
                }
              }
            },
            {
              "id": "gid://shopify/OrderTransaction/2345678901",
              "kind": "AUTHORIZATION",
              "totalUnsettledSet": {
                "presentmentMoney": {
                  "amount": "10.00",
                  "currencyCode": "USD"
                }
              }
            }
          ]
        }
      }
    }
  {% endcapture %}

  {% assign result = result_json | parse_json %}
{% endif %}

{% assign order = result.data.order %}

{% comment %}
  -- the current total price will reflect changes made after the initial sale: discounts, shipping changes, item adjustments
  -- reducing by total received (i.e. partially paid orders) is for payment gateways that support multiple captures (e.g. Shopify Payments)
{% endcomment %}

{% assign current_total_price = order.currentTotalPriceSet.presentmentMoney.amount | times: 1.0 %}
{% assign total_received = order.totalReceivedSet.presentmentMoney.amount | times: 1.0 %}
{% assign left_to_capture = current_total_price | minus: total_received %}

{% unless event.preview %}
  {% log
    current_total_price: current_total_price,
    total_received: total_received,
    left_to_capture: left_to_capture
  %}
{% endunless %}

{% if order.displayFinancialStatus == "AUTHORIZED" or order.displayFinancialStatus == "PARTIALLY_PAID" %}
  {% assign authorized_transactions = order.transactions | where: "kind", "AUTHORIZATION" %}

  {% comment %}
    -- multiple authorizations could be on the order due to upsells or item additions
  {% endcomment %}

  {% for transaction in authorized_transactions %}
    {% comment %}
      -- capture the unsettled amount of this transaction without exceeding the amount left to capture
    {% endcomment %}

    {% assign unsettled_amount = transaction.totalUnsettledSet.presentmentMoney.amount | times: 1.0 %}
    {% assign amount_to_capture = unsettled_amount | at_most: left_to_capture %}

    {% unless event.preview %}
      {% log
        transaction_id: transaction.id,
        unsettled_amount: unsettled_amount,
        amount_to_capture: amount_to_capture
      %}
    {% endunless %}

    {% if amount_to_capture > 0 %}
      {% comment %}
        -- reduce the amount left to capture by the amount captured for this transaction
      {% endcomment %}

      {% assign left_to_capture = left_to_capture | minus: amount_to_capture %}

      {% action "shopify" %}
        mutation {
          orderCapture(
            input: {
              id: {{ order.id | json }}
              parentTransactionId: {{ transaction.id | json }}
              amount: {{ amount_to_capture | json }}
              currency: {{ transaction.totalUnsettledSet.presentmentMoney.currencyCode }}
            }
          ) {
            transaction {
              id
              status
              parentTransaction {
                id
                kind
              }
            }
            userErrors {
              field
              message
            }
          }
        }
      {% endaction %}
    {% endif %}
  {% endfor %}
{% endif %}
Task code is written in Mechanic Liquid, an extension of open-source Liquid enhanced for automation. Learn more