> ## 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.

# Transaction hooks

> Register PRE_TRANSACTION and POST_TRANSACTION hooks with the Hooks API, handle deliveries, and configure retries.

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>;
};

<Info>Available in version 0.8.4 and later.</Info>

Transaction hooks let you plug custom logic into your transaction lifecycle, before or after a transaction is processed, without changing how Blnk records the ledger entry.

Register an endpoint for each stage you care about, and Blnk calls it with the transaction context so you can validate, enrich, or trigger side effects at the right moment.

<Note>
  **Note:** Blnk only sends `transaction.*` events to transaction hooks. Other events are not supported. See [Supported events](/webhooks/events) for the full list of transaction event names.
</Note>

***

## Hook types

Blnk supports two hook types:

* `PRE_TRANSACTION`: before a transaction is applied
* `POST_TRANSACTION`: after a transaction is applied

<Tabs>
  <Tab title="Pre-transaction">
    `PRE_TRANSACTION` runs before a transaction is applied.

    Use it for validation, fraud checks, or enriching transaction data while the transaction is still in an early state, for example `QUEUED`.

    ```json PRE_TRANSACTION payload wrap expandable {3} theme={"system"}
    {
      "event": "transaction.applied",
      "hook_type": "PRE_TRANSACTION",
      "data": {
        "precise_amount": 20000,
        "amount": 200,
        "precision": 100,
        "transaction_id": "txn_9c2f1a8e-4b3d-4c91-9f2a-1a7b6c5d4e3f",
        "parent_transaction": "",
        "source": "bln_7f91a1ae-6073-4b7a-952c-23abf94a6634",
        "destination": "bln_59b83b9c-842c-427f-91eb-43cdeaf5c01a",
        "reference": "INV-2025-0008912",
        "currency": "GBP",
        "description": "Wallet top-up",
        "status": "QUEUED",
        "hash": "623e05ceb7ca7b02a9318092de7b75d8c668628e2db76c42b621b61055b10b3e",
        "allow_overdraft": true,
        "inflight": false,
        "skip_queue": false,
        "created_at": "2025-02-23T09:43:49.623494+01:00",
        "scheduled_for": "0001-01-01T00:00:00Z",
        "inflight_expiry_date": "0001-01-01T00:00:00Z"
      }
    }
    ```
  </Tab>

  <Tab title="Post-transaction">
    `POST_TRANSACTION` runs after a transaction is applied.

    Use it for notifications, workflow updates, or forwarding data to external systems once the transaction has been processed.

    ```json POST_TRANSACTION payload wrap expandable {3} theme={"system"}
    {
      "event": "transaction.applied",
      "hook_type": "POST_TRANSACTION",
      "data": {
        "precise_amount": 20000,
        "amount": 200,
        "precision": 100,
        "transaction_id": "txn_9c2f1a8e-4b3d-4c91-9f2a-1a7b6c5d4e3f",
        "parent_transaction": "",
        "source": "bln_7f91a1ae-6073-4b7a-952c-23abf94a6634",
        "destination": "bln_59b83b9c-842c-427f-91eb-43cdeaf5c01a",
        "reference": "INV-2025-0008912",
        "currency": "GBP",
        "description": "Wallet top-up",
        "status": "APPLIED",
        "hash": "623e05ceb7ca7b02a9318092de7b75d8c668628e2db76c42b621b61055b10b3e",
        "allow_overdraft": true,
        "inflight": false,
        "skip_queue": false,
        "created_at": "2025-02-23T09:43:49.623494+01:00",
        "scheduled_for": "0001-01-01T00:00:00Z",
        "inflight_expiry_date": "0001-01-01T00:00:00Z"
      }
    }
    ```
  </Tab>
</Tabs>

<Note>
  Treat `data` as the contract for your integration; fields depend on the transaction type and Blnk version. Always handle unknown keys safely when parsing JSON.
</Note>

***

## Getting started

<Steps>
  <Step title="Register a hook">
    Register your webhook URL with the [Hooks API](/reference/create-hooks):

    <Warning>
      Only the [master key](/advanced/secure-blnk#master-key) can create, update, view, list, or delete hooks. Regular API keys cannot, even if they include `hooks` scopes.
    </Warning>

    <CodeGroup>
      ```bash cURL wrap theme={"system"}
      curl -X POST "http://YOUR_BLNK_INSTANCE_URL/hooks" \
        -H "X-Blnk-Key: YOUR_MASTER_KEY" \
        -H "Content-Type: application/json" \
        -d '{
          "name": "Pre-transaction validation",
          "url": "https://api.example.com/blnk/hooks/pre",
          "type": "PRE_TRANSACTION",
          "active": true,
          "timeout": 30,
          "retry_count": 3
        }'
      ```
    </CodeGroup>

    | Field         | Description                                                                  |
    | :------------ | :--------------------------------------------------------------------------- |
    | `name`        | Human-readable label for operators and logs.                                 |
    | `url`         | Absolute `HTTPS` or `HTTP` URL that accepts requests from Blnk.              |
    | `type`        | Either `PRE_TRANSACTION` or `POST_TRANSACTION`.                              |
    | `active`      | When `false`, Blnk skips this hook until you enable it again.                |
    | `timeout`     | Maximum time in seconds Blnk waits for your endpoint to respond per attempt. |
    | `retry_count` | Maximum additional attempts after a failed delivery.                         |

    ```json 201 Created wrap theme={"system"}
    {
      "hook_id": "hoo_2vN9KxQpA1wM4rL6yB8cD0eF",
      "name": "Pre-transaction validation",
      "url": "https://api.example.com/blnk/hooks/pre",
      "type": "PRE_TRANSACTION",
      "active": true,
      "timeout": 30,
      "retry_count": 3
    }
    ```
  </Step>

  <Step title="Test the hook">
    Create a transaction with. Blnk calls your hook URL when the transaction reaches the stage you registered, `PRE` or `POST`.

    Confirm your endpoint receives an event with the expected payload and returns `2xx`.

    <CodeGroup>
      ```bash cURL wrap theme={"system"}
      curl -X POST "http://YOUR_BLNK_INSTANCE_URL/transactions" \
        -H "X-blnk-key: <api-key>" \
        -H "Content-Type: application/json" \
        -d '{
          "amount": 200,
          "precision": 100,
          "reference": "INV-2025-0008912",
          "currency": "GBP",
          "source": "@Funding",
          "destination": "@UserWallet",
          "description": "Wallet top-up"
        }'
      ```

      ```typescript TypeScript wrap theme={"system"}
      const response = await blnk.Transactions.create({
        amount: 200,
        precision: 100,
        reference: 'INV-2025-0008912',
        currency: 'GBP',
        source: '@Funding',
        destination: '@UserWallet',
        description: 'Wallet top-up',
      });
      ```

      ```go Go wrap theme={"system"}
      transaction, resp, err := client.Transaction.Create(blnkgo.CreateTransactionRequest{
        ParentTransaction: blnkgo.ParentTransaction{
          Amount:      200,
          Precision:   100,
          Reference:   "INV-2025-0008912",
          Currency:    "GBP",
          Source:      "@Funding",
          Destination: "@UserWallet",
          Description: "Wallet top-up",
        },
      })
      ```
    </CodeGroup>

    <Check>
      Your hook endpoint should log or persist the delivery. If nothing arrives, confirm the hook is `active` and the URL is reachable from Blnk Core,
    </Check>
  </Step>
</Steps>

***

## Retries and timeouts

When your hook endpoint is unreachable, times out, or returns a non-`2xx` response, Blnk retries the delivery with backoff-up to the `retry_count` you set when registering the hook.

`timeout` sets how long Blnk waits for your endpoint on each attempt.

`retry_count` sets the maximum **additional** attempts after a failed delivery.

A delivery that failed while your receiver was down can arrive minutes later once the endpoint recovers.

<Warning>
  Return a `2xx` response only after you have safely recorded or processed the event. Slow handlers can delay retries and increase duplicate deliveries under load.
</Warning>

***

## Related docs

Update, get, list, and delete hooks via the API reference.

<CardGroup cols={2}>
  <Card title="Update hooks" icon="pen" href="/reference/update-hooks">
    Change URL, timeout, or active state.
  </Card>

  <Card title="View hooks" icon="eye" href="/reference/view-hooks">
    Fetch a hook by id.
  </Card>

  <Card title="List hooks by type" icon="list" href="/reference/list-hooks-by-type">
    List PRE or POST hooks.
  </Card>

  <Card title="Delete hooks" icon="trash" href="/reference/delete-hooks">
    Remove a registered hook.
  </Card>
</CardGroup>

***

## Need help?

We are very happy to help you make the most of Blnk, regardless of whether it is your first time or you are switching from another tool.

To ask questions or discuss issues, please [contact us](mailto:support@blnkfinance.com) or [join our Discord community](https://discord.gg/7WNv94zPpx).

<CtaCallout title="Connect your ledger to Blnk Cloud" href="https://cloud.blnkfinance.com/auth/sign-up?utm_source=blnk_docs&utm_medium=documentation&utm_campaign=need-help" buttonLabel="Open Blnk Cloud" trackingEvent="clicked_cloud_signup">
  Sign up and manage your ledger with our back-office dashboard. You can invite teammates to collaborate and manage your ledger operations directly from the dashboard.
</CtaCallout>
