> ## Documentation Index
> Fetch the complete documentation index at: https://docs.blnkfinance.com/llms.txt
> Use this file to discover all available pages before exploring further.

# Building an Escrow Payment Workflow

> Learn how to implement an escrow payment with Blnk.

export const CtaCallout = props => {
  const {title, buttonLabel, href, trackingEvent, buttonTarget, rel = "noopener noreferrer", children} = props;
  const handleCtaClick = () => {
    if (typeof window === "undefined" || !trackingEvent) {
      return;
    }
    try {
      window.dispatchEvent(new CustomEvent("blnk:docs-cta", {
        detail: {
          name: trackingEvent,
          href
        }
      }));
    } catch {}
    try {
      window.posthog?.capture?.(trackingEvent, {
        href
      });
    } catch {}
    const gaPayload = {
      cta_href: href
    };
    try {
      window.gtag?.("event", trackingEvent, gaPayload);
    } catch {}
    try {
      window.dataLayer = window.dataLayer || [];
      window.dataLayer.push({
        event: trackingEvent,
        ...gaPayload
      });
    } catch {}
  };
  const isExternal = typeof href === "string" && (/^https?:\/\//i).test(href);
  const target = buttonTarget ?? (isExternal ? "_blank" : undefined);
  const linkRel = isExternal ? rel : undefined;
  return <section className="cta-callout not-prose relative my-8 w-full min-w-0 overflow-hidden rounded-xl border border-zinc-200 p-5 dark:border-white/10">
      <div className="cta-callout-noise" aria-hidden="true" />
      <div className="cta-callout-layout">
        {title ? <div className="cta-callout-title-row">
            <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 28 28" width="14" height="14" className="cta-callout-icon shrink-0 text-zinc-800 dark:text-zinc-200" aria-hidden="true">
              <g fill="none" fillRule="nonzero">
                <path d="M28 0v28H0V0h28ZM14.691833333333335 27.134333333333334l-0.012833333333333334 0.0023333333333333335 -0.08283333333333333 0.04083333333333334 -0.023333333333333334 0.004666666666666667 -0.016333333333333335 -0.004666666666666667 -0.08283333333333333 -0.04083333333333334c-0.011666666666666667 -0.004666666666666667 -0.022166666666666668 -0.0011666666666666668 -0.028000000000000004 0.005833333333333334l-0.004666666666666667 0.011666666666666667 -0.019833333333333335 0.49933333333333335 0.005833333333333334 0.023333333333333334 0.011666666666666667 0.015166666666666667 0.12133333333333333 0.08633333333333333 0.0175 0.004666666666666667 0.014000000000000002 -0.004666666666666667 0.12133333333333333 -0.08633333333333333 0.014000000000000002 -0.018666666666666668 0.004666666666666667 -0.019833333333333335 -0.019833333333333335 -0.4981666666666667c-0.0023333333333333335 -0.011666666666666667 -0.0105 -0.019833333333333335 -0.019833333333333335 -0.021Zm0.3091666666666667 -0.13183333333333336 -0.015166666666666667 0.0023333333333333335 -0.21583333333333335 0.1085 -0.011666666666666667 0.011666666666666667 -0.0035000000000000005 0.012833333333333334 0.021 0.5016666666666667 0.005833333333333334 0.014000000000000002 0.009333333333333334 0.008166666666666668 0.23450000000000004 0.1085c0.014000000000000002 0.004666666666666667 0.026833333333333334 0 0.03383333333333334 -0.009333333333333334l0.004666666666666667 -0.016333333333333335 -0.03966666666666667 -0.7163333333333334c-0.0035000000000000005 -0.014000000000000002 -0.011666666666666667 -0.023333333333333334 -0.023333333333333334 -0.025666666666666667Zm-0.8341666666666667 0.0023333333333333335a0.026833333333333334 0.026833333333334334 0 0 0 -0.0315 0.007000000000000001l-0.007000000000000001 0.016333333333333335 -0.03966666666666667 0.7163333333333334c0 0.014000000000000002 0.008166666666666668 0.023333333333333334 0.019833333333333335 0.028000000000000004l0.0175 -0.0023333333333333335 0.23450000000000004 -0.1085 0.011666666666666667 -0.009333333333333334 0.004666666666666667 -0.012833333333333334 0.019833333333333335 -0.5016666666666667 -0.0035000000000000005 -0.014000000000000002 -0.011666666666666667 -0.011666666666666667 -0.21466666666666667 -0.10733333333333334Z" strokeWidth="1.1667" />
                <path fill="currentColor" d="M14 2.916666666666667A1.75 1.75 0 0 1 15.750000000000002 4.666666666666667v6.302333333333334L21.207666666666668 7.816666666666667a1.75 1.75 0 0 1 1.75 3.031L17.5 14l5.457666666666667 3.151166666666667a1.75 1.75 0 0 1 -1.75 3.031l-5.457666666666667 -3.1500000000000004V23.333333333333336a1.75 1.75 0 0 1 -3.5 0v-6.302333333333334L6.792333333333334 20.183333333333337a1.75 1.75 0 1 1 -1.75 -3.031L10.5 14 5.042333333333334 10.848833333333333a1.75 1.75 0 0 1 1.75 -3.031l5.457666666666667 3.1500000000000004V4.666666666666667A1.75 1.75 0 0 1 14 2.916666666666667Z" strokeWidth="1.1667" />
              </g>
            </svg>
            <p className="cta-callout-title min-w-0 font-semibold text-zinc-800 dark:text-zinc-200">
              {title}
            </p>
          </div> : null}
        <div className={`cta-callout-body text-sm leading-normal text-zinc-800 dark:text-zinc-200${title ? " cta-callout-body--indented" : ""}`}>
          {children}
        </div>
        <a href={href} target={target} rel={linkRel} onClick={handleCtaClick} data-docs-cta={trackingEvent || undefined} className="cta-callout-button inline-flex items-center justify-center gap-1 rounded-full bg-white px-3 py-1.5 text-sm font-semibold transition hover:bg-zinc-100 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-white/50 dark:bg-white dark:hover:bg-zinc-200">
          {buttonLabel}
          <span className="cta-callout-button-arrow" aria-hidden="true">
            →
          </span>
        </a>
      </div>
    </section>;
};

In this tutorial, we'll build a simple escrow payment workflow using Blnk's inflight transaction feature. Here's what we'll do:

1. Initiate an escrow transaction.
2. Verify that all conditions have been met.
3. Update the transaction based on the conditions.
4. Handle alternative use cases cases like partial fulfilment.

For this tutorial, we'll use the [Blnk TypeScript SDK](/sdks/typescript) and [Blnk Go SDK](/sdks/go) for the implementation. If you prefer, you can also refer to the [API reference](/reference/create-ledger) for details on the available endpoints.

***

## Designing your map

Before writing code, it's crucial to design a [money movement map](/ledgers/money-movement-map) that outlines how money moves in your system. This serves as the blueprint for your implementation.

For our escrow payments workflow, here's our map:

<img src="https://mintcdn.com/blnk/jKiGmb7nTD9y-R5a/images/tutorials/escrow-payment-map.png?fit=max&auto=format&n=jKiGmb7nTD9y-R5a&q=85&s=a0783e30769f55ffb47de60522d0deda" alt="Escrow payments map" width="907" height="393" data-path="images/tutorials/escrow-payment-map.png" />

[Explore the map yourself here](https://map.blnkfinance.xyz/IgmT0FSLzG)

Read more: [Create inflight →](/transactions/inflight/creating-inflight)

This map shows three key aspects of the escrow fund flow:

* `Buyer` initiates transaction to `Seller`.
* Transaction is held inflight until the escrow conditions are met or not.
* Transaction is completed or discarded depending on the inflight status, and the balances get updated.

***

## Set up your implementation

Based on our map, we'll implement the following steps:

1. Initiate a transaction with inflight between the `Buyer Wallet` and the `Seller Wallet`.
2. Update the status of our inflight transaction.
   * Commit if the escrow payment is successful.
   * Void if the escrow payment failed.

***

## Prerequisites

Before starting, ensure you have:

1. A running Blnk Core instance (e.g. at `http://localhost:5001`).
2. An API key for Blnk (replace `YOUR_API_KEY` in the code examples). Required for authenticated requests.
3. Optionally, you can connect your Blnk Core to your [Blnk Cloud](https://cloud.blnkfinance.com/?utm_source=blnk_docs\&utm_medium=documentation\&utm_campaign=tutorials%2Fquick-start%2Fescrow-payments) workspace to view your ledger data.
4. Balances created for the buyer and seller wallets in your Blnk Ledger. Learn how: [Building a Wallet Management System with Blnk →](/tutorials/quick-start/wallet-management).
5. Funded the buyer wallet balance in ledger.

***

## Initiate escrow transaction

[Create a transaction](/transactions/introduction#recording-a-transaction) with `inflight` enabled.

<CodeGroup>
  ```bash cURL wrap expandable theme={"system"}
  curl -X POST "http://localhost:5001/transactions" \
    -H "X-blnk-key: <api-key>" \
    -H "Content-Type: application/json" \
    -d '{
      "precise_amount": 10043,
      "precision": 100,
      "reference": "escrow_ref_12345",
      "description": "Escrow payment",
      "currency": "USD",
      "source": "<buyer_balance_id>",
      "destination": "<seller_balance_id>",
      "inflight": true,
      "meta_data": {
        "transaction_type": "escrow",
        "escrow_status": "pending",
        "buyer_confirmation": false,
        "seller_confirmation": false
      }
    }'
  ```

  ```typescript TypeScript wrap expandable {13} theme={"system"}
  async function createEscrowTransaction(buyerWalletId, sellerWalletId, amount, uniqueReference, description) {
    
    
    const escrow = await blnk.Transactions.create({
      precise_amount: amount * 100,
      precision: 100,
      reference: uniqueReference,
      description: description || "Escrow payment",
      currency: "USD",
      source: buyerWalletId,
      destination: sellerWalletId,
      inflight: true, // Holds funds in escrow
      meta_data: {
        transaction_type: "escrow",
        escrow_status: "pending",
        buyer_confirmation: false,
        seller_confirmation: false,
      }
    });
    
    console.log("Escrow transaction created:", escrow.data.transaction_id);
    return escrow.data.transaction_id;
  }
  ```

  ```go Go wrap expandable theme={"system"}
  func createEscrowTransaction(buyerWalletID, sellerWalletID string, amount float64, uniqueReference, description string) (string, error) {
      client := getClient()
      if description == "" {
          description = "Escrow payment"
      }
      escrow, _, err := client.Transaction.Create(blnkgo.CreateTransactionRequest{
          ParentTransaction: blnkgo.ParentTransaction{
              PreciseAmount: amount * 100,
              Precision:   100,
              Reference:   uniqueReference,
              Description: description,
              Currency:    "USD",
              Source:      buyerWalletID,
              Destination: sellerWalletID,
              MetaData: map[string]interface{}{
                  "transaction_type":     "escrow",
                  "escrow_status":        "pending",
                  "buyer_confirmation":   false,
                  "seller_confirmation":  false,
              },
          },
          Inflight: true,
      })
      if err != nil {
          return "", err
      }
      fmt.Println("Escrow transaction created:", escrow.TransactionID)
      return escrow.TransactionID, nil
  }
  ```
</CodeGroup>

<Note>
  When `inflight` is enabled, the transaction remains pending, ensuring that neither the source nor the destination balance is affected until the specified conditions are met.
</Note>

***

## Path 1: Successful escrow payment

For a successful escrow payment, commit the inflight transaction to apply it to the buyer and seller balances:

<CodeGroup>
  ```bash cURL wrap theme={"system"}
  curl -X PUT "http://localhost:5001/transactions/inflight/<inflight-transaction-id>" \
    -H "X-blnk-key: <api-key>" \
    -H "Content-Type: application/json" \
    -d '{
      "status": "commit"
    }'
  ```

  ```typescript TypeScript wrap expandable theme={"system"}
  async function commitInflightTransaction(transactionId) {
    
    
    try {
      const response = await blnk.Transactions.updateStatus(transactionId, {
        status: 'commit',
      });
      if (response.status !== 200 || !response.data) {
        throw new Error(response.message ?? 'Failed to commit transaction');
      }
      console.log(`Transaction ${transactionId} successfully committed`);
      return response.data;
    } catch (error) {
      console.error(`Error committing transaction ${transactionId}:`, error);
      throw error;
    }
  }
  ```

  ```go Go wrap theme={"system"}
  func commitInflightTransaction(transactionID string) (*blnkgo.Transaction, error) {
      client := getClient()
      committed, _, err := client.Transaction.Update(transactionID, blnkgo.UpdateStatus{
          Status: blnkgo.InflightStatusCommit,
      })
      if err != nil {
          return nil, err
      }
      fmt.Printf("Transaction %s successfully committed\n", transactionID)
      return committed, nil
  }
  ```
</CodeGroup>

***

## Path 2: Failed escrow payment

If the escrow failed, void the inflight transaction instead:

<CodeGroup>
  ```bash cURL wrap theme={"system"}
  curl -X PUT "http://localhost:5001/transactions/inflight/<inflight-transaction-id>" \
    -H "X-blnk-key: <api-key>" \
    -H "Content-Type: application/json" \
    -d '{
      "status": "void"
    }'
  ```

  ```typescript TypeScript wrap expandable theme={"system"}
  async function voidInflightTransaction(transactionId) {
    
    
    try {
      const response = await blnk.Transactions.updateStatus(transactionId, {
        status: 'void',
      });
      if (response.status !== 200 || !response.data) {
        throw new Error(response.message ?? 'Failed to void transaction');
      }
      console.log(`Transaction ${transactionId} successfully voided`);
      return response.data;
    } catch (error) {
      console.error(`Error voiding transaction ${transactionId}:`, error);
      throw error;
    }
  }
  ```

  ```go Go wrap theme={"system"}
  func voidInflightTransaction(transactionID string) (*blnkgo.Transaction, error) {
      client := getClient()
      voided, _, err := client.Transaction.Update(transactionID, blnkgo.UpdateStatus{
          Status: blnkgo.InflightStatusVoid,
      })
      if err != nil {
          return nil, err
      }
      fmt.Printf("Transaction %s successfully voided\n", transactionID)
      return voided, nil
  }
  ```
</CodeGroup>

***

## Path 3: Partial escrow payment

Escrow payments can also be fulfilled in bits, meaning you can complete part of the total amount over time, e.g., for a \$100 transaction, you can commit \$30, then \$40, then the remaining \$30 later.

Commit a partial amount from the inflight escrow transaction:

<CodeGroup>
  ```bash cURL wrap theme={"system"}
  curl -X PUT "http://localhost:5001/transactions/inflight/<inflight-transaction-id>" \
    -H "X-blnk-key: <api-key>" \
    -H "Content-Type: application/json" \
    -d '{
      "status": "commit",
      "precise_amount": 3000
    }'
  ```

  ```typescript TypeScript wrap expandable theme={"system"}
  async function commitPartialInflightTransaction(transactionId, partialAmount) {
    
    
    try {
      const response = await blnk.Transactions.updateStatus(transactionId, {
        status: 'commit',
        precise_amount: partialAmount * 100,
      });
      if (response.status !== 200 || !response.data) {
        throw new Error(response.message ?? 'Failed to commit partial transaction');
      }
      console.log(`Transaction ${transactionId} successfully committed`);
      return response.data;
    } catch (error) {
      console.error(`Error committing transaction ${transactionId}:`, error);
      throw error;
    }
  }
  ```

  ```go Go wrap theme={"system"}
  func commitPartialInflightTransaction(transactionID string, partialAmount float64) (*blnkgo.Transaction, error) {
      client := getClient()
      committed, _, err := client.Transaction.Update(transactionID, blnkgo.UpdateStatus{
          Status: blnkgo.InflightStatusCommit,
          PreciseAmount: partialAmount * 100,
      })
      if err != nil {
          return nil, err
      }
      fmt.Printf("Transaction %s successfully committed\n", transactionID)
      return committed, nil
  }
  ```
</CodeGroup>

<Tip>
  You can also make multiple partial commits and void the remaining inflight amount if needed.
</Tip>

***

## Conclusion

This tutorial demonstrated how to build a secure escrow payment system using Blnk's inflight feature, enabling safe transactions for buyers and sellers. The system includes initiating escrows, tracking confirmations, releasing funds upon mutual agreement, and handling cancellations.

With Blnk's flexible architecture, you can easily adapt this escrow system to various business models, including from marketplace transactions and freelancer payments to real estate deals and complex business contracts.

<Card title="Inflight transactions" icon="plane" href="/transactions/inflight/creating-inflight">
  Hold, commit, void, and best practices.
</Card>

***

<CtaCallout title="Need help with your use case?" href="https://blnkfinance.com/contact/us?utm_source=blnk_docs&utm_medium=documentation&utm_campaign=home%2Finstall" buttonLabel="Speak with us" trackingEvent="clicked_pro_support">
  Get dedicated support for architecture reviews, integration planning, ledger workflows, and production deployment.
</CtaCallout>
