Prerequisites

In the below exercises we will be creating credential offers in different ways. Please read the credential offers chapter first that explains the concept of credential offers.

Please make sure to have read the credential offers introduction first!
These exercises assume you have installed a wallet that is capable of OID4VCI and compatible in specification version with our Issuer software. If in doubt we suggest to use the Sphereon Wallet from the Android/iOS stores
We suggest to use a tool like Postman or curl to perform these exercises and examples. Although we provide an integrated UI in these docs and links to our swagger hub, we are also developers and in our experience new learnings stick better whilst actually doing instead of clicking.

Credential offers and session initiation

In the below exercises you will learn how to issue credentials by value and by reference. You will actually get them in a mobile wallet!

Exercise 1: Credential offer by value, using pre-authorized_code

Please use the create-credential-offer API (docs) or API (swaggerhub) to create a simple so-called credential offer by value using a pre-authorized_code grant.

This means using an offerMode of "VALUE" and a grants object containing an object of urn:ietf:params:oauth:grant-type:pre-authorized_code with a pre-authorized_code value of your choosing. Use a credential_configuration_id with value "PensionSdJwt".

Right now you will have to provide a random pre-authorized_code value of your choosing. Make sure to keep this value at hand because you will need it later. You can re-use this value during the create-offer phase to overwrite any existing session
The OpenAPI documentation provides example values for the pre-authorized_code. Since we allow overwriting of sessions with this value, do not use the default values, as others might be creating new sessions whilst you are performing the exercise!
If you are running a local agent with ngrok, you cannot use the Swaggerhub examples directly. What you need to do is load the OpenAPI definition through the swagger editor like this: https://petstore.swagger.io/?url=https://api.swaggerhub.com/apis/SphereonInt/OID4VCI/0.1.1 Now you should be able to execute against localhost.

The expected outcome should be an HTTP response with something like:

{
  "uri": "openid-credential-offer://?credential_offer=%7B%22credential_issuer%22%3A%22https%3A%2F%2Fagent.findynet.demo.sphereon.com%2Fdid-web%2Foid4vci%22%2C%22credential_configuration_ids%22%3A%5B%22PensionSdJwt%22%5D%2C%22grants%22%3A%7B%22urn%3Aietf%3Aparams%3Aoauth%3Agrant-type%3Apre-authorized_code%22%3A%7B%22pre-authorized_code%22%3A%22d270fee1-9185-4e60-9901-d291e1338d7a%22%7D%7D%7D"
}

The above is the actual URI value that can be used in a link or QR code (more on that later). A wallet is using that value to start the issuance process with the Issuer. This is what is called an issuer initiated flow, instead of a wallet initiated flow. Reason is that the issuer had to create the offer first. For more info see the credential offers chapter.

Copy the response payload, as we will use it later

Exercise 2: Credential offer by value, using pre-authorized_code and a different base URI

In a so-called same device flow the wallet lives on the same device (phone, desktop) as where the credential-offer is displayed. This would happen in a browser on the device. In that case putting the above credential URI value in a link is enough to trigger the wallet when clicking on it. This is because the deeplink openid-credential-offer:// typically is registered by a wallet that can handle OID4VCI in your mobile phone. On a web-wallet it doesn’t work like that. Hence why the API also supports creating a credential offer using a different base URI (or scheme).

Let’s pretend we have web-wallet at http://localhost/my-wallet. We are not going to use a real web wallet in this exercise! Now create the above credential-offer with the web-wallet base URI using the baseUri parameter

Please use the create-credential-offer API (docs) or API (swaggerhub) to create a simple so-called credential offer by value using a pre-authorized_code grant.

This means using an offerMode of "VALUE" and a grants object containing an object of urn:ietf:params:oauth:grant-type:pre-authorized_code with a pre-authorized_code value used before (we will simply overwrite the session). Use a credential_configuration_id with value "PensionSdJwt" and use a baseUri with your web-wallet address.

Note that the baseUri is a top-level property in the payload!

The expected outcome should be an HTTP response with something like:

{
  "uri": "http://localhost/web-wallet?credential_offer=%7B%22credential_issuer%22%3A%22https%3A%2F%2Fagent.findynet.demo.sphereon.com%2Fdid-web%2Foid4vci%22%2C%22credential_configuration_ids%22%3A%5B%22PensionSdJwt%22%5D%2C%22grants%22%3A%7B%22urn%3Aietf%3Aparams%3Aoauth%3Agrant-type%3Apre-authorized_code%22%3A%7B%22pre-authorized_code%22%3A%22d270fee1-9185-4e60-9901-d291e1338d7a%22%7D%7D%7D"
}

If you had a real web based wallet that can interpret the above link, you would have to authenticate first (if not already authenticated), afterwards the wallet would guide you through the issuance process.

Some wallets do not support the above links (they should ideally). They typically require you to copy the original credential offer with the openid-credential-offer://?credential-offer= schema into a specific form in the wallet. This relies on an issuer allowing you to copy the URL.

Exercise 3: Create a QR code from the URI and scan it with a mobile wallet

In a lot of situations your wallet is on a mobile device, while you might be visiting the issuer website using another device like a tablet or desktop computer. This is the so called cross-device flow. AAlthough new mechanisms are being developer whereby your mobile wallet can register with your browser, right now a credential offer is being conveyed using a QR code. This is a short exercise to show that all you really need to do is display the credential offer value in a QR code to really start the interaction with the issuer from a mobile device. Meaning that if you are not interested in tracking the progress of the issuance process, you only need the create-credential-offer endpoint.

Take the outcome of Exercise 1 and copy the value of the uri field from the JSON response. So do not copy the whole JSON response. You should only copy the value that starts with openid-credential-offer://..... also minus the quotes in front and at the end.

Go to an online QR code generator like: https://www.qr-code-generator.com/ and past the credential offer value. You should get a QR code back. Below is an example QR code. Obviously yours will differ.

The above example QR will not work as the issuer does not know about that session anymore!

Now take you mobile wallet application and make sure you are logged in. Typically the wallet app will have a QR code scanning functionality. You can use that to initiate the issuance process. Most wallets will also register with the Operating System, meaning you can also scan the QR code using your regular photo/camera app instead. Then depending on your brand the wallet will open, or you will get a choice which wallet app to open. Finish the issuance process on your mobile wallet, to see that all it took was creating a credential-offer and displaying that as a QR code.

Exercise 4: Let the API create a QR code and scan it with a mobile wallet

The issuer has built-in support to generate QR codes and return them in the API. The QR images are returned as so-called data URIs. This means they are base64 encoded. Your browser can actually display these. Simply open a seperate tab and paste the data URI value from the response minus the quotes into your address bar at the top.

Your browser will show the image, without needing an internet connection. It simply decodes the QR code into an image and shows it locally

Please use the create-credential-offer API (docs) or API (swaggerhub) to create a simple so-called credential offer by value using a pre-authorized_code grant and this time with qrCodeOpts.

This means using an offerMode of "VALUE" and a grants object containing an object of urn:ietf:params:oauth:grant-type:pre-authorized_code with a pre-authorized_code value of your choosing. Use a credential_configuration_id with value "PensionSdJwt" and add a "qrCodeOpts" object with value {"size": 400}.

Now you should get a response that looks like below:

{
  "uri": "openid-credential-offer://?credential_offer=%7B%22credential_issuer%22%3A%22https%3A%2F%2Fagent.findynet.demo.sphereon.com%2Fdid-web%2Foid4vci%22%2C%22credential_configuration_ids%22%3A%5B%22PensionSdJwt%22%5D%2C%22grants%22%3A%7B%22urn%3Aietf%3Aparams%3Aoauth%3Agrant-type%3Apre-authorized_code%22%3A%7B%22pre-authorized_code%22%3A%22d270fee1-9185-4e60-9901-d291e1338d7a%22%7D%7D%7D",
  "qrCodeDataUri": ""
}

Copy the entire value between quotes (excluding the quotes) of the qrCodeDataUri and paste that into your address bar in a new browser tab. You now should see a QR code.

If you are integrating with our API, you thus have the choice to use your own QR code generator library, or let the API generate them for you. All you then need to do is display them in your frontend

You can now scan the QR code with your wallet or mobile device and you should get an additional credential issued to your wallet.

Exercise 5: Protect the QR code and interaction with a transaction code

As explained in the issuer introduction documentation, a QR code isn’t the most safe method of interacting, especially not with a pre-authorized code like we are using right now. The issuer API has support for transaction/pin codes. This mitigates someone taking a picture over your sholder, and impersonating you. Remember with a pre-authorized code flow, you have already authenticated on a website for instance. Before you started scanning the QR code. By using a transaction/pin code you have an additional value that needs to be entered in the wallet. The idea is that the pincode is send out of bound, so that an attacker that can see the QR code doesn’t see it. So a text-message/SMS or e-mail for instance. Using hidden toggle on the website also works, but obviously is less secure.

Please use the create-credential-offer API (docs) or API (swaggerhub) to create a simple so-called credential offer by value using a pre-authorized_code grant and this time with qrCodeOpts and with “.

This means using an offerMode of "VALUE" and a grants object containing an object of urn:ietf:params:oauth:grant-type:pre-authorized_code with a pre-authorized_code value of your choosing. Use a credential_configuration_id with value "PensionSdJwt" and add a "qrCodeOpts" object with value {"size": 400} and a new transaction code object called tx_code and value

{
  "input_mode": "numeric",
  "length": 4
}

The tx_code needs to be part of the urn:ietf:params:oauth:grant-type:pre-authorized_code object.

Now you should get a response that looks like below:

{
  "uri": "openid-credential-offer://?credential_offer=%7B%22credential_issuer%22%3A%22https%3A%2F%2Fagent.findynet.demo.sphereon.com%2Fdid-web%2Foid4vci%22%2C%22credential_configuration_ids%22%3A%5B%22PensionSdJwt%22%5D%2C%22grants%22%3A%7B%22urn%3Aietf%3Aparams%3Aoauth%3Agrant-type%3Apre-authorized_code%22%3A%7B%22pre-authorized_code%22%3A%22d270fee1-9185-4e60-9901-d291e1338d7a%22%7D%7D%7D",
  "qrCodeDataUri": "",
  "txCode": {
    "input_mode": "numeric",
    "length": 4
  },
  "userPin": "7029"
}

Note the addition of the txCode in the response, that confirms a numeric transaction code of length 4 is being used. Also note that the API returns the random pin of that length.

Scan the QR code with the wallet. During the issuance process you should now be asked to enter the pin code in the wallet.

The Sphereon wallet currently only supports pincodes in numeric form and of length 4. The API supports other lengths as well, but the wallet will not be able to handle these. The API does not support text input mode and will still use numeric

Exercise 6: Create an offer by reference

Up until now we have been creating offers by value. It also has the option to create them by reference. Please read the credential offers chapter again if you do not know the difference anymore. In the previous exercise we created a credential that was pretty large. If you need to incorporate that into a website this might not be the most appealing. You will notice that if you create the same QR code with for instance a size of 150 instead of 400 it becomes nicer to incorporate, but at the same time a lot of devices now might have problems in actually properly scanning the QR code. The reason is that there is quite a bit of information in the QR code. Let’s fix that by using a Credential Offer by reference instead of value. This means the Issuer will provide a URL called the credential_offer_uri in the credential-offer object. That URI will host the actual offer and the wallet will resolve that URI to fetch the offer using a GET request.

Our issuer creates a random correlation Id and uses that in the URL. There is an option to provide your own correlationId as well. This correlation Id is only used in this URL. In subsequent steps the wallet will not use or see this value. It will rely on other state and nonce values. You backend can use the correlationId to track progress for instance, without ever knowing the state values.

Please use the create-credential-offer API (docs) or API (swaggerhub) to create a simple so-called credential offer by reference using a pre-authorized_code grant and this time with qrCodeOpts.

This means using an offerMode of "REFERENCE" and a grants object containing an object of urn:ietf:params:oauth:grant-type:pre-authorized_code with a pre-authorized_code value of your choosing. Use a credential_configuration_id with value "PensionSdJwt" and add a "qrCodeOpts" object with value {"size": 400}.

Now you should get a response that looks like below:

{
  "uri": "openid-credential-offer://?credential_offer_uri=%7Bhttps%3A%2F%2Fagent.findynet.demo.sphereon.com%2Fdid-web%2Foid4vci%22%2C%22randomvalue%22%22%7D%7D%7D"
}

Notice the credential_offer_uri in the response instead of credential_offer. Also notice that the string length is shorter than before. If you look at the QR code you should see there is less data in it. This allows you to use it in a smaller sized QR code, without having issues with users not being able to scan them!

Credential offer session status

Up until now we only have been using one endpoint to start the issuance process from the API and then we received the credentials in our wallet. Typically you will have a portal or web-app that is guiding the user. This is the case both in the same-device as cross-device flow. When you create web frontend for the issuance process, you will want your frontend to react to states changes in the issuance process, to inform the user about errors, success etc.

The create-credential-offer endpoint we used below internally creates a session. The session contains all kind of information needed by the issuer; one of the important properties it has in the session is the state. The state can have the following values:

stateDescription
OFFER_CREATEDAn offer is created. This is the initial state
ACCESS_TOKEN_REQUESTEDOptional state, given the token endpoint could also be on a separate AS
ACCESS_TOKEN_CREATEDOptional state, given the token endpoint could also be on a separate AS
CREDENTIAL_REQUEST_RECEIVEDCredential request received. Next state would either be error or issued
CREDENTIAL_ISSUEDThe credential iss issued from the issuer’s perspective
ERRORAn error occurred
NOTIFICATION_CREDENTIAL_ACCEPTEDThe holder/user stored the credential in the wallet (If notifications are enabled)
NOTIFICATION_CREDENTIAL_DELETEDThe holder/user did not store the credential in the wallet (If notifications are enabled)
NOTIFICATION_CREDENTIAL_FAILUREThe holder/user encountered an error (If notifications are enabled)
The above states are more or less in order. The credential issued or error are mutually exclusive. The notifications are an optional feature. It can be enabled on the issuer side. Once enabled a wallet does not have to respect it because of privacy reasons. Be aware that you thus cannot rely on them only for success/storage by the user or for failure and/or deletion of the credential.

Exercise 7: Get the initial status

In this exercise we will retrieve the initial status of the credential offer. Without a wallet involved. The state will be OFFER_CREATED

Please use the credential-offer-status API (docs) or API (swaggerhub) to check the progress of the session using the pre-authorized_code value you supplied during credential offer creation. You could also create a new offer as explained in the above exercises. It is wise to create the QR code using the API

You should get a response like this:

{
  "createdAt":1739044208549,
  "lastUpdatedAt":1739044208549,
  "status":"OFFER_CREATED"
}

The status is from the table above.

Exercise 8: Check the status whilst accepting the credential in the wallet

In this exercise we will continue where the previous exercise ended. We will use the wallet to scan the QR code. Make sure that you create the QR code from the offer, or use the QR code the API supplied (see earlier exercises in case of doubt). Scan the QR code with the mobile wallet. Once the wallet is in the accept/decline screen, you should see the state as CREDENTIAL_ISSUED

Please use the credential-offer-status API (docs) or API (swaggerhub) to check the progress of the session using the pre-authorized_code value you supplied during credential offer creation.

You should get a response like this:

{
  "createdAt":1739044208549,
  "lastUpdatedAt":1739044208549,
  "status":"CREDENTIAL_ISSUED"
}

The status is from the table above.

The reason the status is CREDENTIAL_ISSUED is because it actually is issued from the perspective of the Issuer at this point. The wallet does not know (yet) whether the holder will accept or decline the credential. If you now accept the credential in the wallet the status value will move to NOTIFICATION_CREDENTIAL_ACCEPTED if you call the API, but only if the notification endpoint is enabled and only if the wallet has both implemented that feature and enabled it.

Advanced Exercise 9: Add the previously designed credential to the issuer

In the credential design we have thought about a new credential. If you are running the agent locally and via ngrok, then you can attempt to perform this exercise.

Be aware that the current approach to configure the agent is not going to be the final solution

Open the file packages/agent/conf/exercises/oid4vci_metadata/example-issuer.json. Go to the credential_configurations_supported JSON structure. Now copy the entire “PensionSdJwt” object including the key. Add it above or below the PensionSdJwt. Name it after what your credential represents.

Now adjust the values, mainly in the display section. If you want to use images and logo’s, make sure to have uploaded them to a URL somewhere. For instance use imagebb.com Make sure to also update the vct value to the name of your credential. This should be the same as the key you used. Then update the claims object so that it contains the correct values for your credential.

After you are done with that part go to the bottom of the file to the template_mappings section. Copy

{
  "credential_types": [
    "PensionSdJwt"
  ],
  "template_path": "PensionSdJwt.hbs.json",
  "format": "vc+sd-jwt"
}

To a new object. Rename PensionSDJwt for both the credential_types and template_path into the name of your credential.

Now copy the file packages/agent/conf/exercises/templates/PensionSdJwt.hbs.json into a file YourCredentialName.hbs.json that you referenced above

Update the hbs file and make sure you are putting the correct claims in there that you designed before and that match the claims in the metadata above.

Now stop the agent and restart the agent again. This imports the new definition/metadata and enables the new template file.

You should now be able to create a new offer. Make sure to provide the correct configuration_id in the payload instead of the PensionSdJwt. Also make sure that the credentialDataSupplier values are updated to reflect the values you would like to see in the issued credential. If everything went well, you should get your branded credential in your wallet.