# Inbound API Key Re-issue Runbook **Status:** BREAKING change — action required on every environment that uses the inbound API (`POST /api/{methodName}`). **Date:** 2026-06-02 **Migration:** `RetireInboundApiKeyStore` This runbook covers the migration of inbound API authentication from the legacy SQL Server `X-API-Key` scheme to the shared `ZB.MOM.WW.Auth.ApiKeys` store. After this change **all existing inbound API keys are invalidated** and every API client must be re-issued a new credential. --- ## 1. What changed and why | | Before | After | |---|---|---| | Header | `X-API-Key: ` | `Authorization: Bearer sbk__` | | Verification | Deterministic HMAC hash, looked up in SQL Server | Peppered, constant-time HMAC compare in the shared `ZB.MOM.WW.Auth.ApiKeys` verifier | | Storage | SQL Server `ApiKeys` table (config DB) | `ZB.MOM.WW.Auth.ApiKeys` SQLite store | | Authorization | `ApiMethod.ApprovedApiKeyIds` CSV linking methods to key IDs | Per-key **scopes**, where each scope string is an allowed method name (ordinal, case-sensitive) | **Why:** the inbound credential path now reuses the shared auth library that the rest of the `ZB.MOM.WW.*` family uses, with a single, tested, peppered verifier and a proper one-time-token issuance model. The deterministic SQL Server hash table and its method-link CSV are retired. The legacy `ApiKeyHasher` / `IApiKeyHasher` and the in-repo `ApiKeyValidator` are gone — inbound auth runs through `IApiKeyVerifier`. > The old `X-API-Key` credentials are **not migrated**. There is no automated > conversion: the stored hashes are not reversible, and the new tokens have a > different shape (`sbk__`). Every key must be re-issued. --- ## 2. Required configuration (per environment) Set these under the ScadaBridge configuration for each environment (appsettings, environment variables, or your secret store): | Key | Value | Notes | |---|---|---| | `ScadaBridge:InboundApi:ApiKeyStore:SqlitePath` | Filesystem path to the SQLite key store | Defaults to `/data/inbound-api-keys.sqlite` if unset. Choose a durable, backed-up path on a writable volume. | | `ScadaBridge:InboundApi:ApiKeyPepper` | A strong, random string, **≥ 16 characters** | **DIFFERENT per environment.** Keep it secret (secret store, not source control). This is the HMAC pepper that binds every stored key to this deployment; it is also the verifier's pepper secret. | Notes: - The pepper must be present and at least 16 characters or the host fails fast at startup (`AddZbApiKeyAuth`). - Changing the pepper after keys are issued invalidates all keys in that environment (they would no longer verify). Set it once, per environment, and keep it stable. - The token prefix is `sbk` and migrations run on startup by default (`ScadaBridge:InboundApi:ApiKeyStore:RunMigrationsOnStartup = true`); these are wired by the Host and normally need no operator change. --- ## 3. Database migration step Apply the EF Core migration `RetireInboundApiKeyStore` to the SQL Server configuration database. It: - drops the `ApiKeys` table, and - drops the `ApprovedApiKeyIds` column from `ApiMethods`. If migrations are applied automatically on deploy (the default for the central node), this happens as part of the rollout. To apply manually: ```bash dotnet ef database update RetireInboundApiKeyStore \ --project src/ZB.MOM.WW.ScadaBridge.ConfigurationDatabase \ --startup-project src/ZB.MOM.WW.ScadaBridge.Host ``` > Applying this migration **permanently drops** the old key data. Take a database > backup first if you need a record of the prior `ApiKeys` rows for audit purposes > (the hashes are not usable credentials, but the names/enabled flags may be of > record-keeping value). The new inbound keys live in the **SQLite** store (section 2), not in SQL Server. --- ## 4. Operator re-issue procedure Re-issue one key per client. Each key is created with the exact method names it is allowed to call (its scopes). ### Option A — Admin UI 1. Navigate to **`/admin/api-keys`** in the central UI. 2. **Create** a new key: enter a display name and select the allowed method(s). 3. The one-time token `sbk__` is shown **exactly once** — copy it now. It cannot be retrieved later. 4. Distribute the token securely to the owning client. ### Option B — CLI ```bash scadabridge --url security api-key create \ --name \ --methods ``` - `--methods` is a comma-separated list of allowed method names — these become the key's scopes. A method name must match the registered `ApiMethod.Name` **exactly** (case-sensitive). - The command prints `API key created. KeyId: ` and then the one-time token on stdout (the "save this now — it will not be shown again" advisory goes to stderr, so piping stdout captures only the token). Capture the `sbk_…` token at issue time; it is the only moment the secret is available. To later change which methods a key may call: ```bash scadabridge --url security api-key set-methods --key-id --methods ``` --- ## 5. Client change Each API client must replace its header: - **Remove:** `X-API-Key: ` - **Add:** `Authorization: Bearer sbk__` Example: ```http POST /api/CreateOrder HTTP/1.1 Host: scadabridge.example.com Authorization: Bearer sbk_7f3a...._9c1e.... Content-Type: application/json { "orderId": "..." } ``` The token is the full `sbk__` string exactly as issued — do not split or transform it. --- ## 6. Verification 1. **Authn (valid key):** call an allowed method with the new Bearer token → `200` (or the method's normal result). 2. **Authn (no/old credential):** call with no `Authorization` header, or with the old `X-API-Key` header only → `401` with `{"error":"Invalid or missing API key"}`. 3. **Authz (out of scope):** call a method the key is **not** scoped for → `403` with `{"error":"API key not approved for this method"}`. A non-existent method name returns the identical `403` body (enumeration-safe — by design). 4. **Audit:** a successful call records the verified key's display name as the audit actor; an auth failure records `Actor=null`. Confirm via the audit log. 5. Confirm no client is still sending `X-API-Key` (those requests now fail `401`). --- ## 7. Rollback The migration `Down` recreates the `ApiKeys` table and the `ApprovedApiKeyIds` column, **but the dropped key rows are not restored** — `Down` only rebuilds empty structures. Rolling the migration back does **not** recover any credential. Therefore "rollback" means **reverting the deployment** to the prior build (which still speaks `X-API-Key`), not reverting the keys: 1. Redeploy the previous ScadaBridge build. 2. If you took a SQL Server backup before section 3, restore the `ApiKeys` table from it so the old keys verify again. 3. Without that backup, the old keys are gone and must be re-created under the legacy scheme as well. Because rollback is costly and lossy, prefer rolling **forward**: complete the re-issue in section 4 and fix any straggler clients rather than reverting.