A compile-time enforced authorization framework for Rust applications.
In typical applications, authorization checks are performed in potentially random segments of the code. This leads to implicit assumptions on what kinds of permissions or checks have been enforced at various parts of the codebase. For example:
```rust
fn handler(req: Request) -> Result
// makes no assumptions on a user's permissions or access, initially
fn privileged_fn(user: User) -> Result
// action otherprivilegedfn(user) }
// Implicitly depends on user having the "SimplePermissions" permission or role.
fn otherprivilegedfn(user: User) -> Result
// other action Ok(()) } ```
DACquiri does things differently.
With DACquiri, you explicitly declare your authorization requirements in the function definition. DACquiri will, at compile-time, enforce all code-paths invoking your function will have checked the appropriate authorization requirements beforehand.
With DACquiri, you:
Missing an authorization check? That's a compile-time error.
Missing DACquiri? That's a dev-time error.
DACquiri codifies permissions checks into the type system using a wrapper struct called GrantChain
. For example, let's imagine you have two permissions called P1
and P2
. If you've checked both of these permissions on some User
object, you might expect to now have a type GrantChain<P2, GrantChain<P1, User>>
.
The magic of DACquiri is that it doesn't matter in which order you check permissions, just that you've checked them at some point. Regardless of the order, the outer GrantChain
will implement both HasGrant<P1>
as well as HasGrant<P2>
. This is true for no matter how many grants you add to the chain.
Grants can be checked with the try_grant
function where you'll specify which Grant
you are currently checking. The actual check is performed in the has_grant
function you must implement when implementing Grant
on your PrincipalT
.
Here's a simplistic example of two permissions (PermissionOne
and PermissionTwo
) that we'll define as being grantable to all User
objects (for the sake of this example).
```rust use dacquiri::prelude::*;
struct User { name: String }
fn checkpermissionone(user: user) -> GrantResult<()> { Ok(()) }
fn checkpermissiontwo(user: User) -> GrantResult<()> { Ok(()) }
pub trait RequiresPermissionOne {}
pub trait RequiresPermissionTwo {}
pub trait RequiresBothPermissions {}
fn requirespermissionone(caller: &impl RequiresPermissionOne) { println!("The caller must have checked that you have PermissionOne"); }
fn requirespermissiontwo(caller: &impl RequiresPermissionTwo) { println!("The caller must have checked that you have PermissionTwo"); }
fn requiresbothpermission(caller: impl RequiresBothPermissions) { println!("The caller must have checked that you had both PermissionOne and PermissionTwo"); }
fn main() -> Result<(), String> { let user = User { name: format!("d0nut") };
let p1_user = user.try_grant::<PermissionOne, _>(())?;
requires_permission_one(&p1_user);
let p2_user = p1_user.try_grant::<PermissionTwo, _>(())?;
requires_permission_two(&p2_user);
requires_both_permission(p2_user);
} ```