docs/content/Deploying/Access-Control.mdx
import { Callout } from 'nextra/components'
DocsGPT has two independent permission planes:
admin or user role for the whole instance. Admins can manage other users and see instance-wide usage and audit data.team_admin or team_member, and resources can be shared to teams or individual members.The two planes never mix: a global admin is a superuser over all teams, but a team_admin is not a global admin.
| Role | Capabilities |
|---|---|
user | The default. Owns their own conversations, sources, agents, prompts, and tools. |
admin | Everything a user can do, plus the admin dashboard: user management, force-logout, role management, and instance-wide usage/audit. |
To see the roles the current request resolves to, call:
GET /api/user/me → { "user_id": "...", "roles": ["user"], "email": "...", "name": "...", "picture": "..." }
This is the canonical way to check a caller's effective roles (distinct from /api/config, which only reports the instance auth_type).
There are four ways an account becomes an admin:
grant_admin script — run python scripts/grant_admin.py <user_id> on the server to grant (or --revoke / --list) the admin role directly. This is the canonical bootstrap for the first admin.OIDC_ADMIN_GROUPS auto-grants admin to members of the listed IdP groups. See below.LOCAL_MODE_ADMIN=true grants admin when AUTH_TYPE=None (self-host, no authentication).| Setting | Default | Description |
|---|---|---|
OIDC_ADMIN_GROUPS | — | Comma-separated IdP groups whose members are granted the global admin role. Re-checked at every login and silent renewal. Unset = no OIDC admin mapping. |
LOCAL_MODE_ADMIN | false | Grants admin in no-auth mode only (AUTH_TYPE=None). |
Granting admin through the dashboard itself requires already being an admin, which creates a chicken-and-egg problem on a fresh deployment. Break it one of these ways:
python scripts/grant_admin.py <user_id> (the user_id is the OIDC sub). The grant is written to user_roles with source='manual' and takes effect on the user's next request. Use --list to see current admins and --revoke to remove a manual grant.OIDC_ADMIN_GROUPS to a group you belong to, and sign in — you become an admin automatically.LOCAL_MODE_ADMIN=true.When OIDC_ADMIN_GROUPS is set, group membership is mapped to the admin role at every sign-in and every silent renewal — so removing a user from the admin group revokes their admin at the next renewal, just like the sign-in allowlist. It is independent of OIDC_ALLOWED_GROUPS (which controls whether a user may sign in at all). Leaving OIDC_ADMIN_GROUPS unset never mass-revokes admin.
OIDC_ADMIN_GROUPS=platform-admins
# OIDC_GROUPS_CLAIM=groups # only if your IdP uses a different claim name
If your IdP only exposes groups via the userinfo endpoint, DocsGPT backfills them from there during reconciliation, the same as the allowlist.
Admins get a dashboard backed by a REST surface under /api/admin (every endpoint requires the admin role). All mutating actions are written to the auth_events audit log with the acting admin recorded in the metadata.
| Method | Path | Description |
|---|---|---|
GET | /api/admin/overview | Instance overview counts. |
GET | /api/admin/users | List users (paginated; supports a user_id filter). |
GET | /api/admin/users/<id> | Drill into a single user. |
PATCH | /api/admin/users/<id> | Activate / deactivate a user (deactivation revokes their sessions). |
POST | /api/admin/users/<id>/role | Grant the admin role (guards against removing the last admin). |
DELETE | /api/admin/users/<id>/role | Revoke the admin role. |
POST | /api/admin/users/<id>/revoke-sessions | Force-logout a user. |
GET | /api/admin/admins | List current admins. |
GET | /api/admin/usage | Token-usage series and top users. |
GET | /api/admin/audit | Authentication/admin audit feed. |
GET | /api/admin/devices/audit | Remote-device audit feed. |
GET | /api/admin/teams | Instance-wide oversight of all teams. |
Teams let a group of users collaborate and share resources. Teams are self-serve — any user can create one and manage its membership.
| Role | Capabilities |
|---|---|
team_member | Belongs to the team; can use resources shared with the team. |
team_admin | Manages membership and team settings. Implies team_member. |
The team owner is a distinct concept from team_admin: ownership can be transferred, and a global admin is treated as a superuser over every team.
| Method | Path | Description |
|---|---|---|
GET / POST | /api/teams | List your teams / create a team. |
GET / PATCH / DELETE | /api/teams/<id> | Read, update, or delete a team. |
GET / POST | /api/teams/<id>/members | List members / add a member. |
PATCH / DELETE | /api/teams/<id>/members/<member_id> | Change a member's role / remove (or leave). |
POST | /api/teams/<id>/transfer_owner | Transfer ownership. |
Membership and removal are guarded so a team can never be left without an admin (the last-admin guard).
<Callout type="info" emoji="ℹ️"> Members are added by email (or subject id). The invitee **must have signed in at least once** so DocsGPT can resolve them to a user account. </Callout>Four resource types can be shared: agents, sources, prompts, and tools. Sharing is additive — it grants access to others without changing ownership.
| Method | Path | Description |
|---|---|---|
GET / POST / DELETE | /api/teams/<id>/grants | List, create, or revoke a share within a team. |
GET | /api/resource_shares | List shares visible to the caller. |
Sharing rules:
viewer (read-only) or editor (read and modify).editor is not the same as owner — an editor can change a resource but cannot delete it or re-share it.Access-control actions are appended to the auth_events table alongside the authentication events. This includes admin actions — admin_user_activated / admin_user_deactivated, admin_sessions_revoked, role_granted / role_revoked (with metadata.source = manual or oidc_group) — and team events (team.create, team.member_add, team.member_role, team.member_remove, team.share, team.unshare, team.transfer_owner, team.delete). The acting admin is recorded in the event metadata.
auth_events table.