Skip to main content
AI/MLagentfront

frontmcp-authorities

Use when implementing authorization, access control, RBAC, ABAC, or ReBAC for tools, resources, or prompts. Covers JWT claims mapping, authority profiles, and policy enforcement.

Stars
144
Source
agentfront/frontmcp
Updated
2026-05-30
Slug
agentfront--frontmcp--frontmcp-authorities
View on GitHubRaw SKILL.md

// install — copy + paste into any project

mkdir -p .claude/skills && curl -fsSL https://raw.githubusercontent.com/agentfront/frontmcp/HEAD/libs/skills/catalog/frontmcp-authorities/SKILL.md -o .claude/skills/frontmcp-authorities.md

Drops the SKILL.md into .claude/skills/frontmcp-authorities.md. Works with Claude Code, Cursor, and any agent that loads SKILL.md files from .claude/skills/.

FrontMCP Authorities

Built-in RBAC/ABAC/ReBAC authorization system for FrontMCP entry types. Each flow has native checkEntryAuthorities and filterByAuthorities stages that enforce access control policies declared via the authorities field on entry decorators. Configured via @FrontMcp({ authorities: { claimsMapping, profiles, scopeMapping } }) — no plugin needed. Flow stages handle enforcement, and developers can hook into them with Will, Did, and Around decorators. Supports named profiles for reuse, JWT claims mapping for any identity provider, inline policies with roles/permissions/attributes/relationships, and composable combinators (allOf, anyOf, not).

When to Use This Skill

Must Use

  • Adding role-based or permission-based access control to tools, resources, or prompts
  • Restricting MCP entry visibility based on the caller's JWT claims
  • Enforcing tenant isolation (ABAC) or relationship checks (ReBAC) on entries

Recommended

  • Setting up a multi-tenant server where different tenants see different tools
  • Building an admin vs. user distinction across your MCP surface
  • Combining multiple authorization models (e.g., RBAC + ABAC) on the same entry

Skip When

  • You only need authentication (login/token validation) without authorization (see frontmcp-config / configure-auth)
  • You are building a public server with no access restrictions (use mode: 'public')
  • You need OAuth scopes at the transport level, not entry-level policies (see configure-auth-modes)

Decision: Use this skill whenever you need to control who can access which entries based on roles, permissions, attributes, or relationships.

CRITICAL: Ask About JWT Shape First

Before writing any authorities configuration, the coding agent MUST ask the developer:

"What identity provider (IdP) are you using, and what does your JWT payload look like? I need to know where roles, permissions, and tenant ID are located in the claims."

Why this matters: Every IdP places roles and permissions in different JWT claim paths. Auth0 uses namespaced URIs (https://myapp.com/roles), Keycloak nests them under realm_access.roles, Okta uses groups, Cognito uses cognito:groups, and Frontegg uses flat roles/permissions. Writing claimsMapping without knowing the actual token shape will produce silent authorization failures where every user is denied.

What to collect before proceeding:

  1. Identity provider name (Auth0, Keycloak, Okta, Cognito, Frontegg, custom)
  2. A sample decoded JWT payload (redacted sensitive values)
  3. The claim path for roles (e.g., realm_access.roles)
  4. The claim path for permissions (e.g., permissions or scope)
  5. The claim path for tenant/org ID if multi-tenant (e.g., org_id, tenantId)

See references/claims-mapping.md for IdP-specific claim paths.

Prerequisites

  • FrontMCP SDK installed (@frontmcp/sdk)
  • @frontmcp/auth available (peer dependency of SDK, provides all authorities types)
  • An authentication mode configured (see frontmcp-config / configure-auth-modes) so that authInfo is populated on incoming requests
  • Knowledge of the developer's JWT token structure (see critical section above)

Steps

Step 1: Add the Authorities Config

Add the authorities field to your @FrontMcp decorator. No plugin import needed — authorities is a built-in framework feature.

import { FrontMcp } from '@frontmcp/sdk';

@FrontMcp({
  name: 'my-server',
  authorities: {
    // configured in next steps
  },
})
export class MyServer {}

Step 2: Configure JWT Claims Mapping

Set claimsMapping to tell the engine where roles, permissions, and user/tenant identifiers live in your IdP's JWT. Each value is a dot-path into the decoded JWT claims object.

// Keycloak example — in @FrontMcp({ authorities: { ... } })
authorities: {
  claimsMapping: {
    roles: 'realm_access.roles',
    permissions: 'resource_access.my-client.roles',
    tenantId: 'org_id',
    userId: 'sub',
  },
}

// Auth0 example
authorities: {
  claimsMapping: {
    roles: 'https://myapp.com/roles',
    permissions: 'permissions',
    tenantId: 'org_id',
  },
}

If no claimsMapping is provided, the engine falls back to (in order):

  • roles: authInfo.user.rolesauthInfo.extra.authorization.scopes[]
  • permissions: authInfo.user.permissions[]

The OAuth-scopes-as-roles fallback means a token with no roles claim but with scope: "admin read" will be treated as having roles: ['admin', 'read']. Configure explicit claimsMapping.roles to opt out. For non-standard token shapes, use claimsResolver instead (see Common Patterns below).

Step 3: Register Named Profiles

Profiles let you define reusable authorization policies and reference them by name in decorators. Register them in the profiles field.

// In @FrontMcp({ authorities: { ... } })
authorities: {
  claimsMapping: { roles: 'realm_access.roles', permissions: 'scope' },
  profiles: {
    admin: {
      roles: { any: ['admin', 'superadmin'] },
    },
    authenticated: {
      attributes: {
        conditions: [{ path: 'user.sub', op: 'exists', value: true }],
      },
    },
    matchTenant: {
      attributes: {
        conditions: [
          { path: 'claims.org_id', op: 'eq', value: { fromInput: 'tenantId' } },
        ],
      },
    },
    editor: {
      permissions: { any: ['content:write', 'content:publish'] },
    },
  },
}

For type-safe profile names, augment the global interface:

declare global {
  interface FrontMcpAuthorityProfiles {
    admin: true;
    authenticated: true;
    matchTenant: true;
    editor: true;
  }
}

Step 4: Add Authorities to Entries

Use the authorities field on any entry decorator (@Tool, @Resource, @Prompt, @Skill). Three forms are supported:

String (profile reference):

@Tool({ name: 'delete_user', authorities: 'admin' })
export default class DeleteUserTool extends ToolContext { ... }

String array (multiple profiles, AND semantics):

@Tool({ name: 'update_tenant_settings', authorities: ['authenticated', 'matchTenant'] })
export default class UpdateTenantSettingsTool extends ToolContext { ... }

Inline policy object:

@Tool({
  name: 'publish_content',
  authorities: {
    roles: { any: ['editor', 'admin'] },
    permissions: { all: ['content:publish'] },
  },
})
export default class PublishContentTool extends ToolContext { ... }

Combinators for complex policies:

@Tool({
  name: 'sensitive_action',
  authorities: {
    allOf: [
      { roles: { any: ['admin'] } },
      { attributes: { conditions: [{ path: 'env.NODE_ENV', op: 'eq', value: 'production' }] } },
    ],
  },
})
export default class SensitiveActionTool extends ToolContext { ... }

Async guards (one-shot DB/Redis/API checks):

For dynamic, async authorization that does not warrant a reusable custom evaluator, use the guards field. Each guard receives the same AuthoritiesEvaluationContext and returns true on grant, or false/a denial string on deny. Guards run in sequence and combine with other policy fields via operator (default AND).

import type { AuthorityGuardFn } from '@frontmcp/auth';

const requireActiveSubscription: AuthorityGuardFn = async (ctx) => {
  const active = await db.isSubscriptionActive(ctx.user.sub);
  return active ? true : 'subscription is not active';
};

@Tool({
  name: 'premium_feature',
  authorities: {
    roles: { any: ['user'] },
    guards: [requireActiveSubscription],
  },
})
export default class PremiumFeatureTool extends ToolContext { ... }

Use guards for one-off async checks; promote to a registered custom evaluator when the same logic is reused across many entries (see references/custom-evaluators.md).

Optional: Map Denials to OAuth Scope Challenges (scopeMapping)

If your transport layer issues OAuth scope challenges (RFC 6750 insufficient_scope), declare a scopeMapping so authority denials are converted into the right WWW-Authenticate challenge with the required scopes. Mapping is explicit only — no automatic permission-to-scope inference.

authorities: {
  scopeMapping: {
    roles: { admin: ['admin:all'] },
    permissions: { 'repo:write': ['repo'] },
    profiles: { admin: ['admin:all'] },
  },
}

When a request is denied because the user lacks role admin, the response surfaces the admin:all scope as the required challenge.

Optional: Extract Custom Auth Context Fields (pipes)

pipes are functions that run during auth context construction and merge their output into FrontMcpAuthContext. Use them to extract custom typed fields from JWT claims so they are available to your tools as strongly typed accessors. Declare the resulting fields by augmenting ExtendFrontMcpAuthContext:

A pipe receives the raw JWT claims (a Readonly<Record<string, unknown>>) and returns a Partial<ExtendFrontMcpAuthContext> (sync or async). It does not receive the AuthInfo envelope — there is no .user accessor on the input.

// Pipe signature (defined inline; do not import — AuthContextPipe is not
// re-exported from `@frontmcp/auth`):
//   (claims: Readonly<Record<string, unknown>>) =>
//     Partial<ExtendFrontMcpAuthContext> | Promise<Partial<ExtendFrontMcpAuthContext>>

const tenantPipe = (claims: Readonly<Record<string, unknown>>) => ({
  tenantId: claims['tenantId'] as string | undefined,
});

declare global {
  interface ExtendFrontMcpAuthContext {
    tenantId?: string;
  }
}

// In @FrontMcp({ authorities: { ... } })
authorities: {
  pipes: [tenantPipe],
}

Scenario Routing Table

Scenario Approach Reference
Simple role gate (admin-only tool) authorities: 'admin' profile references/authority-profiles.md
Permission-based access authorities: { permissions: { all: ['x'] } } references/rbac-abac-rebac.md
Tenant isolation ABAC with { fromInput: 'tenantId' } references/rbac-abac-rebac.md
Resource ownership check ReBAC with relationship resolver references/rbac-abac-rebac.md
IP allowlist or custom logic Custom evaluator via custom.* references/custom-evaluators.md
Different IdP (Auth0/Keycloak/Okta) Configure claimsMapping references/claims-mapping.md
Admin OR (editor AND same-tenant) anyOf / allOf combinators references/authority-profiles.md
Custom pre/post authority logic Hook with Will/Did/Around on checkEntryAuthorities stage references/custom-evaluators.md
Replace built-in check with OPA/Cedar Around('checkEntryAuthorities') hook references/custom-evaluators.md
Audit authority decisions Did('checkEntryAuthorities') hook for logging/metrics references/custom-evaluators.md
Tenant allowlist in Redis/DB Async custom evaluator with custom.* field references/custom-evaluators.md
Subscription check before tool runs Async custom evaluator or Will('checkEntryAuthorities') hook references/custom-evaluators.md
Feature flag gate on a tool Async custom evaluator checking flag service references/custom-evaluators.md

Common Patterns

Pattern Correct Incorrect Why
Claims mapping claimsMapping: { roles: 'realm_access.roles' } claimsMapping: { roles: 'roles' } (for Keycloak) Keycloak nests roles under realm_access.roles; using the wrong path silently resolves to []
Profile reference authorities: 'admin' authorities: { profile: 'admin' } Profiles are referenced as plain strings, not nested objects
Multiple profiles authorities: ['authenticated', 'matchTenant'] authorities: 'authenticated, matchTenant' Use an array, not a comma-separated string
Inline RBAC { roles: { any: ['admin'] } } { roles: ['admin'] } roles expects an object with all and/or any arrays
Dynamic value ref { fromInput: 'tenantId' } '{{ tenantId }}' Use DynamicValueRef objects, not template strings
OR combinator { anyOf: [policy1, policy2] } { operator: 'OR', ...policy1, ...policy2 } Use anyOf for clarity; operator applies to top-level fields within a single policy
ReBAC resolver Register relationshipResolver in plugin options Inline the DB query in the policy The resolver is a separate interface; policies only declare the relationship to check

Verification Checklist

Configuration

  • authorities: { ... } is set on the @FrontMcp decorator
  • claimsMapping paths match the actual JWT token structure from your IdP
  • All profile names referenced in authorities: 'name' are registered in profiles
  • Authentication mode is configured (not public) so authInfo is populated
  • If using ReBAC, relationshipResolver is provided in plugin options

Runtime

  • Unauthenticated requests to protected entries receive a denial error
  • Users with the correct roles/permissions can access their entries
  • tools/list response omits entries the caller is not authorized to see
  • ABAC conditions with { fromInput: ... } resolve correctly from tool arguments
  • Custom evaluators return proper AuthoritiesResult objects

Type Safety

  • FrontMcpAuthorityProfiles is augmented for type-safe profile references
  • @frontmcp/auth is imported so the authorities field is available on decorators

Troubleshooting

Problem Cause Solution
profile 'admin' is not registered Profile used in decorator but not in authorities.profiles config Add the profile to the profiles field in @FrontMcp({ authorities })
All users denied despite correct roles claimsMapping.roles path does not match the actual JWT claim path Decode a real JWT and verify the dot-path resolves to the roles array
authorities field not recognized on decorator @frontmcp/auth not imported (metadata augmentation not active) Add import '@frontmcp/auth' (or import type ... from '@frontmcp/auth') anywhere in your project to activate the metadata augmentation. There is no @frontmcp/auth/authorities subpath.
ABAC condition always fails { fromInput: 'tenantId' } but tool input field is named tenant_id The fromInput key must exactly match the tool's input schema field name
ReBAC always denies No relationshipResolver provided Implement RelationshipResolver and pass it to plugin options
Custom evaluator not found Key in custom.* policy does not match registered evaluator name Ensure the evaluator is registered with the same key used in the policy
List endpoints show all entries No authorities config in @FrontMcp() or hook priority conflict Verify authorities: { ... } is set on @FrontMcp() decorator
AuthorityDeniedError has no detail deniedBy field shows generic message Check the evaluatedPolicies array on the error for which policy type failed
TS error: evaluators/relationshipResolver/claimsResolver not on AuthoritiesConfig The runtime Zod schema accepts these keys but the exported AuthoritiesConfig interface in authorities.profiles.ts only declares claimsMapping, profiles, scopeMapping, pipes. Pass the config inline (TS infers from the decorator's broader type) or cast the typed config as the interface catches up.

Examples

This skill currently exposes only references; see references/ for guidance.

Accessing This Skill

Skills are distributed as plain SKILL.md files plus a sibling references/ and examples/ tree, so consumers can pick whichever access mode fits:

Mode How it works
Filesystem Read libs/skills/catalog/frontmcp-authorities/ directly from a clone of the catalog repo, or from a published @frontmcp/skills install. SKILL.md is the entry point.
frontmcp CLI frontmcp skills list, frontmcp skills read frontmcp-authorities, frontmcp skills read frontmcp-authorities:references/<file>.md, frontmcp skills install frontmcp-authorities — no server required.
MCP skill:// When a developer mounts this skill into their own FrontMCP server (@FrontMcp({ skills: [...] })), the SDK exposes it via SEP-2640 resources: skill://frontmcp-authorities/SKILL.md, skill://frontmcp-authorities/references/{file}.md, etc. The server’s skill://index.json returns the SEP-2640 discovery document for everything mounted on it.

The catalog itself is not an MCP server. The skill:// URIs only resolve when a server has been configured to host this skill.

Reference

  • Auth Architecture — Full three-layer model: server auth, auth providers, authorities, vault, scope challenges
  • Authorities Documentation — RBAC/ABAC/ReBAC details, profiles, combinators, hooking
  • Source: libs/auth/src/authorities/ (engine, types, evaluators), flow stages in libs/sdk/src/tool/flows/, libs/sdk/src/resource/flows/, libs/sdk/src/prompt/flows/, libs/sdk/src/agent/flows/
  • Related skills: frontmcp-config, frontmcp-development, frontmcp-extensibility