See it in action on the hev-shop demo store.

Operations

ApiKey CRD

An ApiKey is a minted credential as a resource. Layer owns the credential lifecycle — mint, verify, revoke, expire — and what the key opens is declared per resource: each entitlement names a VectorStore, a Warehouse, or Layer itself, and carries the scopes and claims for that target. Claims are opaque to Layer — an external system can use Layer as its key store and keep authorization decisions to itself.

Keys have two authoring surfaces that round-trip through one schema: kubectl get apikey -o yaml and GET /v2/keys/{keyId} are two spellings of the same object.

apiVersion: hevlayer.com/v1alpha1
kind: ApiKey
metadata:
  name: cohort-reader
  namespace: layer
spec:
  owner: acme
  description: cohort read access
  entitlements:
    vectorstore.prod-turbopuffer:
      scopes: [read]
      namespaces: ["cohort-*"]
    warehouse.prod-snowflake:
      claims:
        - "notes:cohort:*:read"
  expiresAfter: 365d
status:
  keyId: 0a1b2c3d-…
  phase: Active
  lookupHash: sha256:…
  createdAt: "2026-06-10T00:00:00Z"
  expiresAt: "2027-06-10T00:00:00Z"
  secretRef:
    name: apikey-cohort-reader

Spec

FieldPurpose
ownerOptional free-form owner label, echoed in list and authenticate responses.
descriptionOptional free-form description.
entitlementsMap keyed by target resource. Each entry carries scopes, namespaces, and claims for that target.
expiresAfterDuration or never. Defaults to 365d; status.expiresAt is computed at mint.

Entitlements

KeyTarget
vectorstore.<name>Data-plane access through the named store. scopes (read, write) gate routes whose Index resolves to that store; namespaces globs constrain which upstream namespaces.
warehouse.<name>A list of opaque claims strings bound to the source system. Layer stores and echoes them; the application routes on them. No client route reaches a source — clients touch indexes, not warehouses — so the entitlement grants nothing in Layer and inerts when the warehouse is deleted.
layerThe control plane itself. scopes: [admin] covers key management and Pipeline/Function create/delete/control routes, and satisfies read and write everywhere.

Scope meanings match inbound auth: read covers query, fetch, scans, and metrics; write covers namespace writes and worker routes.

claims is a list of opaque strings, allowed on any entitlement and the only field on a warehouse entitlement. Layer stores them, returns them from list, get, and authenticate, and never interprets them — an existing permission grammar (service:resource_type:resource_id:action strings, a legacy entitlement vocabulary) drops in verbatim, and the consuming application maps them to its own authorization.

An entitlement whose target does not exist grants nothing and surfaces as a status condition (EntitlementTargetMissing) — not an admission error, so keys and their targets can be applied in either order. Check the condition after applying: a typo in a target name looks the same as a missing target.

A key whose entitlements carry only claims — no scopes — is a pure external-store key: it authenticates, but opens no Layer route.

Minting

REST. POST /v2/keys generates the token, creates the ApiKey resource, and returns the token in the response — once. The raw token is never persisted; Layer stores only one-way hashes on the resource.

POST   /v2/keys                       # 201 { keyId, …, token } — token returned once
GET    /v2/keys                       # metadata only; ?includeRevoked
GET    /v2/keys/{keyId}
POST   /v2/keys/{keyId}/revoke        # idempotent
DELETE /v2/keys/{keyId}               # hard delete
POST   /v2/keys/authenticate          # body { token } → 200 { keyId, entitlements, … } | 401

Key-management routes require a key with the layer entitlement at admin scope. POST /v2/keys/authenticate is unauthenticated by construction — the token is the credential.

CRD. Apply an ApiKey with no credential. The operator mints the token, writes it to a Secret named in status.secretRef (key token), and moves phase from Pending to Active. The Secret is the token delivery; it is owned by the ApiKey and garbage-collected with it. Rotation is delete-and-reapply — a new key, a new token.

Verification

External systems present the raw token to POST /v2/keys/authenticate and get back keyId (a stable actor id) plus the full entitlements map, then make their own authorization decisions from the claims. The gateway also accepts any Active key’s token as a bearer on its own routes, enforcing the entitlement for the store or control-plane surface the route resolves to.

Verification is one indexed lookup plus one hash check against a watch-fed in-memory map — the hot path never reads the control plane per request. status.lastSeenAt advances at most once per five minutes per key.

PhaseMeaning
PendingCRD-authored key awaiting mint.
ActiveVerifiable; token works.
RevokedPOST /v2/keys/{keyId}/revoke was called; token refused.
Expiredstatus.expiresAt passed; token refused.

Deleting a VectorStore or Warehouse inerts every entitlement that names it: the keys stay Active for their other entitlements, and the deletion is finalizer-guarded on the target’s side while keys still reference it.

Kubernetes RBAC

CRD authoring makes kubectl a minting surface, so the chart ships roles to delegate key administration without cluster-admin:

ClusterRoleGrants
hevlayer-key-adminFull verbs on apikeys, plus get on delivered token Secrets. Can mint, revoke, and collect tokens.
hevlayer-key-viewerget/list/watch on apikeys. No Secret access — status hashes are one-way, so viewing is audit, not credential access.

Neither role aggregates into the built-in view/edit/admin ClusterRoles: namespace viewer never silently means key viewer. Bindings are the cluster operator’s explicit act; set rbac.keyRoleBindings in Helm values to render them for the single-team case.

Bootstrapping

LAYER_GATEWAY_API_KEY is the bootstrap credential: it mints the first admin key —

spec:
  entitlements:
    layer:
      scopes: [admin]

— after which routine minting uses minted admin keys. Cluster operators can equally bootstrap by applying an ApiKey resource, since CRD authoring needs only kubectl access.

esc