docs/developer/core-concepts/staff-roles.mdx
Admin users manage the store via the Admin Panel. They have roles that control what they can access.
erDiagram
AdminUser ||--o{ RoleUser : "has many"
RoleUser }o--|| Role : "belongs to"
RoleUser }o--|| Store : "scoped to"
AdminUser ||--o{ Invitation : "invites"
AdminUser {
string id
string email
}
RoleUser {
string role_id
string resource_type
string resource_id
}
Role {
string name
}
Invitation {
string email
string status
string token
datetime expires_at
}
Admin users can have different roles that control their permissions:
| Role | Description |
|---|---|
admin | Full access to all Admin Panel features |
Use the Spree CLI to create admin users:
spree user create
The CLI will prompt you for the email and password. You can also pass them directly:
spree user create --email [email protected] --password secret123
The created user gets the admin role on the default store.
Staff authenticate against the Admin API, which issues a short-lived JWT used for subsequent requests. How a staff member proves who they are is pluggable — Spree ships email/password out of the box, and you can plug in any external identity provider (Okta, Microsoft Entra ID, Google Workspace, a custom JWT issuer, SAML, etc.) without changing the rest of the API.
A staff member logs in via the POST /api/v3/admin/auth/login endpoint (see Admin API Authentication). The request's provider field selects a registered authentication strategy. When provider is omitted it defaults to email, the built-in email/password strategy (which you can also disable and restrict the admin to your preferred SSO provider).
flowchart TB
A["POST /api/v3/admin/auth/login"] --> B{"provider field"}
B -->|"omitted or email"| C["Built-in EmailPasswordStrategy"]
B -->|"okta, saml, ..."| D["Your custom strategy"]
C --> E["Strategy authenticates, returns the staff user"]
D --> E
E --> F["Spree issues a JWT (aud=admin_api) + HttpOnly refresh cookie"]
Whichever strategy authenticates the request, Spree issues the same credentials in return, so downstream code and the admin SPA never need to know which provider was used:
aud: admin_api), short-lived by design;HttpOnly cookie scoped to /api/v3/admin/auth (the admin flow keeps it out of the response body — see Admin Auth & Cookie Refresh).The same strategy registry exists on the customer side, so storefront sign-in is pluggable the same way — the only difference is the user class and that the Store API returns the refresh token in the body rather than a cookie.
Follow the Custom API Authentication how-to for details how to create a custom authentication strategy and register it with the admin API. Once registered, you can use it from the admin SPA or any API client by passing its name in the provider field of the login request.
You can invite new admins through the Admin Panel or programmatically.
Via Admin Panel:
Programmatically:
Using the Admin SDK, call client.invitations.create:
import { createAdminClient } from '@spree/admin-sdk'
const client = createAdminClient({
baseUrl: 'https://store.example.com',
secretKey: 'sk_xxx',
})
const invitation = await client.invitations.create({
email: '[email protected]',
role_id: 'role_xxx',
})
Creating an invitation publishes invitation.created, which sends the email. Either way, the invitee receives an email with an invitation link. If they already have an account, they log in to accept. Otherwise, they create an account first.
flowchart TB
A[Admin creates invitation] --> B[Invitation email sent]
B --> C[Invitee clicks link]
C --> D{Has account?}
D -->|Yes| E[Log in]
D -->|No| F[Create account]
E --> G[Accept invitation]
F --> G
G --> H[Role assigned to store]
| Attribute | Description |
|---|---|
email | Invitee's email address |
token | Secure token for the invitation link |
status | pending or accepted |
expires_at | Expiration date (default: 2 weeks) |
resource | The store being granted access to |
role | The role to assign upon acceptance |
The invitation system publishes events you can subscribe to:
| Event | Description |
|---|---|
invitation.created | Invitation was created (triggers email) |
invitation.accepted | Invitation was accepted and role assigned |
invitation.resent | Invitation was resent to the invitee |
Spree uses CanCanCan for authorization. Permissions apply to both customers (Store API access) and admins (Admin Panel access).
See the Customize Permissions guide for details on creating custom roles and permission sets.