Describe permissions
Authentication doesn't make sense if the APIs can't be protected by a mechanism
takes use of it. Permissions usually take the account from the request and
validate against permission rules.
Permission on the account itself
Let's implement a quite simple logic: a user can update and delete himself.
Create a file named schema.teo
with this content.
connector {
provider : .sqlite,
url : "sqlite:./database.sqlite"
}
server {
bind : ( "0.0.0.0" , 5053 )
}
@identity.tokenIssuer ( $identity . jwt ( expired : 3600 * 24 * 365 ))
@identity.jwtSecret ( ENV [ "JWT_SECRET" ]!)
@canMutate ( $when (.update | .delete , $match ( $account , [
$case ( $cast ( type User ) , $is ( $self )). asAny ,
$case ( $cast ( type Null ) , $invalid . asAny )
]) , otherwise : $valid . asAny ))
model User {
@id @autoIncrement @readonly
id: Int
@unique @onSet ( $if ( $presents , $isEmail )) @identity.id
email: String
@writeonly @onSet ( $presents . bcrypt . salt )
@identity.checker ( $get (.value). presents . bcrypt . verify ( $self . get (.password). presents ))
password: String
name: String
include handler identity.signIn
include handler identity.identity
}
middlewares [ identity . identityFromJwt ( secret : ENV [ "JWT_SECRET" ]!)]
Create a file named .env
.
JWT_SECRET = my_top_secret
Start the server.
cargo teo serve
Create a user and try to update the user. Send this JSON input to
/User/create
.
{
"create" : {
"email" : "john@gmail.com" ,
"password" : "Aa123456" ,
"name" : "John"
}
}
Hide HTTP response
{
"data" : {
"id" : 1 ,
"email" : "john@gmail.com" ,
"name" : "John"
}
}
Send this to /User/update
.
{
"where" : {
"id" : 1
} ,
"update" : {
"name" : "John Larrison"
}
}
Hide HTTP response
{
"error" : {
"type" : "Unauthorized" ,
"message" : "input is invalid"
}
}
You get a unauthorized error since you are not signed in. Sign in with the
account. Send this to /User/signIn
.
{
"credentials" : {
"email" : "john@gmail.com" ,
"password" : "Aa123456"
}
}
Hide HTTP response
{
"data" : {
"id" : 1 ,
"email" : "john@gmail.com" ,
"name" : "John"
} ,
"meta" : {
"token" : "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJpZCI6eyJpZCI6MX0sIm1vZGVsIjpbIlVzZXIiXSwiZXhwIjoxNzQyNjc4MTA5fQ.qCubhmJGJzXqq4U5HDaTqHtFYYuzm3akklggAKugQ7Y"
}
}
Update the user again with header authorization
set to Bearer #token#
.
{
"where" : {
"id" : 1
} ,
"update" : {
"name" : "John Larrison"
}
}
Hide HTTP response
{
"data" : {
"id" : 1 ,
"email" : "john@gmail.com" ,
"name" : "John Larrison"
}
}
This time, Teo detects the correct user identity and updates the user.
Permission on direct owned models
Update schema.teo
with this content. This time we define a new model Post
.
connector {
provider : .sqlite,
url : "sqlite:./database.sqlite"
}
server {
bind : ( "0.0.0.0" , 5053 )
}
@identity.tokenIssuer ( $identity . jwt ( expired : 3600 * 24 * 365 ))
@identity.jwtSecret ( ENV [ "JWT_SECRET" ]!)
@canMutate ( $when (.update | .delete , $match ( $account , [
$case ( $cast ( type User ) , $is ( $self )). asAny ,
$case ( $cast ( type Null ) , $invalid . asAny )
]) , otherwise : $valid . asAny ))
model User {
@id @autoIncrement @readonly
id: Int
@unique @onSet ( $if ( $presents , $isEmail )) @identity.id
email: String
@writeonly @onSet ( $presents . bcrypt . salt )
@identity.checker ( $get (.value). presents . bcrypt . verify ( $self . get (.password). presents ))
password: String
name: String
@relation ( fields : .id, references : .userId)
posts: Post []
include handler identity.signIn
include handler identity.identity
}
@canMutate ( $match ( $account , [
$case ( $cast ( type User ) , $get (.id). eq ( $self . get (.userId). presents )). asAny ,
$case ( $cast ( type Null ) , $invalid . asAny )
]))
model Post {
@id @autoIncrement @readonly
id: Int
title: String
content: String
@foreignKey
userId: Int
@relation ( fields : .userId, references : .id)
user: User
}
middlewares [ identity . identityFromJwt ( secret : ENV [ "JWT_SECRET" ]!)]
Look at the new code, a post can be created, updated and deleted by the owner.
Other accounts cannot mutate this post. Feel free to create posts with
different users and try with the API requests.
Admin's permissions
It's common that a web platform has a lot of admins. Admins have different
roles. Replace the schema with this.
connector {
provider : .sqlite,
url : "sqlite:./database.sqlite"
}
server {
bind : ( "0.0.0.0" , 5053 )
}
@identity.tokenIssuer ( $identity . jwt ( expired : 3600 * 24 * 365 ))
@identity.jwtSecret ( ENV [ "JWT_SECRET" ]!)
@canMutate ( $when (.update | .delete , $match ( $account , [
$case ( $cast ( type User ) , $is ( $self )). asAny ,
$case ( $cast ( type Admin ) , $valid . asAny ),
$case ( $cast ( type Null ) , $invalid . asAny )
]) , otherwise : $valid . asAny ))
model User {
@id @autoIncrement @readonly
id: Int
@unique @onSet ( $if ( $presents , $isEmail )) @identity.id
email: String
@writeonly @onSet ( $presents . bcrypt . salt )
@identity.checker ( $get (.value). presents . bcrypt . verify ( $self . get (.password). presents ))
password: String
name: String
@relation ( fields : .id, references : .userId)
posts: Post []
include handler identity.signIn
include handler identity.identity
}
@canMutate ( $match ( $account , [
$case ( $cast ( type User ) , $get (.id). eq ( $self . get (.userId). presents )). asAny ,
$case ( $cast ( type Null ) , $invalid . asAny ),
$case ( $cast ( type Admin ) , $when (.delete , $valid , otherwise : $invalid ). asAny )
]))
model Post {
@id @autoIncrement @readonly
id: Int
title: String
content: String
@foreignKey
userId: Int
@relation ( fields : .userId, references : .id)
user: User
}
enum Role {
root
normal
}
@identity.tokenIssuer ( $identity . jwt ( expired : 3600 * 24 * 365 ))
@identity.jwtSecret ( ENV [ "JWT_SECRET" ]!)
@canMutate ( $match ( $account , [
$case ( $cast ( type Admin ) , $any ([
$get (.role). presents . eq (.root). asAny ,
$is ( $self ). asAny
])). asAny ,
]))
@canRead ( $account . cast ( type Admin ))
model Admin {
@id @autoIncrement @readonly
id: Int
@unique @onSet ( $if ( $presents , $isEmail )) @identity.id
email: String
@writeonly @onSet ( $presents . bcrypt . salt )
@identity.checker ( $get (.value). presents . bcrypt . verify ( $self . get (.password). presents ))
password: String
role: Role
name: String
}
middlewares [ identity . identityFromJwt ( secret : ENV [ "JWT_SECRET" ]!)]
Let's describe the rule that we just newly created.
Admin can mutate users
Admin can delete a post if the post contains illegal or offensive content
Only admin can read admins
Only the normal admin himself and root admins can mutate the admin record
Indirect permissions
We cannot simply describe the indirect permissions with the pipeline items.
Let's write some programmatic code to do this. Let's say, there are many
projects belongs to teams, and a user can join any team. A team has many users.
Only team user can read or mutate the projects. Let's transform our thoughts
to code like this.
connector {
provider : .sqlite,
url : "sqlite:./database.sqlite"
}
server {
bind : ( "0.0.0.0" , 5053 )
}
entity {
provider : .rust,
dest : "./src/entities"
}
@identity.tokenIssuer ( $identity . jwt ( expired : 3600 * 24 * 365 ))
@identity.jwtSecret ( ENV [ "JWT_SECRET" ]!)
@canMutate ( $when (.update | .delete , $match ( $account , [
$case ( $cast ( type User ) , $is ( $self )). asAny ,
$case ( $cast ( type Admin ) , $valid . asAny ),
$case ( $cast ( type Null ) , $invalid . asAny )
]) , otherwise : $valid . asAny ))
model User {
@id @autoIncrement @readonly
id: Int
@unique @onSet ( $if ( $presents , $isEmail )) @identity.id
email: String
@writeonly @onSet ( $presents . bcrypt . salt )
@identity.checker ( $get (.value). presents . bcrypt . verify ( $self . get (.password). presents ))
password: String
name: String
@relation ( fields : .id, references : .userId)
posts: Post []
@relation ( through : TeamUser , local : .user, foreign : .team)
teams: Team []
include handler identity.signIn
include handler identity.identity
}
@canMutate ( $match ( $account , [
$case ( $cast ( type User ) , $get (.id). eq ( $self . get (.userId). presents )). asAny ,
$case ( $cast ( type Null ) , $invalid . asAny ),
$case ( $cast ( type Admin ) , $when (.delete , $valid , otherwise : $invalid ). asAny )
]))
model Post {
@id @autoIncrement @readonly
id: Int
title: String
content: String
@foreignKey
userId: Int
@relation ( fields : .userId, references : .id)
user: User
}
enum Role {
root
normal
}
@identity.tokenIssuer ( $identity . jwt ( expired : 3600 * 24 * 365 ))
@identity.jwtSecret ( ENV [ "JWT_SECRET" ]!)
@canMutate ( $match ( $account , [
$case ( $cast ( type Admin ) , $any ([
$get (.role). presents . eq (.root). asAny ,
$is ( $self ). asAny
])). asAny ,
]))
@canRead ( $account . cast ( type Admin ))
model Admin {
@id @autoIncrement @readonly
id: Int
@unique @onSet ( $if ( $presents , $isEmail )) @identity.id
email: String
@writeonly @onSet ( $presents . bcrypt . salt )
@identity.checker ( $get (.value). presents . bcrypt . verify ( $self . get (.password). presents ))
password: String
role: Role
name: String
}
model Team {
@id @autoIncrement @readonly
id: Int
name: String
@relation ( through : TeamUser , local : .team, foreign : .user)
users: User []
@relation ( fields : .id, references : .teamId)
projects: Project []
}
@id ([.teamId, .userId])
model TeamUser {
@foreignKey
userId: Int
@foreignKey
teamId: Int
@relation ( fields : .userId, references : .id)
user: User
@relation ( fields : .teamId, references : .id)
team: Team
}
declare pipeline item canReadOrMutateProject : User -> Ignored
@canRead ( $account . cast ( type User ). canReadOrMutateProject )
@canMutate ( $account . cast ( type User ). canReadOrMutateProject )
model Project {
@id @autoIncrement @readonly
id: Int
@foreignKey
teamId: Int
@relation ( fields : .teamId, references : .id)
team: Team
}
middlewares [ identity . identityFromJwt ( secret : ENV [ "JWT_SECRET" ]!)]
Transform the current directory into a project.
cargo init . --bin
Replace content of Cargo.toml
with this.
[package]
name = "hello-teo-permissions"
version = "0.1.0"
edition = "2021"
[dependencies]
teo = { version = "0.3.9" }
tokio = { version = "1.36" }
Let's generate the entity from the schema for programming.
cargo teo generate entity
Create the main program file src/main.rs
.
pub mod entities;
use entities :: { Teo , User , Project };
use tokio :: main;
use teo :: prelude :: {pipeline :: item :: validator :: Validity , App , Result , Error , teon};
#[main]
async fn main () -> Result <()> {
let app = App :: new () ? ;
app . main_namespace () . define_validator_pipeline_item ( "canReadOrMutateProject" , move | user : User , project : Project , teo : Teo | async move {
if teo . team_user () . find_unique ( teon! ({
"where" : {
"teamId" : project . team_id (),
"userId" : user . id (),
}
})) .await?. is_none () {
Ok :: < Validity , Error >( Validity :: Invalid ( "user doesn't belong to this project's team" . to_owned ()))
} else {
Ok ( Validity :: Valid )
}
});
app . run () .await
}
Now start the server with the app entrance. Try create, update and read with
different user accounts.
cargo run -- serve
Try to create a project on a team that the user doesn't belong to, will cause
an error.
{
"create" : {
"teamId" : 2
}
}
Hide HTTP response
{
"error" : {
"type" : "Unauthorized" ,
"message" : "user doesn't belong to this project's team"
}
}
Summary
Like anything else in Teo, permissions are readable and clear, too. Developing
protected APIs and defining permission rules are quite fast and easy.