GithubHelp home page GithubHelp logo

fido2-node-clients's Introduction

FIDO2 NodeJS Clients

This repository contains command-line NodeJS clients for exercising FIDO2 registration and login flows for both IBM Security Verify Access (ISVA), and IBM Security Verify (ISV).

Getting started

Follow the steps in this section for one-time setup of your client.

Establishing FIDO Attestation artifacts

The first step to using the assets in this repository is to generate yourself a set of attestation keys for the various attestation types supported by the client. A script is provided that run on MacOS and Linux.

After cloning the repository change into the certs directory and run the ./generate_attestation_certs.sh script. An example is shown here:

[certs] $ ./generate_attestation_certs.sh 
Creating Root CA key: rootCA.key
Creating Root CA certificate: rootCA.pem
Creating U2F key: u2f.key
Creating U2F CSR: u2f.csr
Creating U2F certificate: u2f.pem
Certificate request self-signature ok
subject=C=US, O=IBM, CN=U2F-SIGNER
Creating packed key: packed.key
Creating packed CSR: packed.csr
Creating packed certificate: packed.pem
Certificate request self-signature ok
subject=C=US, O=IBM, OU=Authenticator Attestation, CN=PACKED-SIGNER
Creating packed attestation aaguid: packed.aaguid
Creating self attestation aaguid: self.aaguid
Generating U2F metadata file: fidotest-u2f.json
Generating PACKED metadata file: fidotest-packed.json
Generating SELF metadata file: fidotest-self.json
Generating encryption passphrase file: encpass.txt
FIDO2_CLIENT_CONFIG={"encryptionPassphrase":"feec44a74f1494cfbfe40878a4bb09a430ab4b1c","fido-u2f":{"privateKeyHex":"dbab46b3ebf409cf682bd7e59d97989862f344a98d4191932c3a31658164464a","publicKeyHex":"04bb7bcf834da96720b8119c7c2aeb09a61282207a3bbfc04ac48116c0adaeddabf2b765a7a187570d78c7cba67a6d23a3bc21c17d4f88a1b1d8afe9036638d892","cert":"MIIBpTCCAUqgAwIBAgIUfX2/5BmUao8UrUMicKfVCac1+ScwCgYIKoZIzj0EAwIwLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1QwIBcNMjQwNzAzMDI0OTU2WhgPMjA1MTExMTgwMjQ5NTZaMDAxCzAJBgNVBAYTAlVTMQwwCgYDVQQKDANJQk0xEzARBgNVBAMMClUyRi1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAAS7e8+DTalnILgRnHwq6wmmEoIgeju/wErEgRbAra7dq/K3Zaehh1cNeMfLpnptI6O8IcF9T4ihsdiv6QNmONiSo0IwQDAdBgNVHQ4EFgQUC3H699tIiWu7FTHLtwmGLaec9nIwHwYDVR0jBBgwFoAUTx3eQejb1vXgcgrexh+GvY23WwEwCgYIKoZIzj0EAwIDSQAwRgIhAK7xndLIdvSABJDiyY2Ekr5S1lpD41//fhGIlGHaFF7SAiEAn0BUFb7ro5DyGByhZJaSi+sDSisdDlaQDD/bFkx4Xew="},"packed":{"aaguid":"3e25b68b-0920-4bbd-88ad-46f6e1c8f154","privateKeyHex":"5a9526f4e6dc1a9e15c1f754c291e54fc7b4480e47b66357afc57464280ae0c5","publicKeyHex":"044f4f301c0a9b65e8ae4898b041dda77758e68e668287516e416f4e463c28d69a8aab077ff8cdaa6b942f0c0b492459a26f56535f3b03fea58142a7b5796b83cb","cert":"MIIB2DCCAX+gAwIBAgIUfX2/5BmUao8UrUMicKfVCac1+SgwCgYIKoZIzj0EAwIwLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1QwIBcNMjQwNzAzMDI0OTU2WhgPMjA1MTExMTgwMjQ5NTZaMFcxCzAJBgNVBAYTAlVTMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRpb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAARPTzAcCptl6K5ImLBB3ad3WOaOZoKHUW5Bb05GPCjWmoqrB3/4zaprlC8MC0kkWaJvVlNfOwP+pYFCp7V5a4PLo1AwTjAMBgNVHRMBAf8EAjAAMB0GA1UdDgQWBBTCyXyJh9P1lzYOxGkB1Dneq9REgjAfBgNVHSMEGDAWgBRPHd5B6NvW9eByCt7GH4a9jbdbATAKBggqhkjOPQQDAgNHADBEAiAWbACcza6fXogwHuHQA355777eVdYo/DUvAIoaWs6unQIgG5Zm7uIKDkvpv+6REDIxirtzTD3P2ePl7EvC+tE1uw8="},"packed-self":{"aaguid":"49334923-79b5-48b9-98ed-97c1ec266cdc"}}

This script produces three JSON FIDO Metadata files (in MDS3 format), which can be loaded onto your FIDO2 server (either an ISV tenant, or your ISVA server):

  • fidotest-u2f.json
  • fidotest-packed.json
  • fidotest-self.json

The script also generates a configuration variable called FIDO2_CLIENT_CONFIG which will be used in the next step.

Note: There is also a Node.JS equivalent of the bash script provided called generate_attestation_certs.js. This should be functionally equivalent to generate_attestation_certs.sh except that it doesn't generate unnecessary CSR files, nor does it require some of the input files used by the openssl command line utility. You can use it as an alternative to generate_attestation_certs.sh simply by using Node.JS on the command line:

[certs] $ node generate_attestation_certs.js
<equivalent output as above>

Both the bash script and the Node.JS script are idemptotent - you can run them multiple times and they will only regenerate artifacts that do not already exist.

Creating the .env file

Change back to the top-level directory of the repository code, and copy the .env.example file to a new file .env

The .env file will need to be populated with configuration data for either your ISV tenant or ISVA server. The instructions in this readme are going to focus on configuration for ISV (SaaS), as each ISVA environment can be a little different. The ISVA configuration has been tested against a demonstration site that I manange (https://fidointerop.securitypoc.com) and provided it is up and running you are welcome to try running the client against that site as well. Follow the instructions in the .env.example file for how to setup the client against fidointerop.securitypoc.com.

Updating .env for your ISV Tenant

Edit the .env file and update the values of ISV_TENANT_ENDPOINT, RPID and ORIGIN to match your tenant.

Set the value of the FIDO2_CLIENT_CONFIG in the .env file to the value produced in the output of the generate_attestation_certs.sh script run during th previous step.

Pick an ISV username (you can look for your own username by checking the AZN_CRED_PRINCIPAL_NAME in the https://YOUR_TENANT/ivcreds credential viewer) and set the username value in the variable ISV_USERNAME.

Login to your tenant as an administrator, navigate to the admin UI, then Security -> API Access and create an OAuth API client with at least the following entitlements:

  • Read users and groups (readUserGroups)
  • Manage second-factor authentication enrollment for all users (manageEnrollMFAMethodAnyUser)
  • Authenticate any user (authnAnyUser)

Gather the OAuth client ID and secret, and put their values in the .env files for the variables OAUTH_CLIENT_ID and OAUTH_CLIENT_SECRET respectively. If you have an OAUTH_CLIENT_ID and OAUTH_CLIENT_SECRET then you do not need the OIDC_USER_ACCESS_TOKEN variable and can leave it commented out.

If you don't have administrative access to the tenant you are working with, but do have a normal user account and can login to the tenant, then there is still a way to exercise the client. Visit the security settings page of your user profile (https://YOUR_TENANT/usc/settings/security) with the network debugger in your browser open, and look for any of the XHR requests (eg. the one for registrations), and extract the value of the OAuth access token from the Authorization Request Header (the part AFTER Bearer ). You can set this value in the .env file variable OIDC_USER_ACCESS_TOKEN and comment out the OAUTH_CLIENT_ID and OAUTH_CLIENT_SECRET entries. In this case you MUST set the ISV_USERNAME value to your own username. Note that using this technique, you will need to occassionally get a new access token and update it in .env since they will expire with your browser session.

Installing required node packages

In testing, I was using node v21.7.2, though anything "current" should work. From the top-level directory of the repository, install required node packages with npm install. Example:

[fido2-node-clients] $ npm install

added 5 packages, and audited 6 packages in 468ms

2 packages are looking for funding
  run `npm fund` for details

found 0 vulnerabilities

Running the first example

Provided you have followed the configuration instructions above, you should now be ready to run the first example program, which creates a FIDO2 registration against your account, and then immediately also performs an assertion flow.

Run the example as shown (note that I have reformatted the output to pretty-print the JSON strings and added some blank lines for readability):

[fido2-node-clients] $ node isv_example1.js 

2024-07-03T04:30:20.769Z performAttestation sending attestation options to ISV:
{
  "userId": "50BCRQVNC0",
  "authenticatorSelection": {
    "userVerification": "required",
    "requireResidentKey": true
  }
}


2024-07-03T04:30:21.319Z performAttestation sending attestation result to ISV: 
{
  "id": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
  "rawId": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
  "response": {
    "clientDataJSON": "eyJvcmlnaW4iOiJodHRwczovL215aWRwLmljZS5pYm1jbG91ZC5jb20iLCJjaGFsbGVuZ2UiOiJtUGc0eWdRMGRkbXd5eURRZG4xMkEtN0FBYkZDX19qTF9DckJrbDJUMzZVIiwidHlwZSI6IndlYmF1dGhuLmNyZWF0ZSJ9",
    "attestationObject": "o2NmbXRmcGFja2VkZ2F0dFN0bXSjY2FsZyZjeDVjgZkB3BgwGIIBGNgYMBiCARh_GKADAgECAhQYfRh9GL8Y5BgZGJQYahiPFBitGEMYIhhwGKcY1QkYpxg1GPkYKBgwCgYIGCoYhhhIGM4YPQQDAhgwGC4YMQsYMAkGAxhVBAYTAhhVGFMYMQwYMAoGAxhVBAoMAxhJGEIYTRgxERgwDwYDGFUEAwwIGEYYSRhEGE8YVBhFGFMYVBgwGCAXDRgyGDQYMBg3GDAYMxgwGDIYNBg5GDUYNhhaGBgPGDIYMBg1GDEYMRgxGDEYOBgwGDIYNBg5GDUYNhhaGDAYVxgxCxgwCQYDGFUEBhMCGFUYUxgxDBgwCgYDGFUECgwDGEkYQhhNGDEYIhgwGCAGAxhVBAsMGBkYQRh1GHQYaBhlGG4YdBhpGGMYYRh0GG8YchggGEEYdBh0GGUYcxh0GGEYdBhpGG8YbhgxFhgwFAYDGFUEAwwNGFAYQRhDGEsYRRhEGC0YUxhJGEcYThhFGFIYMBhZGDATBgcYKhiGGEgYzhg9AgEGCBgqGIYYSBjOGD0DAQcDGEIABBhPGE8YMBgcChibGGUY6BiuGEgYmBiwGEEY3RinGHcYWBjmGI4YZhiCGIcYURhuGEEYbxhOGEYYPBgoGNYYmhiKGKsHGH8Y-BjNGKoYaxiUGC8MCxhJGCQYWRiiGG8YVhhTGF8YOwMY_hilGIEYQhinGLUYeRhrGIMYyxijGFAYMBhOGDAMBgMYVRgdEwEBGP8EAhgwABgwGB0GAxhVGB0OBBYEFBjCGMkYfBiJGIcY0xj1GJcYNg4YxBhpARjUGDkY3hirGNQYRBiCGDAYHwYDGFUYHRgjBBgYGDAWGIAUGE8YHRjeGEEY6BjbGNYY9RjgGHIKGN4YxhgfGIYYvRiNGLcYWwEYMAoGCBgqGIYYSBjOGD0EAwIDGEcAGDAYRAIYIBYYbAAYnBjNGK4YnxheGIgYMBgeGOEY0AMYfhh5GO8YvhjeGFUY1hgoGPwYNRgvABiKGBoYWhjOGK4YnQIYIBgbGJYYZhjuGOIKDhhLGOkYvxjuGJEQGDIYMRiKGLsYcxhMGD0YzxjZGOMY5RjsGEsYwhj6GNEYNRi7D2NzaWeYRhgwGEQCGCAYaxi8GEsY5hiRGH4Y6BhJGB0YjRhUGJ4YlhiqGBkYdBhcGGgY5xjWGPQYnxjeGKcY3hibGIkY4BjOGEIY6xi5AhggGHkYuxiNGI0YKAUYoBiODRhRGLkY3xhyGH8YaBjGGMYYxhgvGEAY6RgYGOwYNRirExiBGLQYTxhqGOkYg2hhdXRoRGF0YZkBKBjeGHkY4RjNFxjnGBgYKxgwGIsY-BioGEAYGhjsGEUYXxiUGGsYGRhFGJoY1xgqGNQYSRhiDBi3GF8Ymxg8GEUYZhiEGNMY3Rg-GCUYthiLCRggGEsYvRiIGK0YRhj2GOEYyBjxGFQAGGAYUxhhGGwYdBhlGGQYXxhfERjOGOMOGDsYHBi4GLgYKxg_GNMLGK4YoxhKGNsYoBgqGLAYJRivAhhfGFwY9BiuGK4YkhAYsxijGPcYwBEYlhhUGOMYwg8YjhhbFBi4GCAYTRi2GE8YIxg2GK8YGxhVGHoYKxhQGIYEGBwYSxi8GI0YcBg5GHcYtRg2GHQYWRgbGHIYShjDGLIYVBjwGEEY_hhRGO0YNRhuGKcY1xgjGE0YPRj6GIIYpRhhGDECGGEYMxgmGGIYLRgxARhiGC0YMhiYGCAYGBiVGBgYlBIYGBhsGBgY6RgYGEcYGBieGBgYOBgYGFgYGBgZGBgYvhgYGOMYGBjOGBgYjRgYGBwYGBi5GBgYaBgYGJoYGBh2GBgYVxgYGGMYGBjMGBgYTxgYGMsYGBghGBgY4gMYGBhVGBgYLhgYGEcYGBi4GBgYuBhiGC0YMxiYGCAYGBgqGBgYjRgYGGEYGBgfGBgYKRgYGKcBGBgYpRgYGNUYGBjpGBgYyRgYGDwYGBhmGBgYqBgYGLMYGBjMGBgYmRgYGNMYGBg_GBgYWhgYGLgYGBhnGBgYahgYGFgYGBhcGBgYYw0YGBiWGBgYTxgYGLwYGBjZGBgY6g"
  },
  "type": "public-key",
  "getClientExtensionResults": {},
  "nickname": "NodeClient - mPg4ygQ0ddmwyyDQdn12A-7AAbFC__jL_CrBkl2T36U",
  "enabled": true,
  "getTransports": [
    "platform"
  ]
}


2024-07-03T04:30:22.067Z performAttestation got attestationResultResponse: 
{
  "id": "0f1ab6c2-cba6-4469-b09f-2f05a1718f37",
  "userId": "50BCRQVNC0",
  "type": "fido2",
  "created": "2024-07-03T04:30:21.928Z",
  "updated": "2024-07-03T04:30:21.928Z",
  "attempted": "2024-07-03T04:30:21.936Z",
  "enabled": true,
  "validated": true,
  "attributes": {
    "attestationType": "Basic",
    "attestationFormat": "packed",
    "nickname": "NodeClient - mPg4ygQ0ddmwyyDQdn12A-7AAbFC__jL_CrBkl2T36U",
    "userVerified": true,
    "userPresent": true,
    "credentialId": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
    "credentialPublicKey": "v2EzYi03Yi0xAWItMlgglZQSbOlHnjhYGb7jzo0cuWiadldjzE/LIeIDVS5HuLhhMQJhMyZiLTNYICqNYR8ppwGl1enJPGaos8yZ0z9auGdqWFxjDZZPvNnq/w==",
    "rpId": "myidp.ice.ibmcloud.com",
    "counter": 1719981021,
    "aaGuid": "3E25B68B-0920-4BBD-88AD-46F6E1C8F154",
    "icon": "",
    "description": "FIDOTEST-PACKED",
    "x5c": [
      "-----BEGIN CERTIFICATE-----\nMIIB2DCCAX+gAwIBAgIUfX2/5BmUao8UrUMicKfVCac1+SgwCgYIKoZIzj0EAwIw\nLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1Qw\nIBcNMjQwNzAzMDI0OTU2WhgPMjA1MTExMTgwMjQ5NTZaMFcxCzAJBgNVBAYTAlVT\nMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRp\nb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMB\nBwNCAARPTzAcCptl6K5ImLBB3ad3WOaOZoKHUW5Bb05GPCjWmoqrB3/4zaprlC8M\nC0kkWaJvVlNfOwP+pYFCp7V5a4PLo1AwTjAMBgNVHRMBAf8EAjAAMB0GA1UdDgQW\nBBTCyXyJh9P1lzYOxGkB1Dneq9REgjAfBgNVHSMEGDAWgBRPHd5B6NvW9eByCt7G\nH4a9jbdbATAKBggqhkjOPQQDAgNHADBEAiAWbACcza6fXogwHuHQA355777eVdYo\n/DUvAIoaWs6unQIgG5Zm7uIKDkvpv+6REDIxirtzTD3P2ePl7EvC+tE1uw8=\n-----END CERTIFICATE-----"
    ],
    "transports": []
  },
  "references": {
    "rpUuid": "117b66c2-7e5b-41fa-a0ee-d321f1f5e6ce"
  }
}


2024-07-03T04:30:22.068Z performAssertion sending assertion options to ISV: 
{
  "userVerification": "required",
  "userId": "50BCRQVNC0"
}

2024-07-03T04:30:22.543Z performAssertion: assertionOptionsResponse: 
{
  "rpId": "myidp.ice.ibmcloud.com",
  "timeout": 240000,
  "challenge": "qoMVtSzjumYDEDG5DgKXJ2sgnKwdDfUrqSezgocCIHY",
  "allowCredentials": [
    {
      "id": "_d-41HkhN5CXzP7WhR9fo75Y5b3mSRmCo6VV4WSNkdcvGeD0r1sCLOdvdLLf9pyBUgVywIUn5vnvqoPL5QkHQw",
      "type": "public-key"
    },
    {
      "id": "sYUkLRssFBIBFb2SB3inia7MHvg",
      "type": "public-key"
    },
    {
      "id": "WVOED0418_EnmeDfgY_9oA",
      "type": "public-key"
    },
    {
      "id": "T74dYxB79zt7ZFPQyTgR8g",
      "type": "public-key"
    },
    {
      "id": "LrbD3qb3frQYOW5cufR5qJjLhfrHJN2Vx43g8BvHKN4",
      "type": "public-key"
    },
    {
      "id": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
      "type": "public-key"
    },
    {
      "id": "JVigN6QEf6vy5U6pAFTKAMoaGmQ",
      "type": "public-key"
    },
    {
      "id": "dF7MXlvzrDIcfjDoFUsU5g",
      "type": "public-key"
    }
  ],
  "extensions": {},
  "userVerification": "required"
}


2024-07-03T04:30:22.587Z performAssertion sending assertion result to ISV: 
{
  "id": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
  "rawId": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
  "response": {
    "clientDataJSON": "eyJvcmlnaW4iOiJodHRwczovL215aWRwLmljZS5pYm1jbG91ZC5jb20iLCJjaGFsbGVuZ2UiOiJxb01WdFN6anVtWURFREc1RGdLWEoyc2duS3dkRGZVcnFTZXpnb2NDSUhZIiwidHlwZSI6IndlYmF1dGhuLmdldCJ9",
    "authenticatorData": "3nnhzRfnGCswi_ioQBrsRV-UaxlFmtcq1EliDLdfmzwFZoTT3g",
    "signature": "MEYCIQDBCB8GPKtbYmM2_W0xkbBi2vCmPACefg5r_Akww3hJLAIhAOulPDU6SkzmIxrwHcl6O4lb_QxMD_BO_zpxPCzXmKdZ",
    "userHandle": ""
  },
  "type": "public-key",
  "getClientExtensionResults": {}
}


2024-07-03T04:30:23.105Z performAssertion got assertionResultResponse: 
{
  "id": "0f1ab6c2-cba6-4469-b09f-2f05a1718f37",
  "userId": "50BCRQVNC0",
  "type": "fido2",
  "created": "2024-07-03T04:30:21.928Z",
  "updated": "2024-07-03T04:30:21.928Z",
  "attempted": "2024-07-03T04:30:22.998Z",
  "enabled": true,
  "validated": true,
  "attributes": {
    "attestationType": "Basic",
    "attestationFormat": "packed",
    "nickname": "NodeClient - mPg4ygQ0ddmwyyDQdn12A-7AAbFC__jL_CrBkl2T36U",
    "userVerified": true,
    "userPresent": true,
    "credentialId": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
    "credentialPublicKey": "v2EzYi03Yi0xAWItMlgglZQSbOlHnjhYGb7jzo0cuWiadldjzE/LIeIDVS5HuLhhMQJhMyZiLTNYICqNYR8ppwGl1enJPGaos8yZ0z9auGdqWFxjDZZPvNnq/w==",
    "rpId": "myidp.ice.ibmcloud.com",
    "counter": 1719981022,
    "aaGuid": "3E25B68B-0920-4BBD-88AD-46F6E1C8F154",
    "icon": "",
    "description": "FIDOTEST-PACKED",
    "x5c": [
      "-----BEGIN CERTIFICATE-----\nMIIB2DCCAX+gAwIBAgIUfX2/5BmUao8UrUMicKfVCac1+SgwCgYIKoZIzj0EAwIw\nLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1Qw\nIBcNMjQwNzAzMDI0OTU2WhgPMjA1MTExMTgwMjQ5NTZaMFcxCzAJBgNVBAYTAlVT\nMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRp\nb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMB\nBwNCAARPTzAcCptl6K5ImLBB3ad3WOaOZoKHUW5Bb05GPCjWmoqrB3/4zaprlC8M\nC0kkWaJvVlNfOwP+pYFCp7V5a4PLo1AwTjAMBgNVHRMBAf8EAjAAMB0GA1UdDgQW\nBBTCyXyJh9P1lzYOxGkB1Dneq9REgjAfBgNVHSMEGDAWgBRPHd5B6NvW9eByCt7G\nH4a9jbdbATAKBggqhkjOPQQDAgNHADBEAiAWbACcza6fXogwHuHQA355777eVdYo\n/DUvAIoaWs6unQIgG5Zm7uIKDkvpv+6REDIxirtzTD3P2ePl7EvC+tE1uw8=\n-----END CERTIFICATE-----"
    ],
    "transports": []
  },
  "references": {
    "rpUuid": "117b66c2-7e5b-41fa-a0ee-d321f1f5e6ce"
  }
}

2024-07-03T04:30:23.106Z assertionResultResponse: 
<REDACTED - SAME content as above>

2024-07-03T04:30:23.107Z ==============================================================

2024-07-03T04:30:23.107Z authenticatorRecords: 
{
  "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC": {
    "rpId": "myidp.ice.ibmcloud.com",
    "privateKeyHex": "f74a1c5ed2e9efc4e51e7d39b743d750968c6fb492e685b9f6bffa5470c6e4eb",
    "credentialID": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
    "userHandle": "NTBCQ1JRVk5DMA"
  }
}

You can also login to the security settings page of your account on the ISV tenant using your browser and see the new registration:

example1_registration

An interesting exercise is to use the fido2viewer that I have previously shared and use it to both unpack the attestation object from the registration request, and independently verify the signature of the assertion request.

Running the second example

The second example is isv_example2.js and this is just a very simple assertion flow. It will require you extract the authenticatorRecords from the previous example (this is the last JSON object printed in the example above), and update that in the isv_example2.js file.

You can then run the example which is actually just a repeat of the last step of example1 (again I have reformatted this output):

[fido2-node-clients] $ node isv_example2.js 
2024-07-03T04:44:44.164Z performAssertion sending assertion options to ISV:
{
  "userVerification": "required"
}

2024-07-03T04:44:44.538Z performAssertion: assertionOptionsResponse: 
{
  "rpId": "myidp.ice.ibmcloud.com",
  "timeout": 240000,
  "challenge": "qxJiJ5l9AXn98ZFJRkWmT_Bno7-eCJoXBhyAyu9oqgo",
  "userVerification": "required"
}

2024-07-03T04:44:44.595Z performAssertion sending assertion result to ISV: 
{
  "id": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
  "rawId": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
  "response": {
    "clientDataJSON": "eyJvcmlnaW4iOiJodHRwczovL215aWRwLmljZS5pYm1jbG91ZC5jb20iLCJjaGFsbGVuZ2UiOiJxeEppSjVsOUFYbjk4WkZKUmtXbVRfQm5vNy1lQ0pvWEJoeUF5dTlvcWdvIiwidHlwZSI6IndlYmF1dGhuLmdldCJ9",
    "authenticatorData": "3nnhzRfnGCswi_ioQBrsRV-UaxlFmtcq1EliDLdfmzwFZoTXPA",
    "signature": "MEQCIC-fcT-U5nXeOc8TAoBMxezUYcbENw0lHOJ57sx4ZTnjAiBPLGO0IdNATXbqFz_HkyP3-jg-OjuygIJL1v7eRjsKmA",
    "userHandle": "NTBCQ1JRVk5DMA"
  },
  "type": "public-key",
  "getClientExtensionResults": {}
}

2024-07-03T04:44:45.104Z performAssertion got assertionResultResponse: 
{
  "id": "0f1ab6c2-cba6-4469-b09f-2f05a1718f37",
  "userId": "50BCRQVNC0",
  "type": "fido2",
  "created": "2024-07-03T04:30:21.928Z",
  "updated": "2024-07-03T04:30:21.928Z",
  "attempted": "2024-07-03T04:44:44.964Z",
  "enabled": true,
  "validated": true,
  "attributes": {
    "attestationType": "Basic",
    "attestationFormat": "packed",
    "nickname": "NodeClient - mPg4ygQ0ddmwyyDQdn12A-7AAbFC__jL_CrBkl2T36U",
    "userVerified": true,
    "userPresent": true,
    "credentialId": "U2FsdGVkX18RzuMOOxy4uCs_0wuuo0rboCqwJa8CX1z0rq6SELOj98ARllTjwg-OWxS4IE22TyM2rxtVeitQhgQcS7yNcDl3tTZ0WRtySsOyVPBB_lHtNW6n1yNNPfqC",
    "credentialPublicKey": "v2EzYi03Yi0xAWItMlgglZQSbOlHnjhYGb7jzo0cuWiadldjzE/LIeIDVS5HuLhhMQJhMyZiLTNYICqNYR8ppwGl1enJPGaos8yZ0z9auGdqWFxjDZZPvNnq/w==",
    "rpId": "myidp.ice.ibmcloud.com",
    "counter": 1719981884,
    "aaGuid": "3E25B68B-0920-4BBD-88AD-46F6E1C8F154",
    "icon": "",
    "description": "FIDOTEST-PACKED",
    "x5c": [
      "-----BEGIN CERTIFICATE-----\nMIIB2DCCAX+gAwIBAgIUfX2/5BmUao8UrUMicKfVCac1+SgwCgYIKoZIzj0EAwIw\nLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1Qw\nIBcNMjQwNzAzMDI0OTU2WhgPMjA1MTExMTgwMjQ5NTZaMFcxCzAJBgNVBAYTAlVT\nMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRp\nb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMB\nBwNCAARPTzAcCptl6K5ImLBB3ad3WOaOZoKHUW5Bb05GPCjWmoqrB3/4zaprlC8M\nC0kkWaJvVlNfOwP+pYFCp7V5a4PLo1AwTjAMBgNVHRMBAf8EAjAAMB0GA1UdDgQW\nBBTCyXyJh9P1lzYOxGkB1Dneq9REgjAfBgNVHSMEGDAWgBRPHd5B6NvW9eByCt7G\nH4a9jbdbATAKBggqhkjOPQQDAgNHADBEAiAWbACcza6fXogwHuHQA355777eVdYo\n/DUvAIoaWs6unQIgG5Zm7uIKDkvpv+6REDIxirtzTD3P2ePl7EvC+tE1uw8=\n-----END CERTIFICATE-----"
    ],
    "transports": []
  },
  "references": {
    "rpUuid": "117b66c2-7e5b-41fa-a0ee-d321f1f5e6ce"
  }
}

2024-07-03T04:44:45.104Z assertionResultResponse: 
<REDACTED - SAME content as above>

fido2-node-clients's People

Contributors

sbweeden avatar sramsamy avatar

Stargazers

Benjamin avatar Royce Williams avatar

Watchers

Royce Williams avatar  avatar

Forkers

sramsamy

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.