docs:add token introspection documentation based on blog post (#6118)
* Modifying docs to comply with Google style guide for docs. * Removed typo. * Added FE and BE integration with ZITADEL * Added FE and BE integration with ZITADEL * Added FE and BE integration with ZITADEL * Update docs/docs/guides/solution-scenarios/frontend-calling-backend-API.mdx Co-authored-by: mffap <mpa@zitadel.com> * Update docs/docs/guides/solution-scenarios/frontend-calling-backend-API.mdx Co-authored-by: mffap <mpa@zitadel.com> * Addressed @mffap's question on the PR. * Added docs for detailed explanations for token introspection. * Update docs/docs/guides/integrate/token-introspection/basic-auth.mdx Co-authored-by: Fabi <fabienne@zitadel.com> * Update docs/sidebars.js Co-authored-by: Fabi <fabienne@zitadel.com> * Update docs/docs/guides/integrate/token-introspection/private-key-jwt.mdx * Addressed @hifabienne's review comments. * Addressed @hifabienne's review comments. --------- Co-authored-by: Dakshitha Ratnayake <dakshitharatnayake@Dakshithas-MacBook-Pro-2.local> Co-authored-by: mffap <mpa@zitadel.com> Co-authored-by: Fabi <fabienne@zitadel.com>
@ -0,0 +1,21 @@
|
|||||||
|
Upon successful introspection, regardless of the token type or introspection method, a response with the boolean `active` is returned, indicating if the provided token is active and if the requesting client is part of the token audience. If `active` is true, further information will be provided:
|
||||||
|
|
||||||
|
| **Property** | **Description** |
|
||||||
|
| --- | --- |
|
||||||
|
| `aud` | The audience of the token |
|
||||||
|
| `client_id` | The client_id of the application the token was issued to |
|
||||||
|
| `exp` | Time the token expires (as unix time) |
|
||||||
|
| `iat` | Time the token was issued at (as unix time) |
|
||||||
|
| `iss` | Issuer of the token |
|
||||||
|
| `jti` | Unique id of the token |
|
||||||
|
| `nbf` | Time the token must not be used before (as unix time) |
|
||||||
|
| `scope` | Space delimited list of scopes granted to the token |
|
||||||
|
| `token_type` | Type of the inspected token. Value is always Bearer |
|
||||||
|
| `username` | ZITADEL's login name of the user. Consists of username@primarydomain |
|
||||||
|
|
||||||
|
|
||||||
|
Depending on the granted scopes, additional information about the authorized user is provided.
|
||||||
|
|
||||||
|
If the authorization fails, an HTTP 401 with invalid_client will be returned.
|
||||||
|
|
||||||
|
In summary, the introspection endpoint plays a crucial role in validating access tokens, either opaque or JWT, ensuring that they are not revoked.
|
103
docs/docs/guides/integrate/token-introspection/basic-auth.mdx
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
---
|
||||||
|
title: Basic authentication
|
||||||
|
---
|
||||||
|
|
||||||
|
import IntrospectionResponse from './_introspection-response.mdx';
|
||||||
|
|
||||||
|
This is a guide on how to secure your API using [Basic Authentication](https://zitadel.com/docs/apis/openidoauth/authn-methods#client-secret-basic).
|
||||||
|
|
||||||
|
## Register the API in ZITADEL
|
||||||
|
|
||||||
|
1. Go to your project and click on the **New** button as shown below.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-1.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
2. Give a name to your application (Test API 2 is the name given below) and select type **API**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-2.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
3. Select **Basic** as the authentication method and click **Continue**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-3.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
4. Now review your configuration and click **Create**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-4.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
5. You will now see the API’s **Client ID** and the **Client Secret**. Copy them and click **Close**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-5.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
6. When you click **URLs** on the left, you will see the relevant OIDC URLs. Note down the **issuer** URL, **token_endpoint** and **introspection_endpoint**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-6.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
7. Also note down the **Resource ID** of your project.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-basic-auth-7.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
## Token introspection
|
||||||
|
|
||||||
|
With Basic Authentication, you will receive a Client ID and Client Secret for your API. Send your client_id and client_secret as a Basic Auth Header in the following format:
|
||||||
|
|
||||||
|
```
|
||||||
|
Authorization: "Basic " + base64( formUrlEncode(client_id) + ":" + formUrlEncode(client_secret) )
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
The request from the API to the introspection endpoint should be in the following format:
|
||||||
|
|
||||||
|
```
|
||||||
|
curl --request POST \
|
||||||
|
--url {your_domain}/oauth/v2/introspect \
|
||||||
|
--header 'Content-Type: application/x-www-form-urlencoded' \
|
||||||
|
--header 'Authorization: Basic {your_basic_auth_header}' \
|
||||||
|
--data token=VjVxyCZmRmWYqd3_F5db9Pb9mHR5fqzhn...
|
||||||
|
```
|
||||||
|
|
||||||
|
Here's an example of how this is done in Python code:
|
||||||
|
|
||||||
|
```
|
||||||
|
def introspect_token(self, token_string):
|
||||||
|
url = ZITADEL_INTROSPECTION_URL
|
||||||
|
data = {'token': token_string, 'token_type_hint': 'access_token', 'scope': 'openid'}
|
||||||
|
auth = HTTPBasicAuth(API_CLIENT_ID, API_CLIENT_SECRET)
|
||||||
|
resp = requests.post(url, data=data, auth=auth)
|
||||||
|
resp.raise_for_status()
|
||||||
|
return resp.json()
|
||||||
|
```
|
||||||
|
|
||||||
|
## Introspection response
|
||||||
|
|
||||||
|
<IntrospectionResponse/>
|
||||||
|
|
||||||
|
Follow this [tutorial](https://github.com/zitadel/examples-api-access-and-token-introspection/tree/main/api-basic-authentication) to learn how to register an API application using Basic Auth with ZITADEL and test it.
|
||||||
|
|
@ -0,0 +1,192 @@
|
|||||||
|
---
|
||||||
|
title: JSON Web Token profile
|
||||||
|
---
|
||||||
|
|
||||||
|
import IntrospectionResponse from './_introspection-response.mdx';
|
||||||
|
|
||||||
|
This is a guide on how to secure your API using [JSON Web Token (JWT) profile (recommended)](https://zitadel.com/docs/apis/openidoauth/authn-methods#client-secret-basic).
|
||||||
|
|
||||||
|
## Register the API in ZITADEL and generate private and public keys
|
||||||
|
|
||||||
|
1. Go to your project and click on the **New** button as shown below.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-1.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
2. Give a name to your application (Test API is the name given below) and select type **API**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-2.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
3. Select **JWT** as the authentication method and click **Continue**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-3.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
4. Now review your configuration and click **Create**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-4.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
5. You will now see the API’s **Client ID**. You will not see a client secret because we are using a private JWT key.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-5.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
6. Next, we must create the key pairs. Click on **New**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-6.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
7. Select **JSON** as the type of key. You can also set an expiration time for the key or leave it empty. Click on **Add**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-7.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
8. Download the created key by clicking the **Download** button and then click **Close**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-8.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
9. The key will be downloaded.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-9.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
10. When you click on URLs on the left, you will see the relevant OIDC URLs. Note down the **issuer** URL, **token_endpoint** and **introspection_endpoint**.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-10.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
11. The key that you downloaded will be of the following format.
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"type":"application",
|
||||||
|
"keyId":"<YOUR_KEY_ID>",
|
||||||
|
"key":"-----BEGIN RSA PRIVATE KEY-----\<YOUR_PRIVATE_KEY>\n-----END RSA PRIVATE KEY-----\n",
|
||||||
|
"appId":"<YOUR_APP_ID>",
|
||||||
|
"clientId":"<YOUR_CLIENT_ID>"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
12. Also note down the **Resource ID** of your project.
|
||||||
|
|
||||||
|
<img
|
||||||
|
src="/docs/img/guides/integrate/token-introspection-jwt-profile-11.png"
|
||||||
|
width="75%"
|
||||||
|
alt="Register the API"
|
||||||
|
/>
|
||||||
|
|
||||||
|
## Token introspection
|
||||||
|
|
||||||
|
You must send a client_assertion as a JWT signed with the API’s private key for ZITADEL to validate the signature against the registered public key.
|
||||||
|
|
||||||
|
Request parameters:
|
||||||
|
|
||||||
|
| **Parameter** | **Description** |
|
||||||
|
---|---
|
||||||
|
| `client_assertion` | When using JWT profile for token or introspection endpoints, you must provide a JWT as an assertion generated with the structure shown below and signed with the downloaded key. |
|
||||||
|
| `client_assertion_type` | `urn:ietf:params:oauth:client-assertion-type:jwt-bearer` |
|
||||||
|
|
||||||
|
|
||||||
|
You must create your client assertion JWT with the following format:
|
||||||
|
|
||||||
|
Header:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"alg": "RS256",
|
||||||
|
"kid": "81693565968962154" (keyId from your key file)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
Payload:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"iss": "78366401571920522@acme", (clientId from your key file)
|
||||||
|
"sub": "78366401571920522@acme", (clientId from your key file)
|
||||||
|
"aud": "https://{your_domain}", (your ZITADEL domain/issuer URL)
|
||||||
|
"exp": 1605183582, (Unix timestamp of the expiry)
|
||||||
|
"iat": 1605179982 (Unix timestamp of the creation signing time of the JWT, MUST NOT be older than 1h)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Create the JSON Web Token with the above header and payload, and sign it with the private key in your key file. You can do this programmatically or use tools like [https://github.com/zitadel/zitadel-tools](https://github.com/zitadel/zitadel-tools) and [https://dinochiesa.github.io/jwt/](https://dinochiesa.github.io/jwt/).
|
||||||
|
|
||||||
|
The request from the API to the introspection endpoint should be in the following format:
|
||||||
|
```bash
|
||||||
|
curl --request POST \
|
||||||
|
--url {your_domain}/oauth/v2/introspect \
|
||||||
|
--header 'Content-Type: application/x-www-form-urlencoded' \
|
||||||
|
--data client_assertion_type=urn:ietf:params:oauth:client-assertion-type:jwt-bearer \
|
||||||
|
--data client_assertion=eyJhbGciOiJSUzI1Ni... \
|
||||||
|
--data token=VjVxyCZmRmWYqd3_F5db9Pb9mHR
|
||||||
|
```
|
||||||
|
|
||||||
|
Here's an example of how this is done in Python code:
|
||||||
|
|
||||||
|
```python
|
||||||
|
def introspect_token(self, token_string):
|
||||||
|
#Create JWT for client assertion
|
||||||
|
payload = {
|
||||||
|
"iss": API_PRIVATE_KEY_FILE["client_id"],
|
||||||
|
"sub": API_PRIVATE_KEY_FILE["client_id"],
|
||||||
|
"aud": ZITADEL_DOMAIN,
|
||||||
|
"exp": int(time.time()) + 60 * 60, # Expires in 1 hour
|
||||||
|
"iat": int(time.time())
|
||||||
|
}
|
||||||
|
headers = {
|
||||||
|
"alg": "RS256",
|
||||||
|
"kid": API_PRIVATE_KEY_FILE["key_id"]
|
||||||
|
}
|
||||||
|
jwt_token = jwt.encode(payload, API_PRIVATE_KEY_FILE["private_key"], algorithm="RS256", headers=headers)
|
||||||
|
|
||||||
|
#Send introspection request
|
||||||
|
headers = {"Content-Type": "application/x-www-form-urlencoded"}
|
||||||
|
data = {
|
||||||
|
"client_assertion_type": "urn:ietf:params:oauth:client-assertion-type:jwt-bearer",
|
||||||
|
"client_assertion": jwt_token,
|
||||||
|
"token": token_string
|
||||||
|
}
|
||||||
|
response = requests.post(ZITADEL_INTROSPECTION_URL, headers=headers, data=data)
|
||||||
|
response.raise_for_status()
|
||||||
|
token_data = response.json()
|
||||||
|
print(f"Token data from introspection: {token_data}")
|
||||||
|
return token_data
|
||||||
|
|
||||||
|
```
|
||||||
|
## Introspection response
|
||||||
|
|
||||||
|
<IntrospectionResponse/>
|
||||||
|
|
||||||
|
Follow this [tutorial](https://github.com/zitadel/examples-api-access-and-token-introspection/tree/main/api-jwt) to learn how to register an API application using JWT Profile with ZITADEL and test it.
|
||||||
|
|
@ -150,6 +150,22 @@ module.exports = {
|
|||||||
"guides/integrate/logout",
|
"guides/integrate/logout",
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
type: "category",
|
||||||
|
label: "Token introspection",
|
||||||
|
link: {
|
||||||
|
type: "generated-index",
|
||||||
|
title: "Token introspection",
|
||||||
|
slug: "/guides/integrate/token-introspection",
|
||||||
|
description:
|
||||||
|
"Token introspection is the process of checking whether an access token is valid and can be used to access protected resources. You have an API that acts as an OAuth resource server and can be accessed by user-facing applications. To validate an access token by calling the ZITADEL introspection API, you can use the JSON Web Token (JWT) Profile (recommended) or Basic Authentication for token introspection. It's crucial to understand that the API is entirely separate from the front end. The API shouldn’t concern itself with the token type received. Instead, it's about how the API chooses to call the introspection endpoint, either through JWT Profile or Basic Authentication. Many APIs assume they might receive a JWT and attempt to verify it based on signature or expiration. However, with ZITADEL, you can send either a JWT or an opaque Bearer token from the client end to the API. This flexibility is one of ZITADEL's standout features.",
|
||||||
|
},
|
||||||
|
collapsed: true,
|
||||||
|
items: [
|
||||||
|
"guides/integrate/token-introspection/private-key-jwt",
|
||||||
|
"guides/integrate/token-introspection/basic-auth",
|
||||||
|
],
|
||||||
|
},
|
||||||
{
|
{
|
||||||
type: "category",
|
type: "category",
|
||||||
label: "Authenticate service users",
|
label: "Authenticate service users",
|
||||||
@ -169,13 +185,13 @@ module.exports = {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "category",
|
type: "category",
|
||||||
label: "Build your own Login-UI",
|
label: "Build your own login UI",
|
||||||
link: {
|
link: {
|
||||||
type: "generated-index",
|
type: "generated-index",
|
||||||
title: "Build your own Login-UI",
|
title: "Build your own login UI",
|
||||||
slug: "/guides/integrate/login-ui",
|
slug: "/guides/integrate/login-ui",
|
||||||
description:
|
description:
|
||||||
"In the following guides you will learn how to create your own login ui with our APIs. The different scenarios like username/password, external identity provider, etc will be shown.",
|
"In the following guides you will learn how to create your own login UI with our APIs. The different scenarios like username/password, external identity provider, etc. will be shown."
|
||||||
|
|
||||||
},
|
},
|
||||||
collapsed: true,
|
collapsed: true,
|
||||||
|
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-1.png
vendored
Normal file
After Width: | Height: | Size: 356 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-2.png
vendored
Normal file
After Width: | Height: | Size: 561 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-3.png
vendored
Normal file
After Width: | Height: | Size: 361 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-4.png
vendored
Normal file
After Width: | Height: | Size: 260 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-5.png
vendored
Normal file
After Width: | Height: | Size: 350 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-6.png
vendored
Normal file
After Width: | Height: | Size: 345 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-basic-auth-7.png
vendored
Normal file
After Width: | Height: | Size: 347 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-1.png
vendored
Normal file
After Width: | Height: | Size: 310 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-10.png
vendored
Normal file
After Width: | Height: | Size: 345 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-11.png
vendored
Normal file
After Width: | Height: | Size: 347 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-2.png
vendored
Normal file
After Width: | Height: | Size: 627 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-3.png
vendored
Normal file
After Width: | Height: | Size: 382 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-4.png
vendored
Normal file
After Width: | Height: | Size: 270 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-5.png
vendored
Normal file
After Width: | Height: | Size: 353 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-6.png
vendored
Normal file
After Width: | Height: | Size: 295 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-7.png
vendored
Normal file
After Width: | Height: | Size: 330 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-8.png
vendored
Normal file
After Width: | Height: | Size: 356 KiB |
BIN
docs/static/img/guides/integrate/token-introspection-jwt-profile-9.png
vendored
Normal file
After Width: | Height: | Size: 333 KiB |