---
title: Client credentials grant with PingAM
description: This example shows how a client service accesses an OAuth 2.0-protected resource by using its OAuth 2.0 client credentials.
component: pinggateway
version: 2026
page_id: pinggateway:gateway-guide:oauth2-clientcredentials
canonical_url: https://docs.pingidentity.com/pinggateway/2026/gateway-guide/oauth2-clientcredentials.html
revdate: 2025-04-01T17:53:34Z
---

# Client credentials grant with PingAM

This example shows how a client service accesses an OAuth 2.0-protected resource by using its OAuth 2.0 client credentials.

![ClientCredentialsOAuth2ClientFilter](https://kroki.io/plantuml/svg/eNqNVU1z2jAQvetX7HDhUpOUQ2fKITMOM0k5MO2QHHMR9oKVKpIryTj0H_Vv9Jd1Zdnhw5iKi2H99u3btyvBbsYM5rrcG7EtHPz9A9Pbz18hocf0C_wQaguLHJUTbk8wU2rDndCKMXguhIVM5wj0dBrWCJXFHPA9k5UVO5R7EIoQSmHmc6AWrrhCCVZvXM0NgjZg0exEhnbChjNAK6qhNxs0FmyVFRcZvDaJWy6hYRBoGdSFhoLv0IfQkGihCMVhLVTuq0lKVBaBbw3iG4H-o33CxjeMUVQipBkVtR7JFXxPK8qbTm6hNNqRD1TLoNWVyfAT-eVhB0wmha-VkSJfg0vLeOW0qt7WaBgruXEiEyUnzGgesE-hyxFwCyHE1vodRotHMLpy2HImR5yTV6vViAF9ThifMiNKx9cSv3GVSzQNaS_aTwx154cKTUfTEH4Q0rVUETiGKgdq4KxZb3y6BA_XRvwO0_ett8zpkoW2PfCRO6z5vu1fc18j-bA_6ewfsiGIWrWoUOSoieHXfapn_RPVQjmjbRnOQFiOJu4pZNdAFLJfoJPRcHQ_Dh4GZ5O73gxn7SxoF39VaB3rIS4mrbA2gjxts_zJUliD42aLFzkiJj6jOar8Q0hERnKXLr2WoIE3PpEUMqo9US8qLP3YHp-to0OVLiOlrdBVRp3UiFMYxb5QrzTrE_ZwEbVmvCiaKod7pBvNnC1_gTynjYgSM7yyM3jQhi7Mg__D2OQuakln0H47dS0qt5vshXw_ssj6zT6dmbrRsQqumTXAfM2zLtp3unvTLcsxwpaa_n7YPyK3yFM=)

1. Set up the AM as an Authorization Server:

   1. Register a PingGateway agent with the following values, as described in [Register a PingGateway agent in AM](preface.html#register-agent-am):

      * Agent ID: `ig_agent`

      * Password: `password`

      * Token Introspection: `Realm Only`

        |   |                                                                                                                   |
        | - | ----------------------------------------------------------------------------------------------------------------- |
        |   | Use secure passwords in a production environment. Consider using a password manager to generate secure passwords. |

   2. Create an OAuth 2.0 Authorization Server:

      1. Select Services > Add a Service > OAuth2 Provider.

      2. Add a service with the default values.

   3. Create an OAuth 2.0 client to request access tokens, using client credentials for authentication:

      1. Select Applications > OAuth 2.0 > Clients, and add a client with the following values:

         * Client ID : `client-service`

         * Client secret : `password`

         * Scope(s) : `client-scope`

      2. (Optional) On the Core tab, switch to using a client secret associated with a secret label by setting a Secret Label Identifier and mapping the label to a secret.

         To learn more, read [Create a client profile](https://docs.pingidentity.com/pingam/8.1/am-oauth2/oauth2-register-client.html#client-secret-label-identifier) and [Map and rotate secrets](https://docs.pingidentity.com/pingam/8.1/security/secret-mapping.html) in the AM documentation.

      3. On the Advanced tab, select the following value:

         * Grant Types : `Client Credentials`

2. Set up PingGateway:

   1. Set up PingGateway for HTTPS, as described in [Configure PingGateway for TLS (server-side)](../installation-guide/securing-connections.html#server-side-tls).

   2. Set an environment variable for the PingGateway agent password, and then restart PingGateway:

      ```console
      $ export AGENT_SECRET_ID='cGFzc3dvcmQ='
      ```

      The password is retrieved by a SystemAndEnvSecretStore, and must be base64-encoded.

   3. Add the following route to PingGateway:

      * Linux

        `$HOME/.openig/config/routes/oauth2-protected-resource.json`

      * Windows

        `%appdata%\OpenIG\config\routes\oauth2-protected-resource.json`

      ```json
      {
        "name": "oauth2-protected-resource",
        "condition": "${find(request.uri.path, '^/oauth2-protected-resource')}",
        "heap": [
          {
            "name": "SystemAndEnvSecretStore-1",
            "type": "SystemAndEnvSecretStore"
          },
          {
            "name": "AmService-1",
            "type": "AmService",
            "config": {
              "agent": {
                "username": "ig_agent",
                "passwordSecretId": "agent.secret.id"
              },
              "secretsProvider": "SystemAndEnvSecretStore-1",
              "url": "http://am.example.com:8088/openam/"
            }
          }
        ],
        "handler": {
          "type": "Chain",
          "config": {
            "filters": [
              {
                "name": "OAuth2ResourceServerFilter-1",
                "type": "OAuth2ResourceServerFilter",
                "config": {
                  "scopes": [ "client-scope" ],
                  "requireHttps": false,
                  "accessTokenResolver": {
                    "name": "TokenIntrospectionAccessTokenResolver-1",
                    "type": "TokenIntrospectionAccessTokenResolver",
                    "config": {
                      "amService": "AmService-1",
                      "providerHandler": {
                        "type": "Chain",
                        "config": {
                          "filters": [
                            {
                              "type": "HttpBasicAuthenticationClientFilter",
                              "config": {
                                "username": "ig_agent",
                                "passwordSecretId": "agent.secret.id",
                                "secretsProvider": "SystemAndEnvSecretStore-1"
                              }
                            }
                          ],
                          "handler": "ForgeRockClientHandler"
                        }
                      }
                    }
                  }
                }
              }
            ],
            "handler": {
              "type": "StaticResponseHandler",
              "config": {
                "status": 200,
                "headers": {
                  "Content-Type": [ "text/html; charset=UTF-8" ]
                },
                "entity": "<html><body><h2>Access Granted</h2></body></html>"
              }
            }
          }
        }
      }
      ```

      Source: [oauth2-protected-resource.json](../_attachments/config/routes/oauth2-protected-resource.json)

      Notice the following features of the route:

      * The route matches requests to `/oauth2-protected-resource`.

      * The `OAuth2ResourceServerFilter` expects an OAuth 2.0 access token in the header of the incoming request, with the scope `client-scope`.

      * The filter uses a TokenIntrospectionAccessTokenResolver to resolve the access token. The introspect endpoint is protected with HTTP Basic Authentication, and the `providerHandler` uses an HttpBasicAuthenticationClientFilter to provide the resource server credentials.

      * For convenience in this test, `"requireHttps"` is false. In production environments, set it to true.

      * After the filter successfully validates the access token, it creates a new context from the Authorization Server response, containing information about the access token.

      * The StaticResponseHandler returns a message that access is granted.

   4. Add the following route to PingGateway:

      * Linux

        `$HOME/.openig/config/routes/client-credentials.json`

      * Windows

        `%appdata%\OpenIG\config\routes\client-credentials.json`

      ```json
      {
        "name": "client-credentials",
        "baseURI": "http://ig.example.com:8080",
        "condition" : "${find(request.uri.path, '^/client-credentials')}",
        "heap" : [ {
          "name" : "clientSecretAccessTokenExchangeHandler",
          "type" : "Chain",
          "capture" : "all",
          "config" : {
            "filters" : [ {
              "type" : "ClientSecretBasicAuthenticationFilter",
              "config" : {
                "clientId" : "client-service",
                "clientSecretId" : "client.secret.id",
                "secretsProvider" : {
                  "type" : "Base64EncodedSecretStore",
                  "config" : {
                    "secrets" : {
                      "client.secret.id" : "cGFzc3dvcmQ="
                    }
                  }
                }
              }
            } ],
            "handler" : "ForgeRockClientHandler"
          }
        }, {
          "name" : "oauth2EnabledClientHandler",
          "type" : "Chain",
          "capture" : "all",
          "config" : {
            "filters" : [ {
              "type" : "ClientCredentialsOAuth2ClientFilter",
              "config" : {
                "tokenEndpoint" : "http://am.example.com:8088/openam/oauth2/access_token",
                "endpointHandler": "clientSecretAccessTokenExchangeHandler",
                "scopes" : [ "client-scope" ]
              }
            } ],
            "handler" : "ForgeRockClientHandler"
          }
        } ],
        "handler" : {
          "type" : "ScriptableHandler",
          "config" : {
            "type" : "application/x-groovy",
            "clientHandler" : "oauth2EnabledClientHandler",
            "source" : [ "request.uri.path = '/oauth2-protected-resource'", "return http.send(context, request);" ]
          }
        }
      }
      ```

      Source: [client-credentials.json](../_attachments/config/routes/client-credentials.json)

      Note the following features of the route:

      * The route matches requests to `/client-credentials`.

      * The ScriptableHandler rewrites the request to target the `/oauth2-protected-resource` endpoint and then calls the HTTP client that has been redefined to use the oauth2EnabledClientHandler.

      * The oauth2EnabledClientHandler calls the ClientCredentialsOAuth2ClientFilter to obtain an access token from AM.

      * The ClientCredentialsOAuth2ClientFilter calls the clientSecretAccessTokenExchangeHandler to exchange tokens on the authorization endpoint.

      * The clientSecretAccessTokenExchangeHandler calls a ClientSecretBasicAuthenticationFilter to authenticate the client through the HTTP basic access authentication scheme, and a ForgeRockClientHandler to propagate the request.

      * The route `oauth2-protected-resource.json` uses the AM introspection endpoint to resolve the access token and display its contents.

3. Test the setup:

   1. In your browser's privacy or incognito mode, go to to <https://ig.example.com:8443/client-credentials>.

   2. If you see warnings that the site isn't secure, respond to the warnings to access the site.

      A message shows that access is granted.
