Fulfillment Status Digest

Generate a daily fulfillment triage digest: all open orders segmented by fulfillment age and flagged for holds or exceptions.

shopify-admin-fulfillment-status-digest


Purpose

Produces a daily ops triage digest of all unfulfilled and partially-fulfilled orders, segmented by how long they've been waiting. Flags orders with active holds. Replaces the manual process of scrolling through the Shopify admin Orders page to find aging orders and exceptions — this skill fetches every open order, computes its age, buckets it into configurable time segments, and surfaces any orders currently on a fulfillment hold, giving the ops team a complete exception queue in a single read-only operation.


Prerequisites

  • Authenticated Shopify CLI session: shopify auth login --store
  • API scopes: read_orders

  • Parameters


    ParameterTypeRequiredDefaultDescription
    storestringyesStore domain (e.g., mystore.myshopify.com)
    formatstringnohumanOutput format: human or json
    dry_runboolnofalsePreview operations without executing mutations
    aging_thresholds_daysarrayno[1, 3, 7]Day boundaries for age buckets (e.g., [1,3,7] creates: 0–1d, 1–3d, 3–7d, 7d+)
    include_holdsboolnotrueInclude orders with active fulfillment holds in a separate section
    limitintegerno250Maximum orders to fetch per page

    Workflow Steps


  • OPERATION: orders — query
  • Inputs: first: , query: "fulfillment_status:unfulfilled OR fulfillment_status:partial", sort by CREATED_AT ascending (oldest first), paginate until complete

    Expected output: All open orders with createdAt, name, displayFulfillmentStatus; compute age = now − createdAt in days; bucket into aging_thresholds_days segments


  • OPERATION: fulfillmentOrders — query (via nested order.fulfillmentOrders)
  • Inputs: For each order from Step 1: fulfillmentOrders(first: 5) to check status and requestStatus; flag any with status: ON_HOLD

    Expected output: Hold status per order, holdUntil if set; contribute to the Holds section of the digest


    GraphQL Operations


    # orders:query — validated against api_version 2025-01
    query FulfillmentStatusDigest($first: Int!, $after: String, $query: String) {
      orders(first: $first, after: $after, query: $query, sortKey: CREATED_AT) {
        edges {
          node {
            id
            name
            createdAt
            displayFulfillmentStatus
            displayFinancialStatus
            totalPriceSet {
              shopMoney { amount currencyCode }
            }
            customer {
              id
              firstName
              lastName
            }
            fulfillmentOrders(first: 5) {
              edges {
                node {
                  id
                  status
                  requestStatus
                  fulfillAt
                }
              }
            }
          }
        }
        pageInfo {
          hasNextPage
          endCursor
        }
      }
    }
    

    Note: fulfillmentOrders is a nested field on the Order type — the fulfillmentOrders:query frontmatter entry documents that this operation accesses fulfillment order data.


    Session Tracking


    Claude MUST emit the following output at each stage. This is mandatory.


    On start, emit:

    ╔══════════════════════════════════════════════╗
    ║  SKILL: fulfillment-status-digest            ║
    ║  Store: <store domain>                       ║
    ║  Started: <YYYY-MM-DD HH:MM UTC>             ║
    ╚══════════════════════════════════════════════╝
    

    After each step, emit:

    [N/TOTAL] <QUERY|MUTATION>  <OperationName>
              → Params: <brief summary of key inputs>
              → Result: <count or outcome>
    

    On completion, emit:


    For format: human (default):

    ══════════════════════════════════════════════
    OUTCOME SUMMARY
      Total open orders:          <n>
      By age bucket (0-1d):       <n>
      By age bucket (1-3d):       <n>
      By age bucket (3-7d):       <n>
      By age bucket (7d+):        <n>
      Orders on hold:             <n>
      Errors:                     0
      Output:                     none
    ══════════════════════════════════════════════
    

    For format: json, emit:

    {
      "skill": "fulfillment-status-digest",
      "store": "<domain>",
      "started_at": "<ISO8601>",
      "completed_at": "<ISO8601>",
      "dry_run": false,
      "steps": [
        { "step": 1, "operation": "FulfillmentStatusDigest", "type": "query", "params_summary": "limit: <n>, query: fulfillment_status:unfulfilled OR partial", "result_summary": "<n> orders fetched", "skipped": false },
        { "step": 2, "operation": "fulfillmentOrders", "type": "query", "params_summary": "nested per order, first: 5", "result_summary": "<n> orders on hold", "skipped": false }
      ],
      "outcome": {
        "total_open_orders": 0,
        "buckets": [
          { "label": "0-1d", "count": 0 },
          { "label": "1-3d", "count": 0 },
          { "label": "3-7d", "count": 0 },
          { "label": "7d+", "count": 0 }
        ],
        "orders_on_hold": 0,
        "errors": 0,
        "output_file": null
      }
    }
    

    Output Format


    Fulfillment Age Digest —

    Age BucketOrder CountOldest Order
    0–1 daysn#XXXX
    1–3 daysn#XXXX
    3–7 daysn#XXXX
    7+ daysn#XXXX (⚠️ review)

    Orders On Hold (if include_holds: true and holds exist):

    OrderHold SinceFulfillment Status
    #XXXX3 daysON_HOLD

    Error Handling

    ErrorCauseRecovery
    No orders returnedNo open orders in systemStore is fully fulfilled — no action needed
    fulfillmentOrders returns emptyOrder has no fulfillment assignments yetOrder may not have been assigned to a location
    Rate limit (429)Large order volume with paginationReduce limit to 100

    Best Practices

  • Run this digest first thing each morning before processing any orders — it gives you the exception queue in one view.
  • Orders in the 7d+ bucket are your highest priority; investigate and either fulfill or place an explicit hold with a reason.
  • Use format: json to pipe the digest into a Slack notification or dashboard script.
  • Combine with order-hold-and-release to act on exceptions identified in this digest without leaving the CLI.
  • For stores with 500+ open orders, set limit: 100 and expect pagination — the digest will still aggregate correctly across all pages.