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

# Secure Your Blnk Server

> Enable secure mode, manage secret keys, and follow best practices for a secure environment.

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

This guide walks you through running your Blnk server in secure mode and protecting your master key. For day-to-day API access, create [scoped API keys](/api-keys/overview).

Before you start, ensure you have a working instance of Blnk Core:

<Card title="Deploy Blnk" icon="sparkles" href="/home/install">
  Local install or Blnk Cloud.
</Card>

***

## Enable authentication

Enable secure mode by setting `server.secure` to `true` and providing a strong `server.secret_key`:

<CodeGroup>
  ```bash blnk.env theme={"system"}
  BLNK_SERVER_SECURE=true
  BLNK_SERVER_SECRET_KEY=your_strong_secret_key
  ```

  ```json blnk.json theme={"system"}
  {
    "server": {
      "secure": true,
      "secret_key": "your_strong_secret_key"
    }
  }
  ```
</CodeGroup>

When secure mode is enabled, every API request must include a valid key in the `X-Blnk-Key` header. Requests without a key are rejected.

The `secret_key` value becomes your **master key**. It has full access to every endpoint on the server, so use it only for administrative tasks and initial setup. For regular service traffic, create scoped API keys.

Pass the master key in the `X-Blnk-Key` header:

```bash cURL wrap theme={"system"}
curl -X GET "http://localhost:5001/ledgers" \
  -H "X-Blnk-Key: <master-key>"
```

<Warning>
  Do not commit the master key to version control. In production, store it in a secret manager or inject it through environment variables.
</Warning>

***

## About the master key

The master key is the `secret_key` from your configuration. Unlike [scoped API keys](/api-keys/overview), it is not bound to an owner or scopes, and it can call any endpoint on the server. Blnk also uses it to sign outbound [webhook](/webhooks/global-webhooks) and [transaction hook](/webhooks/transaction-hooks) deliveries.

We recommend using the master key only for initial setup and administrative tasks listed below. For regular API calls, use the scoped keys you create.

1. **Create scoped keys.** Create your first scoped keys with `POST /api-keys`. After that, services should use scoped keys, not the master key.

2. **Manage keys across owners.** List, create, and revoke keys for any owner by passing `owner` in the request. Scoped keys can manage keys only within their own `owner_id`. See [Owner context](/api-keys/owner-context).

3. **Manage hooks.** Register, update, list, and delete [transaction hooks](/webhooks/transaction-hooks). Hook management rejects scoped keys, even with `hooks:*` scopes.

4. **Verify webhooks.** Blnk signs outbound deliveries with `X-Blnk-Signature` using `server.secret_key`. Use the same secret on your receiver to verify authenticity. See [Webhook security](/webhooks/overview#webhook-security-signature-verification).

***

## Security check list

Enabling secure mode is only the start. Work through this checklist to protect your master key, keep configuration out of version control, and limit API access to what each service needs.

<Steps>
  <Step title="Master key management">
    * Use a strong, randomly generated master key
    * Never share or commit it to version control
    * Store it in environment variables or a secret management tool
    * Rotate it on a regular schedule
  </Step>

  <Step title="Configuration management">
    * Exclude `blnk.json` from version control (`.gitignore`)
    * Store sensitive configuration in environment variables
    * Implement secure secret rotation procedures
  </Step>

  <Step title="Access control">
    * Follow the principle of least privilege: use scoped keys for services, not the master key
    * Regularly review API key permissions. See [Manage API keys](/api-keys/manage-keys)
    * Register and manage [transaction hooks](/webhooks/transaction-hooks) with the master key only
  </Step>

  <Step title="Monitoring and auditing">
    * Track failed authentication attempts
    * Monitor API key usage patterns
    * Set up alerts for suspicious activity
    * Regularly review access logs
  </Step>

  <Step title="Regular updates">
    * Keep all components up to date
    * Monitor security advisories
    * Schedule regular maintenance windows
  </Step>
</Steps>

***

## Error handling

<Info>
  Structured errors are available from Blnk Core 0.15.0 and later.
</Info>

When authentication fails, Blnk returns `401 Unauthorized`. These errors apply whether you use the master key or a scoped key.

| Code                   | When it happens                                   |
| :--------------------- | :------------------------------------------------ |
| `AUTH_MISSING_API_KEY` | No `X-Blnk-Key` header was sent with the request. |
| `AUTH_INVALID_API_KEY` | The key value is unknown or malformed.            |
| `AUTH_EXPIRED_API_KEY` | The key has expired or has been revoked.          |

```json 401 Unauthorized wrap theme={"system"}
{
  "error": "Authentication required. Use X-Blnk-Key header",
  "error_detail": {
    "code": "AUTH_MISSING_API_KEY",
    "message": "Authentication required. Use X-Blnk-Key header"
  }
}
```

To resolve the error:

| Code                   | What to do                                                                                          |
| :--------------------- | :-------------------------------------------------------------------------------------------------- |
| `AUTH_MISSING_API_KEY` | Add the `X-Blnk-Key` header to your request.                                                        |
| `AUTH_INVALID_API_KEY` | Verify the key value wasn't truncated or swapped with another environment's key.                    |
| `AUTH_EXPIRED_API_KEY` | Create a replacement key before revoking the old one. See [Manage API keys](/api-keys/manage-keys). |

Permission errors for missing scopes are covered in [Scopes](/api-keys/scopes#error-handling). Owner and delegation errors are covered in [Owner context](/api-keys/owner-context#error-handling).

***

## Related docs

<CardGroup cols={2}>
  <Card title="Scoped API keys" icon="lock-keyhole" href="/api-keys/overview">
    Create and use scoped keys.
  </Card>

  <Card title="Scopes" icon="list-check" href="/api-keys/scopes">
    Pick the right permissions.
  </Card>

  <Card title="Owner context" icon="users" href="/api-keys/owner-context">
    Owner labels and cross-owner admin.
  </Card>

  <Card title="Manage keys" icon="list" href="/api-keys/manage-keys">
    List, revoke, and delegate keys.
  </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>
