Skip to main content

Authentication

Introductionโ€‹

Authentication is the process of proving that people are who they say they are.

Manifest uses JSON Web Tokens (JWT) to do that. When you log in, you basically create a new token that you use in your next requests to prove your identity. This allows us to use Policies to grant or deny the access to some resources based on the user characteristics.

info

Notice the TOKEN_SECRET_KEY variable in your .env file ? This is the key that will encrypt your tokens, you can generate one here.

Adminsโ€‹

Admins are a built-in entity that are the only ones with access to the admin panel (located at http://localhost:1111 by default). The admins are usually the people who manage the application on a day-to-day basis. Only admins can see and manage other admins.

Even though they are the most powerful users of your application, you can still create some policies that will restrict the access even for them.

The seed command will create an admin with the email admin@manifest.build and the password admin. You can create more admins from the admin panel.

tip

In Manifest, the admin panel is non-technical ๐Ÿ˜บ.

It means that you can give credentials to the administrators of your app without worrying that they will end up breaking the system!

Authenticable entitiesโ€‹

You can convert any entity into an authenticatable entity, allowing users to log in with it.

entities:
Patient ๐Ÿค’:
authenticable: true # Makes entity authenticable.
properties:
- name

Authenticable entities have 2 extra properties that are used as credentials to log in: email and password. You do not need to specify them.

The passwords are automatically hashed using bcryt with 10 salt rounds.

Syntaxโ€‹

Loginโ€‹

Log in your credentials as an admin or an authenticable entity.

Example HTTP Request
POST /api/auth/admins/login
Content-Type: application/json
{
"email": "admin@manifest.build",
"password": "password"
}
Example HTTP Response
{
"token": "12345"
}

Then you can add your token to your requests Authorization header using the Bearer prefix:

GET /api/dynamic/cats
Content-Type: application/json
Authorization: Bearer your-token-here

Sign upโ€‹

Any authenticable entity allows new users to sign up if the policies permit it.

Example HTTP Request
POST /api/auth/users/signup
Content-Type: application/json
{
"email": "user@example.com",
"password": "password"
}
Example HTTP Response
{
"token": "12345"
}

Same as login, you can add your token to your requests Authorization header using the Bearer prefix:

GET /api/dynamic/cats
Content-Type: application/json
Authorization: Bearer your-token-here
info

It is not possible to sign up as an admin. If you want to create more admins, do it from the admin panel.

Get current userโ€‹

Get the current logged-in user.

Example HTTP Request
GET /api/auth/contributors/me
Content-Type: application/json
Authorization: Bearer your-token-here
Example HTTP Response
{
id: 1,
email: 'contributor@test.com'
}

Logoutโ€‹

Logout removes the token from future request headers.

Reset the Authorization header as you usually do, and you are good to go!

API Policiesโ€‹

API Policies ensure that we provide specific access to resources for users. They are a way to implement Authorization following the RBAC (Role-Based Access Control) method. Indeed it is possible to create different entities (ex: User, Manager...) with different access to resources.

Policies can be added to entities or endpoints.

Entity rulesโ€‹

Each entity has 5 rules where one or several access policies can be applied:

  • create: create a new item
  • read: see the detail and the list of items
  • update: update an existing item
  • delete: delete an existing item
  • signup: sign up as a new user (only for authenticable entities)

Access policiesโ€‹

The policies for each rule can be added to each entity description as shown below:

entities:
Invoice ๐Ÿงพ:
properties:
- number
- { name: issueDate, type: date }
policies:
create:
- { access: restricted, allow: User }
update:
- access: admin
delete:
- access: forbidden

In this case, everyone can see the Invoice items, only logged-in Users can create new ones. Updating an Invoice is restricted to Admins only and no one can delete them (not even Admins).

warning

If no policy is specified for a rule, the access is public for the related action, thus anyone can manage records.

The only exception is the update of single entities that is set to admin by default for convenience.

PropDescriptionType
accessThe type of access: public, restricted, admin, forbiddenAccessType
allowOnly for restricted access: the entity (or entities) that have accessstring | string[]

Access typesโ€‹

There are 4 possible access types:

AccessDescriptionShort version (emoji)
publicEveryone has access๐ŸŒ
restrictedOnly logged-in users have access to it. If allow key specifies one or several entities, users logged in as other entities will not have access. Admins always have access to restricted rules๐Ÿ”’
adminOnly admins have access๐Ÿ‘จ๐Ÿปโ€๐Ÿ’ป
forbiddenNo one has access, not even admins๐Ÿšซ

Additional examplesโ€‹

entities:
Project ๐Ÿ—‚๏ธ:
properties:
- name
policies:
read:
- { access: restricted, allow: [Contributor, Manager] } # Only some entities (and admins).
create:
- { access: restricted, allow: Manager } # Only managers (and admins).
update:
- access: ๐Ÿ‘จ๐Ÿปโ€๐Ÿ’ป # Only admin.
delete:
- access: ๐Ÿšซ # Forbidden (no one).

Contributor ๐Ÿ‘จโ€๐Ÿ’ผ:
authenticable: true
properties:
- name
policies:
signup:
- access: ๐Ÿšซ # Forbidden (no one).
create:
- { access: ๐Ÿ”’, allow: Manager } # Managers can create contributors.
update:
- { access: ๐Ÿ”’, allow: Manager }
delete:
- { access: ๐Ÿ”’, allow: Manager } # Managers can create contributors.