gatekeep
The soul selects her own society, Then shuts the door;
— Emily Dickinson, "Exclusion" (1890)
gatekeep is a code-first authorization engine for Rust. Policies are ordinary
Rust values, a pure deterministic core evaluates them, and every decision
carries the reasons that produced it.
It is the sibling of keepsake: keepsake keeps relation
lifecycle state — entitlements, holds, sanctions, risk flags, gates — and
gatekeep decides what those facts permit. The two compose but stay independent crates.
Where it fits
Use gatekeep for an in-process authorization boundary authored in Rust, by the team that owns the application. Policies are composable combinators over a frozen set of facts, evaluated synchronously with no IO, so a decision replays exactly from its inputs.
It is not a policy DSL, an authentication or session layer, or a network policy service; those stay with the application or with crates built for them. Because each policy is reified as inspectable data, gatekeep can serialize, hash, diff, and explain a decision. It can also answer "which resources can this principal reach?", not just "may this principal reach this one?".
Usage
use ;
// Outcome grade: how much of a record the caller may read.
// A fact the application resolves before evaluation.
;
Partial evaluation reuses the same policy value with PartialFacts: mark
request-known facts as present or absent, leave resource-level facts unknown,
then lower the residual policy in an application-owned adapter. For SQL-backed
list queries, gatekeep-sqlx maps residual facts to trusted row predicates and
appends a lowered filter and grade projection to a sqlx::QueryBuilder.
Postgres is the default backend; SQLite and MySQL are available behind feature
flags.
For the lowering walkthrough, see the gatekeep-sqlx docs on
docs.rs and the
axum-authorized-list and
axum-keepsake-authorized-list
examples, which resolve request facts in-process and from keepsake.
gatekeep-keepsake resolves gatekeep facts from active keepsake relations. The
default resolver maps the request principal to a keepsake subject, and bindings
can target additional request-scoped subjects through SubjectSlot values in
Context::subjects. Use that for facts attached to something other than the
principal, such as a skill version, repository, account, or source identity. A
missing subject slot is reported as ResolveError::MissingSubject, distinct
from an unbound or unproducible fact.
Why it exists
The Rust authz ecosystem leans on external DSLs. A policy DSL is worth its overhead across many services and for non-engineer authors; for a single Rust service it mostly adds cost: a second language, the domain re-encoded as entities and attributes, and typos that fail at runtime instead of compile time. gatekeep keeps policies in Rust and reifies them as data, so they stay analyzable.
License
Licensed under either of:
- Apache License, Version 2.0
- MIT license
at your option.