Identity Linking Capability¶
- Capability Name:
dev.ucp.common.identity_linking - Schema:
https://ucp.dev/schemas/common/identity_linking.json
Overview¶
The Identity Linking capability enables a platform to obtain authorization to perform actions on behalf of a user on a business's (relying party) site.
This linkage is foundational for user-authenticated commerce experiences: accessing loyalty benefits, personalized offers, saved addresses, wishlists, and order history. Capabilities without identity linking still operate at public or agent-authenticated access levels — identity linking upgrades the experience, it does not gate it.
This specification uses
OAuth 2.0
as the v1 auth mechanism. The schema is designed for incremental extension:
future versions will add delegated identity provider support and non-OAuth
authentication mechanisms — both via a single config.providers extension
point — without breaking changes to this version (see
Future Extensibility).
Participants¶
| UCP Role | Identity Role | Description |
|---|---|---|
| Platform | User Agent | Trusted intermediary that initiates identity linking and presents user identity tokens to businesses on behalf of the user. |
| Business | Authorization Server / Relying Party | Hosts its own OAuth 2.0 authorization server. Authenticates users and issues access tokens scoped to UCP capabilities. |
| User | Resource Owner | The person whose identity is being linked. Grants explicit consent to the platform during the OAuth authorization flow. |
Access Levels¶
Capabilities operate at three access levels:
| Level | Authentication | Example |
|---|---|---|
| Public | None | Browse a public catalog |
| Agent-authenticated | Platform credentials (client_id / client_secret) |
Guest checkout, create a cart |
| User-authenticated | Platform credentials + user identity token | Saved addresses, full order history, personalized pricing |
Identity linking bridges agent-authenticated to user-authenticated access: the platform obtains a user identity token by completing the OAuth flow described below, and presents it on subsequent requests.
Identity linking and capability negotiation are independent layers. A
capability is advertised and negotiated based on its own profile presence —
never excluded because identity linking is absent. Identity linking, when
present, declares the scopes that gate user-authenticated operations
within negotiated capabilities (see Scopes). A merchant whose
profile lists dev.ucp.shopping.order has it in the negotiated intersection
either way. If their profile also lists identity linking with
dev.ucp.shopping.order:read in config.scopes, operations covered by
that scope require a user identity token.
UCP and OAuth¶
UCP defines commerce semantics (which scopes mean what, which gate which operations); OAuth (RFC 8414) defines identity machinery (endpoints, flows, accepted scope vocabulary); runtime messages carry per-request advisories.
- UCP
config.scopesdeclares hard gates: scopes that require user authentication for the operations they cover. - OAuth
scopes_supported(RFC 8414) declares the accepted scope vocabulary: every scope the authorization server will honor if requested. - The diff (
scopes_supported∖config.scopes) is the optional layer: scopes the merchant accepts but doesn't gate, used to advertise authentication-unlocked features without requiring auth. - UCP
messages[]carry runtime contextual hints: per-request notices likeidentity_optional(see Optional Authentication) signaling that authenticating would unlock value in the current context.
General Guidelines¶
For Platforms¶
-
MUST authenticate token endpoint requests using a method advertised in the business's
token_endpoint_auth_methods_supportedmetadata (RFC 8414):- Confidential clients (server-side platforms that can protect a
credential) SHOULD prefer asymmetric methods —
private_key_jwt(RFC 7523 §2.2) ortls_client_auth(RFC 8705) — and MAY useclient_secret_basic(RFC 6749 §2.3.1, RFC 7617) where the business supports it. - Public clients (native, desktop, browser-extension, and on-device
agent runtimes per
RFC 8252 §8.5)
MUST use
noneand rely on PKCE withS256(RFC 7636) as proof-of-possession of the authorization code. Public clients MUST NOT embed aclient_secret.
Platforms MUST select the strongest method offered by the business that is compatible with the platform's deployment model. * MUST include user identity tokens in the HTTP
Authorizationheader using the Bearer scheme:Authorization: Bearer <access_token>(RFC 6750 §2.1). * MUST processWWW-Authenticate: Bearerchallenges per RFC 6750 §3 on401and403responses to user-authenticated operations. Platforms MUST extract thescopeparameter (when present) to construct subsequent authorization requests, and SHOULD follow theresource_metadatapointer (RFC 9728) when present to discover the protecting authorization server. * MUST implement the OAuth 2.0 Authorization Code flow (RFC 6749 §4.1) as the account linking mechanism. * MUST use PKCE (RFC 7636) withcode_challenge_method=S256for all authorization code exchanges. * MUST validate theissparameter in the authorization response (RFC 9207) to prevent Mix-Up Attacks. The platform MUST verify that theissvalue matches the authorization server's issuer URI (as declared in its RFC 8414 metadata). If the values do not match, the platform MUST abort and discard the authorization response. * SHOULD include a unique, unguessablestateparameter in the authorization request to prevent CSRF (RFC 6749 §10.12). * Revocation and security events: * MUST call the business's token revocation endpoint (RFC 7009) when a user initiates an unlink action on the platform side. * SHOULD support OpenID RISC Profile 1.0 to handle asynchronous account updates and cross-account protection events initiated by the business. - Confidential clients (server-side platforms that can protect a
credential) SHOULD prefer asymmetric methods —
For Businesses¶
- MUST implement OAuth 2.0 (RFC 6749).
- MUST publish authorization server metadata via
RFC 8414
at
/.well-known/oauth-authorization-server. - MUST populate
scopes_supportedin RFC 8414 metadata to allow platforms to detect scope mismatches before initiating an authorization flow. - MUST return the
issparameter in the authorization response (RFC 9207). - MUST enforce PKCE
(RFC 7636)
validation at the token endpoint for all authorization code exchanges.
Requests without a valid
code_verifierMUST be rejected. - MUST enforce exact string matching for the
redirect_uriparameter during authorization requests to prevent open redirects and token theft. Theredirect_uriin the token request MUST be identical to the one in the authorization request. Exception — loopback redirects: For redirect URIs targeting127.0.0.1or[::1], businesses MUST ignore the port component and match on scheme, host, and path only, to accommodate native and desktop clients that obtain an ephemeral port from the OS at runtime (RFC 8252 §7.3). - MUST declare supported client authentication methods in
token_endpoint_auth_methods_supported(RFC 8414) and enforce one of the declared methods at the token endpoint. Businesses SHOULD support at least one asymmetric confidential-client method (private_key_jwtortls_client_auth) and MAY supportnonefor public clients per RFC 8252. Whennoneis advertised, businesses MUST require PKCE withS256and MUST reject any authorization code redemption that lacks a validcode_verifier. Requests that fail the negotiated authentication method MUST be rejected withinvalid_client; requests that fail PKCE MUST be rejected withinvalid_grant. - MUST validate user identity tokens on every user-authenticated request:
verify
iss,aud(the business's resource server identifier),exp, scopes, andclient_id/azp(or equivalent) to confirm the token was issued to the authenticated platform client (RFC 9068 §4). - MUST emit a
WWW-Authenticate: Bearerchallenge per RFC 6750 §3 on401 Unauthorized(identity_required) and403 Forbidden(insufficient_scope) responses to user-authenticated operations. See Error Handling for the full normative requirements. - MUST implement token revocation
(RFC 7009).
Revoking a
refresh_tokenMUST also immediately invalidate allaccess_tokens issued from it. - MUST support revocation requests authenticated with the same client credentials used at the token endpoint.
- SHOULD provide an account creation flow if the user does not already have
an account, or return a
continue_urlin anidentity_requirederror response (see Error Handling) pointing to an onboarding flow. - MUST support standard UCP scopes as defined in the Scopes section.
- SHOULD publish protected resource metadata at
/.well-known/oauth-protected-resource(RFC 9728) and reference it via theresource_metadataparameter inWWW-Authenticatechallenges. This lets platforms discover the authorization server protecting the resource without relying on domain conventions and prepares the deployment for future delegated IdP scenarios where the AS may not live on the business domain. - SHOULD support OpenID RISC Profile 1.0 to signal revocation and account state changes to platforms.
Discovery¶
Platforms resolve business authorization server metadata using the following strict two-tier hierarchy. The issuer URI used as the discovery base is the business's domain as declared in its UCP profile.
-
RFC 8414 (Primary): Fetch
{business-domain}/.well-known/oauth-authorization-server.2xxresponse: use this metadata. Discovery complete.404 Not Found: proceed to step 2.- Any other non-2xx response, network error, or timeout: MUST abort. MUST NOT proceed to step 2.
-
OIDC Discovery (Fallback): Fetch
{business-domain}/.well-known/openid-configuration.2xxresponse: use this metadata. Discovery complete.- Any non-2xx response, network error, or timeout: MUST abort the identity linking process.
Platforms MUST NOT silently fall through on any error other than 404 in
step 1. This prevents partial or undefined behavior when a server is
misconfigured or temporarily unavailable.
The issuer value in the discovered metadata MUST match the discovery
base URI exactly (per
RFC 8414 §3.3).
Platforms MUST NOT normalize (e.g., strip trailing slashes) before
comparison — the value must be a byte-for-byte match.
Account Linking Flow¶
Identity linking uses the OAuth 2.0 Authorization Code flow with PKCE.
Platform Business AS
| |
|-- (1) Discover metadata via RFC 8414 -->|
|<-- authorization_endpoint, token_endpoint, scopes_supported --|
| |
|-- (2) Authorization Request --------->|
| response_type=code |
| client_id, redirect_uri |
| scope=<derived scope set> |
| code_challenge (S256) |
| state |
| |
| [user authenticates and |
| grants consent at business] |
| |
|<-- (3) Authorization Response --------|
| code, state, iss |
| |
| Validate: state matches, iss matches |
| discovered issuer URI |
| |
|-- (4) Token Request ----------------->|
| grant_type=authorization_code |
| code, redirect_uri |
| code_verifier |
| client auth (per advertised |
| token_endpoint_auth_method) |
| |
|<-- (5) Token Response ----------------|
| access_token, refresh_token |
| token_type=Bearer, scope |
Step 2 — Scope set: Platforms derive the authorization scope set from the
business's config.scopes map (see Scope Derivation).
Platforms MUST request only the derived scope set — not a superset.
Step 3 — Validation: The platform MUST verify that the state
parameter matches the value sent in step 2, and that the iss parameter
matches the authorization server's issuer URI from discovered metadata.
If either check fails, the platform MUST discard the authorization
response.
Step 4 — PKCE: The code_verifier MUST correspond to the
code_challenge sent in step 2. Businesses MUST reject token requests
where code_verifier is absent or does not verify against the stored
code_challenge.
Scopes¶
Scopes define the user-authenticated permissions a business grants to a
platform. Businesses declare the scopes they offer in config.scopes of their
dev.ucp.common.identity_linking entry. Each key is the OAuth scope string as
it appears on the wire ({capability}:{scope}, e.g.
dev.ucp.shopping.order:read); each value is a per-scope policy object.
Listing a scope in config.scopes declares that the corresponding
operations require a user identity token. Operations not gated by any
listed scope operate at whatever access level the business permits — public,
agent-authenticated, or otherwise. The business defines the access policy
for non-scoped operations; UCP does not prescribe a default.
Scope Token Format¶
Scope tokens follow the convention {capability-name}:{scope-name}:
dev.ucp.shopping.order:readdev.ucp.shopping.order:managedev.ucp.shopping.checkout:manage
The capability name uses UCP's reverse-DNS naming. The scope name denotes
the permission being granted — typically an operation group on a
resource (read, manage, write) or an entry-point operation (create)
defined by each capability's specification.
Scope names MUST match the pattern ^[a-z][a-z0-9_]*$. Third-party
capabilities follow the same convention using their own reverse-DNS name:
com.example.loyalty:points.
Each capability's specification defines its well-known scopes — the
standard set platforms expect. Businesses MAY declare additional
custom scopes following the same convention to gate operations at
finer granularity (for example, to gate complete independently from the
well-known dev.ucp.shopping.checkout:manage, or to require elevated
authentication for high-value purchases). Platforms MUST treat any
scope listed in config.scopes — well-known or custom — as gating its
operations behind user authentication.
Per-Scope Policy and Metadata¶
Each scope's value is an open object that carries per-scope policy and
metadata. Empty {} means "user auth required, nothing else." Possible
fields include authentication constraints (min_acr, max_token_age,
require_mfa), declarative metadata (claims produced when granted),
or other scope-specific configuration. Platforms MUST ignore
unrecognized fields.
description¶
Optional human-readable description of the scope that platforms can use to present and explain context (requirement and value) to the user.
"dev.ucp.shopping.order:manage": {
"description": {
"plain": "Manage your orders: cancel, return, or modify post-purchase.",
"markdown": "**Manage your orders**: cancel, return, or modify post-purchase."
}
}
Businesses SHOULD provide a description for each scope they declare. Platforms MAY display the text to inform their own consent UX.
Scope Derivation¶
Platforms derive the authorization scope set from the business's
config.scopes before initiating the account linking flow:
- Read
config.scopesfrom the business'sdev.ucp.common.identity_linkingcapability entry. - Filter to scopes whose capability prefix is in the negotiated capability set — ignore scopes for capabilities the platform does not support.
- From the remaining set, select the scopes the platform intends to use (informed by which operations it plans to call; see each capability's spec for operation-to-scope mappings).
- Apply the per-scope policy on each selected scope when constructing the authorization request.
If no scope is required for the operations the platform intends to call, the platform can skip the identity linking flow — operations work with public or agent-authenticated access. However, linking may still be beneficial to unlock session-native personalization (saved addresses, member pricing, order history visibility, etc.); the merchant resolves these from the authenticated user context regardless of which scopes were granted.
Consent Presentation¶
Consent screens are rendered by the business's authorization server. This
specification does not define scope description strings — the authorization
server is responsible for presenting human-readable consent text for the
scopes it supports. Consent screens SHOULD group related scopes
intelligibly rather than listing individual operations: for example, "Allow
[platform] to view your order history" rather than "grant
dev.ucp.shopping.order:read."
Error Handling¶
identity_required¶
When an operation is gated by a scope listed in config.scopes and the
request arrives with an absent, expired, invalid, or unverifiable user
identity token, the business MUST return:
- HTTP
401 Unauthorized - A
WWW-Authenticate: Bearerchallenge header per RFC 6750 §3 - A UCP error response body containing a message with
code: "identity_required"
The WWW-Authenticate header MUST include a realm parameter set to
the business's issuer URI as declared in
RFC 8414
metadata. When the request included a token but it was invalid, expired, or
unverifiable, the header MUST also include error="invalid_token" and
MAY include error_description per RFC 6750 §3. When no token was
presented, the error parameter SHOULD be omitted (RFC 6750 §3.1).
The header SHOULD include a resource_metadata parameter
(RFC 9728 §5.1)
pointing to the protected resource metadata document
(/.well-known/oauth-protected-resource).
The business MAY include a continue_url in the response body for
non-OAuth onboarding flows (e.g., account creation, terms acceptance)
where the user must complete a hosted step before re-authenticating.
continue_url MUST NOT be used to convey a pre-baked OAuth
authorization request; the platform constructs its own authorization
request from the WWW-Authenticate challenge and discovered metadata,
including PKCE values, state, and redirect_uri it owns.
No token presented (first request to a gated operation — error omitted per RFC 6750 §3.1):
HTTP/1.1 401 Unauthorized
WWW-Authenticate: Bearer realm="https://merchant.example.com",
resource_metadata="https://merchant.example.com/.well-known/oauth-protected-resource"
Content-Type: application/json
{
"messages": [
{
"type": "error",
"code": "identity_required",
"content": "User identity is required to access order history.",
"severity": "requires_buyer_review"
}
]
}
Token present but invalid or expired (error="invalid_token" included per RFC 6750 §3):
HTTP/1.1 401 Unauthorized
WWW-Authenticate: Bearer realm="https://merchant.example.com",
error="invalid_token",
error_description="The access token expired",
resource_metadata="https://merchant.example.com/.well-known/oauth-protected-resource"
Content-Type: application/json
{
"messages": [
{
"type": "error",
"code": "identity_required",
"content": "User identity is required to access order history.",
"severity": "requires_buyer_review"
}
]
}
insufficient_scope¶
When a request arrives with a valid user identity token but the token
lacks a scope required by the operation (or fails a scope-level policy
such as min_acr or max_token_age), the business MUST return:
- HTTP
403 Forbidden - A
WWW-Authenticate: Bearerchallenge header per RFC 6750 §3 - A UCP error response body containing a message with
code: "insufficient_scope"
The WWW-Authenticate header MUST include:
realm="<business issuer URI>"error="insufficient_scope"scope="<space-separated full required scope set for the operation>"
and SHOULD include a resource_metadata parameter pointing to the
protected resource metadata document (RFC 9728).
The scope parameter MUST list the full set of scopes required
for the operation, not just the missing ones. The platform compares the
full set against scopes already granted on its current token and uses
incremental authorization to request only the scopes it does not yet
have, avoiding redundant consent prompts for scopes the user has already
approved.
HTTP/1.1 403 Forbidden
WWW-Authenticate: Bearer realm="https://merchant.example.com",
error="insufficient_scope",
scope="dev.ucp.shopping.order:read dev.ucp.shopping.order:manage",
resource_metadata="https://merchant.example.com/.well-known/oauth-protected-resource"
Content-Type: application/json
{
"messages": [
{
"type": "error",
"code": "insufficient_scope",
"content": "This operation requires scopes: dev.ucp.shopping.order:read, dev.ucp.shopping.order:manage",
"severity": "requires_buyer_review"
}
]
}
Note:
identity_requiredandinsufficient_scopeare intentionally distinct. Platforms MUST NOT retry aninsufficient_scoperesponse by re-initiating a fresh account linking flow — they MUST instead request only the missing scope(s) via incremental authorization to preserve previously granted scopes.
Optional Authentication¶
A mechanism for the business to signal to the platform that authentication is available and would provide value in the current context, even though the operation succeeded without it. The platform may then present this signal to the user.
identity_optional¶
Businesses SHOULD include this info-severity code in successful
responses when authentication is available and would meaningfully
unlock additional capabilities in the current context. The content
field conveys the business's value prompt to the platform (e.g.,
"Sign in for member pricing and personalized results.").
{
...,
"messages": [
{
"type": "info",
"code": "identity_optional",
"content": "Sign in for member pricing and personalized results."
}
]
}
Security Considerations¶
- PKCE. PKCE (
S256) is REQUIRED for all authorization code flows. Plain PKCE (plain) MUST NOT be used. Businesses MUST reject authorization code exchanges without a validcode_verifier. - Client authentication. Businesses negotiate client authentication via
token_endpoint_auth_methods_supported(RFC 8414). Confidential clients SHOULD prefer asymmetric methods (private_key_jwt,tls_client_auth) overclient_secret_basicto eliminate shared-secret leak risk. Public clients (native, desktop, and on-device agents per RFC 8252 §8.5) cannot keep aclient_secretconfidential and MUST usenone; for these clients PKCE withS256is the proof-of-possession that authenticates the authorization grant. Businesses MUST NOT requireclient_secret_basicas the only method when serving native or agent platforms. - Mix-Up Attack prevention. Platforms MUST validate the
issparameter in the authorization response (RFC 9207). Businesses MUST returnissin every authorization response. Withoutissvalidation, an attacker that controls one authorization server can redirect a victim's authorization code to a different server. - Authentication challenges. Businesses MUST emit
WWW-Authenticate: Bearerchallenges per RFC 6750 §3 on401and403responses to user-authenticated operations. Platforms MUST process the structuredscopeanderrorparameters to drive authorization flow decisions;error_descriptionis a human-readable hint only and MUST NOT be used for control-flow decisions. Therealmparameter MUST match the business's issuer URI so platforms can correlate the challenge with the correct authorization server. redirect_uriexactness. Businesses MUST enforce exact string matching forredirect_uri. Partial-match or prefix-match implementations are a common source of open redirect and token theft vulnerabilities.issuerexactness. Theissuervalue in RFC 8414 metadata and theissparameter in authorization responses MUST be identical (byte-for-byte). Platforms MUST NOT normalize before comparison. Normalization (e.g., stripping trailing slashes) is a known source ofissvalidation bypass.- Transport security. All communication between platform and business MUST use HTTPS with a minimum of TLS 1.2 (RFC 6749 §1.6).
scopes_supported. Businesses MUST populatescopes_supportedin RFC 8414 metadata. Platforms SHOULD verify that the derived scope set is a subset ofscopes_supportedbefore initiating the authorization flow, to fail fast on scope mismatches rather than at the consent screen.- Token revocation. Platforms MUST revoke user identity tokens at the business's revocation endpoint (RFC 7009) when a user unlinks their account. Businesses MUST reject subsequent requests that present revoked tokens.
Future Extensibility¶
This specification intentionally scopes v1 to business-hosted OAuth 2.0. The schema and protocol are designed to accommodate additional auth patterns as non-breaking extensions in future versions:
Delegated Identity Providers and Mechanism Extensibility (config.providers)¶
A future version will allow businesses to declare trusted identity providers in
a config.providers map, keyed by reverse-domain identifier. Each entry carries
a type discriminator that defaults to oauth2, making this a single extension
point for both delegated OAuth IdPs and future non-OAuth mechanisms such as
wallet attestation or verifiable credentials.
For delegated OAuth IdPs (e.g., com.google, com.shopify): platforms that
have already established an OAuth session with a trusted provider can present a
JWT-based authorization grant to the business's token endpoint instead of
initiating a new browser-based OAuth flow — useful for multi-merchant agentic
commerce where N merchants should not require N separate consent screens.
For non-OAuth mechanisms: entries with a non-oauth2 type value enable
wallet attestation and similar schemes. Platforms select the first entry whose
type they support, using business-preference ordering — analogous to TLS
cipher suite negotiation.
When config.providers is present, the platform uses the provider selection
and identity chaining flows defined in that version. When config.providers is
absent (as in v1), platforms MUST fall back to direct OAuth 2.0 against the
business domain via RFC 8414 discovery.
Forward-compatibility rule for platforms: When config contains fields
not defined in this version of the spec (providers or any other future field),
platforms MUST ignore those fields and proceed using OAuth 2.0 with RFC 8414
discovery on the business domain, as defined here. This ensures v1 platform
implementations remain valid as the spec evolves.
Examples¶
Authorization Server Metadata¶
Example metadata hosted at /.well-known/oauth-authorization-server per
RFC 8414:
{
"issuer": "https://merchant.example.com",
"authorization_endpoint": "https://merchant.example.com/oauth2/authorize",
"token_endpoint": "https://merchant.example.com/oauth2/token",
"revocation_endpoint": "https://merchant.example.com/oauth2/revoke",
"jwks_uri": "https://merchant.example.com/oauth2/jwks",
"scopes_supported": [
"dev.ucp.shopping.order:read",
"dev.ucp.shopping.order:manage"
],
"response_types_supported": ["code"],
"grant_types_supported": ["authorization_code", "refresh_token"],
"code_challenge_methods_supported": ["S256"],
"token_endpoint_auth_methods_supported": [
"private_key_jwt",
"tls_client_auth",
"client_secret_basic",
"none"
],
"authorization_response_iss_parameter_supported": true,
"service_documentation": "https://merchant.example.com/docs/oauth2"
}
Note: authorization_response_iss_parameter_supported: true advertises
RFC 9207 support. code_challenge_methods_supported: ["S256"] signals PKCE.
Both MUST be present in UCP-compliant metadata.
token_endpoint_auth_methods_supported lists every method the business
accepts. The example advertises asymmetric methods (private_key_jwt,
tls_client_auth) for confidential clients, client_secret_basic for
legacy compatibility, and none for public clients (native, desktop, and
on-device agents per
RFC 8252);
when none is advertised, PKCE with S256 is required. Businesses that
do not serve public clients MAY omit none.
Business Profile (/.well-known/ucp)¶
The shape of config.scopes reflects the business's policy.
B2C retailer¶
Public catalog, guest checkout (no scope required), user-bound order operations gated:
{
"ucp": {
"capabilities": {
"dev.ucp.common.identity_linking": [{
"version": "Working Draft",
"spec": "https://ucp.dev/specification/identity-linking",
"schema": "https://ucp.dev/schemas/common/identity_linking.json",
"config": {
"scopes": {
"dev.ucp.shopping.order:read": {},
"dev.ucp.shopping.order:manage": {}
}
}
}]
}
}
}
Reading this config:
dev.ucp.shopping.order:readand:manage— listed → user auth required to obtain.- Catalog, checkout, cart, and everything else — not listed → no user-auth scope required. Public/agent-authenticated access. The user may still request, or the agent may still offer, linking to access additional capabilities such as personalization, saved addresses and credentials, loyalty pricing, etc.
B2B wholesaler¶
No guest checkout — every transaction requires an authenticated user:
{
"ucp": {
"capabilities": {
"dev.ucp.common.identity_linking": [{
"version": "Working Draft",
"spec": "https://ucp.dev/specification/identity-linking",
"schema": "https://ucp.dev/schemas/common/identity_linking.json",
"config": {
"scopes": {
"dev.ucp.shopping.checkout:manage": {},
"dev.ucp.shopping.order:read": {},
"dev.ucp.shopping.order:manage": {}
}
}
}]
}
}
}
The difference: dev.ucp.shopping.checkout:manage is now listed. The
B2C example doesn't gate checkout; anyone can start a guest session. This
merchant requires the user to be authenticated for all checkout
operations — create, update, complete, and cancel.
Whether the user is B2B-eligible, what pricing they see, what payment terms apply — those are user attributes the merchant resolves at runtime, not additional scopes.
End-to-End Walkthrough¶
Setup: Platform (AI shopping agent) + Business (B2C retailer from the example above).
Negotiated capabilities: dev.ucp.shopping.checkout,
dev.ucp.shopping.order, dev.ucp.common.identity_linking.
Step 1 — Scope derivation. Platform reads business's config.scopes:
dev.ucp.shopping.order:read(read order history)dev.ucp.shopping.order:manage(cancel/return)
The user wants the agent to be able to read order history and cancel orders on their behalf, so the platform requests both scopes.
Derived scope set: dev.ucp.shopping.order:read dev.ucp.shopping.order:manage
Step 2 — Discovery. Platform fetches
https://merchant.example.com/.well-known/oauth-authorization-server,
receives 2xx, extracts authorization_endpoint and token_endpoint.
Verifies both scopes are in scopes_supported.
Step 3 — Authorization request. Platform generates PKCE pair
(code_verifier, code_challenge), sends the user to:
GET https://merchant.example.com/oauth2/authorize
?response_type=code
&client_id=platform-client-id
&redirect_uri=https://agent.example.com/callback
&scope=dev.ucp.shopping.order:read dev.ucp.shopping.order:manage
&code_challenge=<S256-hash>
&code_challenge_method=S256
&state=<random>
Reserved characters in redirect_uri and : in scope tokens must be
percent-encoded in the actual request; they are shown decoded here for readability.
Step 4 — Authorization response. User authenticates and consents. Business redirects to:
https://agent.example.com/callback
?code=<auth-code>
&state=<random>
&iss=https://merchant.example.com
Platform validates state matches and iss equals the discovered issuer.
Step 5 — Token exchange. Platform calls token endpoint:
POST https://merchant.example.com/oauth2/token
Authorization: Basic <base64(client_id:client_secret)>
Content-Type: application/x-www-form-urlencoded
grant_type=authorization_code
&code=<auth-code>
&redirect_uri=https://agent.example.com/callback
&code_verifier=<verifier>
Business validates code_verifier against stored code_challenge, returns:
{
"access_token": "<token>",
"token_type": "Bearer",
"expires_in": 3600,
"refresh_token": "<refresh>",
"scope": "dev.ucp.shopping.order:read dev.ucp.shopping.order:manage"
}
Platform now includes Authorization: Bearer <token> on subsequent requests
to user-authenticated capability endpoints.