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).
Follow the steps in this section for one-time setup of your client.
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.
Two alternative yet compatible scripts have been provided to do this.
- certs/generate_attestation_certs.sh, a bash script which does have a dependency
openssl
and thejq
command line utilities. - certs/generate_attestation_certs.js, which you can run with NodeJS directly.
The instructions here show the use and output of the bash script, but ultimately either approach will result in usable artifacts.
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 packed attestation aaguid: packed.aaguid
Creating tpm attestation aaguid: tpm.aaguid
Creating self attestation aaguid: self.aaguid
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 TPM Intermediate key: tpminter.key
Storing TPM Intermediate public key: tpminter.pub
writing RSA key
Creating TPM Intermediate CSR: tpminter.csr
Creating TPM Intermediate certificate: tpminter.pem
Certificate request self-signature ok
subject=CN=FIDOTEST-TPM-INTERMEDIATE
Creating TPM key: tpm.key
Storing TPM public key: tpm.pub
writing RSA key
Creating TPM CSR: tpm.csr
Creating TPM certificate: tpm.pem
Certificate request self-signature ok
subject=
Generating U2F metadata file: fidotest-u2f.json
Generating PACKED metadata file: fidotest-packed.json
Generating PACKED metadata file: fidotest-tpm.json
Generating SELF metadata file: fidotest-self.json
Generating encryption passphrase file: encpass.txt
FIDO2_CLIENT_CONFIG={"encryptionPassphrase":"4eadb883df724517025ad16896922ef8fe5ee3ae","fido-u2f":{"privateKeyHex":"9b8d47700d1a6e2ae975c472025b7c7e88d564f6d5044805b8b94593551b6681","publicKeyHex":"0471d0fe8d7292247e9c12222ceb6acd13b78ed9fcd6d334a6c276a764940eef5d97bc6e0ed30d8eca32afab490989611ed4aef73ab41033b5eade349d2b5caf1b","cert":"MIIBpTCCAUqgAwIBAgIUWx0GEtI3iLZn+Dddys1/thhllTAwCgYIKoZIzj0EAwIwLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1QwIBcNMjQwOTExMDUxODA0WhgPMjA1MjAxMjcwNTE4MDRaMDAxCzAJBgNVBAYTAlVTMQwwCgYDVQQKDANJQk0xEzARBgNVBAMMClUyRi1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAARx0P6NcpIkfpwSIizras0Tt47Z/NbTNKbCdqdklA7vXZe8bg7TDY7KMq+rSQmJYR7Urvc6tBAztereNJ0rXK8bo0IwQDAdBgNVHQ4EFgQUy13UFkZJHjEcn3qxY+zE2cFlkBQwHwYDVR0jBBgwFoAUdHswx9rjRbIR3mI8xAezIHALhIYwCgYIKoZIzj0EAwIDSQAwRgIhAIJ3h2cksBk4G4DxwCzPGSpsV3Avn4IHzi1Z36G5imyGAiEAtvEq4Bj+cXhe9pgWn7qxcCS9hbLLR5aQYqx2jP1357w="},"packed":{"aaguid":"746a6ca8-0163-4ca5-8e8c-56fcad2ecbff","privateKeyHex":"20476ddd80650870333e97da45accce07c4dddbdd4af9ca5a59bf607dd2656d9","publicKeyHex":"04e1a89877d2804a9fd016a68759d6fbfcad87324a1559d9a2b1dfdb7d27f823c73cdbd5897602498b8d91b756c4715f7937618e245dce44be036957328aaef714","cert":"MIIB/TCCAaKgAwIBAgIUWx0GEtI3iLZn+Dddys1/thhllTEwCgYIKoZIzj0EAwIwLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1QwIBcNMjQwOTExMDUxODA0WhgPMjA1MjAxMjcwNTE4MDRaMFcxCzAJBgNVBAYTAlVTMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRpb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAAThqJh30oBKn9AWpodZ1vv8rYcyShVZ2aKx39t9J/gjxzzb1Yl2AkmLjZG3VsRxX3k3YY4kXc5EvgNpVzKKrvcUo3MwcTAMBgNVHRMBAf8EAjAAMCEGCysGAQQBguUcAQEEBBIEEHRqbKgBY0yljoxW/K0uy/8wHQYDVR0OBBYEFBaVz6s2LaerE341uCHf0djbJYLlMB8GA1UdIwQYMBaAFHR7MMfa40WyEd5iPMQHsyBwC4SGMAoGCCqGSM49BAMCA0kAMEYCIQC8Jw3GcygLwPcqWWoyTmsERyECRKDhYUQ9BEThZ+GhLQIhAL0PI3CdYbldlrIdlOHx6HgxpC40lMOFU72tbfq6h+Xi"},"tpm":{"aaguid":"7afd09d2-4a7d-4509-8841-4334887e6182","privateKeyPEM":"-----BEGIN PRIVATE KEY-----\r\nMIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQDT5F6pmB0643KF\r\nFyplDjgMjccm+Mz/XTgjNws8dZTdqpOnjbGXQJPC9ZzXDn8fZ1V77QlhSpq45YAq\r\ns/I8Kcp55y2svWHZu7xk9+wxWyO+DDm81R4/vM9BsJMLAdLItxKzmLl9d0z7wkOF\r\n/JbCN4orQaFM2n5BlL1akLOTQfMv3gS60xbzvzWwQooF/f/xIfXriKL7nRxB7xT/\r\nly/NAHc7e5Bet79F4NmqyJYheg2UMrGQJRZaXwxxzOif62t3T9vBy0HkqiX+WaFX\r\naMfC1mbBOl+maNDOgZukIN6ORxMkkVzsf9/Xo1tl9rGB/npG82UTbgE6yHRAexnL\r\nxEt5TatDAgMBAAECggEAOgl2jjA+bVdIEkEemRBUdMfOkRObnqRLUIXum3A/RpnG\r\nmKGcT2feMQuyGkt7sT/N78QkihPzngoqWUfxZN4zecFCxuoWZ5texi8L8BbdCtxx\r\nYhHjl6ShBgaVt/GXaOeQBUYjJETtfpWehhABwQ+SVjfDNPgxlcfmN2yj6wUo1WaU\r\nFZi8Ana45d+ZuzyhVmj1SnDQHxuiAd0bfXApk1S6vk5g7MbpQ01RJBTWnNBkVicc\r\n/Ds6bW75AFo9mYGRU+GDzA5hVOaG2gbpsVmTnvTJ1dy1wkxVhihSDWwhy3sGhIA5\r\nnmufOFyZQw1z+TrSQ/+1IHMtixArDjd91ciiJfWT6QKBgQD1Xf1hvIozxL2DKNxy\r\nNgk/UFLS65mNWqxNOYKeTVzdWt6NeDo7fgVVqkg2Kr3WiSjIyMYn0IJlc4PCtlHY\r\nHXBTNMjbirC29kjDEt9mHgNMJgzwEyFQkiIhwQ5m6bsW/Ska4CH2YqI0s8Gn8dXa\r\nVo94Awz5vo8+7PDCvwPiOaWtywKBgQDdEwUpSzg9XD35jZUtXZ9yBkMhzZoUNlTv\r\n8MFqtSLyk2+QXyMlhhb3G4UiqnyByzgvigWucf8v9J7CnRs1E5hUo15ouqeZFKIy\r\nQbkqMBFHl3ut9GmjYh46HuHKOqpRr8HnWcDMfDO0ya+B0fMb2FFBihTxze5vEpJc\r\ngg1qsA/JaQKBgQCcEQV6piatmOUFyPa2uoIVQGz99Zc8nsDbecoIEKMRtJu8Fg3M\r\nC3MvMtktgvAuxh0n4PSRHIHkL55OOhNCyC7K7dcYNVYDehF76sQnus1UIRu3lCNr\r\nkReeqqITeXVjwPDlREy137TWmsOBpEPm1Yn5xs75B2krmB/nbwH0YcBtEwKBgBNC\r\nDegL+mk1+NwdXkpmZnZbX0u5iluKZBnGwkKWZuX0u/o6n2gsZyG9jvp3GYOA/XL+\r\ntgH9QjqeN5pCb1VPUKzG0nmTEPh/2KVCHmwzZb3zzz1XFkC6UmHDwO2J8L5O7jhG\r\nFEp741EzZIuNi3z4ehOYgLyTsNqSJpfSaHLLxYRJAoGBAJ3HY7Lh0rBYLU3XFRv/\r\nuoQR08mqxFJ59fUwKdUyeYqvmAEqRxVbLzIA+1Ut2FqcVHOubFc206QO83NlyVEl\r\nI0EDyIQsny/vAWk7qVIMV+L3Pi7x+ImFuEb68IFwF6hBwfnw0Ez1nJkBWJ+iGsFI\r\nrgqQR7P6KsWeKPWR2MjV6GEt\r\n-----END PRIVATE KEY-----\r\n","publicKeyPEM":"-----BEGIN PUBLIC KEY-----\r\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA0+ReqZgdOuNyhRcqZQ44\r\nDI3HJvjM/104IzcLPHWU3aqTp42xl0CTwvWc1w5/H2dVe+0JYUqauOWAKrPyPCnK\r\neectrL1h2bu8ZPfsMVsjvgw5vNUeP7zPQbCTCwHSyLcSs5i5fXdM+8JDhfyWwjeK\r\nK0GhTNp+QZS9WpCzk0HzL94EutMW8781sEKKBf3/8SH164ii+50cQe8U/5cvzQB3\r\nO3uQXre/ReDZqsiWIXoNlDKxkCUWWl8Mcczon+trd0/bwctB5Kol/lmhV2jHwtZm\r\nwTpfpmjQzoGbpCDejkcTJJFc7H/f16NbZfaxgf56RvNlE24BOsh0QHsZy8RLeU2r\r\nQwIDAQAB\r\n-----END PUBLIC KEY-----\r\n","cert":"MIIDlTCCAn2gAwIBAgIUXXnkPlxeR2oekO46MyRmXU1HkLowDQYJKoZIhvcNAQELBQAwJDEiMCAGA1UEAwwZRklET1RFU1QtVFBNLUlOVEVSTUVESUFURTAgFw0yNDA5MTEwNTE4MDRaGA8yMDUyMDEyNzA1MTgwNFowADCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBANPkXqmYHTrjcoUXKmUOOAyNxyb4zP9dOCM3Czx1lN2qk6eNsZdAk8L1nNcOfx9nVXvtCWFKmrjlgCqz8jwpynnnLay9Ydm7vGT37DFbI74MObzVHj+8z0GwkwsB0si3ErOYuX13TPvCQ4X8lsI3iitBoUzafkGUvVqQs5NB8y/eBLrTFvO/NbBCigX9//Eh9euIovudHEHvFP+XL80Adzt7kF63v0Xg2arIliF6DZQysZAlFlpfDHHM6J/ra3dP28HLQeSqJf5ZoVdox8LWZsE6X6Zo0M6Bm6Qg3o5HEySRXOx/39ejW2X2sYH+ekbzZRNuATrIdEB7GcvES3lNq0MCAwEAAaOB4DCB3TAOBgNVHQ8BAf8EBAMCB4AwEAYDVR0lBAkwBwYFZ4EFCAMwDAYDVR0TAQH/BAIwADBIBgNVHREBAf8EPjA8pDowODE2MA0GBWeBBQIDDARpZDoxMA8GBWeBBQICDAZJQk1UUE0wFAYFZ4EFAgEMC2lkOkZGRkZGMUQwMCEGCysGAQQBguUcAQEEBBIEEHr9CdJKfUUJiEFDNIh+YYIwHQYDVR0OBBYEFAg1O3oHeIF65gAnKFR0pc+pXgNAMB8GA1UdIwQYMBaAFL9eH/Hw/Yx8VHwfA+IeVnzc39WpMA0GCSqGSIb3DQEBCwUAA4IBAQAqFdj/JQkzeMsoNyNLvXzCZpc4foeQruYLftyYPGtqSskqIQrls71DiqzoEYc+yZU6E22p/vRM7N9k63TcCtvy89wzmRRlk8TVo0Vb+t4lDpauuD5xNdVv46QgDgvD9D3w6ireVUQz+rtQZDXAq2y2j62B8PLiJAr3YlIy5l5flei4vX38wC2piefoNQ/L54fTNS7txYKwMK8NtGgqjc3XLOgKlv2y+1+DpYE86/xltiyZ+rXqdoduI5iO1+wXBCGSCoFuQldeDUEI3hctAb1/DqxDWGdMVn3wQELg16zwSRY7ScIZwMlfcynHRR8nfO/aPqpBXw20C52ua4z68j7Q","tpmIntercert":"MIICpjCCAkygAwIBAgIUWx0GEtI3iLZn+Dddys1/thhllTIwCgYIKoZIzj0EAwIwLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1QwIBcNMjQwOTExMDUxODA0WhgPMjA1MjAxMjcwNTE4MDRaMCQxIjAgBgNVBAMMGUZJRE9URVNULVRQTS1JTlRFUk1FRElBVEUwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDYvT5hQAs0kIAaKJonYBfu7BUS4CZMHuynuNNpJULv/RqYFB9bGCBsVU2DYwdgNFGUoacnbZxIHyMBaBKMzoF3iq7vGaORYNvKY9OF7Yjw9w7cz/ToFHFWv516fBUre0iWHVIj/bK+Bhygb90wd3DKPx+7YMQgnnw/+C8f1uW7yIUi3wA9idBrGEITgIF2v+L/edTOwVjUli0lqZigUnVSldCsNVjOa+tGhEEeR6ESZ1NYA6jb+KCmibgKQbfCzSDxyLUo75Vu6SnPcrYkG6QnWOuxZRKulwZrWt+e1aCD3Hn67gop2gl/kCJPW1ek4vMdjEA5JSRbZdEhmSXr8FTTAgMBAAGjgYQwgYEwDgYDVR0PAQH/BAQDAgGGMBsGA1UdJQQUMBIGCSsGAQQBgjcVJAYFZ4EFCAMwEgYDVR0TAQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQUv14f8fD9jHxUfB8D4h5WfNzf1akwHwYDVR0jBBgwFoAUdHswx9rjRbIR3mI8xAezIHALhIYwCgYIKoZIzj0EAwIDSAAwRQIgCUUyKn7PKV06xfEJkQiPtUU5ATu9OVkvxrnhXv6k4+gCIQCai+NJufJQQ3Rxnw3PyhTPNS9EHmTkLgFur+nRNVPX3Q=="},"packed-self":{"aaguid":"f6f4ef86-f04e-4fd3-8b01-c2ebd9e56492"}}
This script produces several 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-tpm.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.
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
.
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.
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
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
Performing attestation and assertion for format: packed
2024-08-23T22:05:18.914Z performAttestation sending attestation options to ISV:
{
"userId": "50BCRQVNC0",
"authenticatorSelection": {
"userVerification": "required",
"requireResidentKey": true
}
}
2024-08-23T22:05:19.499Z performAttestation sending attestation result to ISV:
{
"id": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"rawId": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"response": {
"clientDataJSON": "eyJvcmlnaW4iOiJodHRwczovL215aWRwLmljZS5pYm1jbG91ZC5jb20iLCJjaGFsbGVuZ2UiOiJpUDR6d2R4VzE4SGhGa19tek12eTRod0FOZGNGWkpkenJJQzdtR09hU0ZnIiwidHlwZSI6IndlYmF1dGhuLmNyZWF0ZSJ9",
"attestationObject": "o2NmbXRmcGFja2VkZ2F0dFN0bXSjY2FsZyZjeDVjgVkCADCCAfwwggGioAMCAQICFHE1QQNFaQkTVx0zchrx6NrxNfudMAoGCCqGSM49BAMCMC4xCzAJBgNVBAYTAlVTMQwwCgYDVQQKDANJQk0xETAPBgNVBAMMCEZJRE9URVNUMCAXDTI0MDgyMzIxMzQzMFoYDzIwNTIwMTA4MjEzNDMwWjBXMQswCQYDVQQGEwJVUzEMMAoGA1UECgwDSUJNMSIwIAYDVQQLDBlBdXRoZW50aWNhdG9yIEF0dGVzdGF0aW9uMRYwFAYDVQQDDA1QQUNLRUQtU0lHTkVSMFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE0NQFg1CAM-t2xy2gDru87QT0m_yHEgNwGPuMClEBO8XbauyiBfPqsDD_sYZxLrCS8h6551zXDjkPqV581qVOGqNzMHEwDAYDVR0TAQH_BAIwADAhBgsrBgEEAYLlHAEBBAQSBBCcUSv88u5MAqHxzuZyUNiIMB0GA1UdDgQWBBRWOeh1g3W-fEufJx6zyeHtNirEcjAfBgNVHSMEGDAWgBT20J9AA8YJQwKsImfhYigNri0fozAKBggqhkjOPQQDAgNIADBFAiAHYHXtjoFUetRq7XIPigRLQYt-ZEcj0pqDhT8mx2mfGAIhAK5fj3IuycU0Zp0v1581GrjBZSsCGxoIzZ8ocjogjkC6Y3NpZ1hIMEYCIQDpO4Q-NU5oyqwomtEKJU4uV8x7h7-6uNIrapuVGuna-wIhAIQP82SBDIZ7i66bz0tp9J1EUYTZ4UuQshjOyFbFbNfjaGF1dGhEYXRhWOTeeeHNF-cYKzCL-KhAGuxFX5RrGUWa1yrUSWIMt1-bPEUAAAAAnFEr_PLuTAKh8c7mclDYiABgU2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8MpQECAyYgASFYIGju5aPVP83TSlEtqxdhd7NQkYqyVr6z4ByKEPMKihHRIlgg2fWiE2GbeyvAdIFWkZiIGQ6pVF-2l_9kF9eTKI5TySI"
},
"type": "public-key",
"getClientExtensionResults": {},
"nickname": "NodeClient - iP4zwdxW18HhFk_mzMvy4hwANdcFZJdzrIC7mGOaSFg",
"enabled": true,
"getTransports": [
"node"
]
}
2024-08-23T22:05:20.024Z performAttestation got attestationResultResponse:
{
"id": "e87b398b-890c-4ccd-895e-93eddb3ad72a",
"userId": "50BCRQVNC0",
"type": "fido2",
"created": "2024-08-23T22:05:19.890Z",
"updated": "2024-08-23T22:05:19.890Z",
"enabled": true,
"validated": true,
"attributes": {
"attestationType": "Basic",
"attestationFormat": "packed",
"nickname": "NodeClient - iP4zwdxW18HhFk_mzMvy4hwANdcFZJdzrIC7mGOaSFg",
"userVerified": true,
"userPresent": true,
"credentialId": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"credentialPublicKey": "v2EzYi03Yi0xAWItMlggaO7lo9U/zdNKUS2rF2F3s1CRirJWvrPgHIoQ8wqKEdFhMQJhMyZiLTNYINn1ohNhm3srwHSBVpGYiBkOqVRftpf/ZBfXkyiOU8ki/w==",
"rpId": "myidp.ice.ibmcloud.com",
"counter": 0,
"aaGuid": "9C512BFC-F2EE-4C02-A1F1-CEE67250D888",
"icon": "",
"description": "FIDOTEST-PACKED",
"x5c": [
"-----BEGIN CERTIFICATE-----\nMIIB/DCCAaKgAwIBAgIUcTVBA0VpCRNXHTNyGvHo2vE1+50wCgYIKoZIzj0EAwIw\nLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1Qw\nIBcNMjQwODIzMjEzNDMwWhgPMjA1MjAxMDgyMTM0MzBaMFcxCzAJBgNVBAYTAlVT\nMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRp\nb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMB\nBwNCAATQ1AWDUIAz63bHLaAOu7ztBPSb/IcSA3AY+4wKUQE7xdtq7KIF8+qwMP+x\nhnEusJLyHrnnXNcOOQ+pXnzWpU4ao3MwcTAMBgNVHRMBAf8EAjAAMCEGCysGAQQB\nguUcAQEEBBIEEJxRK/zy7kwCofHO5nJQ2IgwHQYDVR0OBBYEFFY56HWDdb58S58n\nHrPJ4e02KsRyMB8GA1UdIwQYMBaAFPbQn0ADxglDAqwiZ+FiKA2uLR+jMAoGCCqG\nSM49BAMCA0gAMEUCIAdgde2OgVR61Grtcg+KBEtBi35kRyPSmoOFPybHaZ8YAiEA\nrl+Pci7JxTRmnS/XnzUauMFlKwIbGgjNnyhyOiCOQLo=\n-----END CERTIFICATE-----"
],
"transports": [],
"backupEligibility": false,
"backupState": false
},
"references": {
"rpUuid": "117b66c2-7e5b-41fa-a0ee-d321f1f5e6ce"
}
}
2024-08-23T22:05:20.025Z performAssertion sending assertion options to ISV:
{
"userVerification": "required",
"userId": "50BCRQVNC0"
}
2024-08-23T22:05:20.464Z performAssertion: assertionOptionsResponse:
{
"rpId": "myidp.ice.ibmcloud.com",
"timeout": 240000,
"challenge": "yvaWhCxRenRPZhpXhbkYTyvC7ODC-lHI0JXAEGTn-xY",
"allowCredentials": [
{
"id": "U2FsdGVkX191Uyvu9WKRM5Q4pdsJJMc1KD9ZAa-m-gmflWxnCB8bq4Taxk3ccx-o_ph02EGfUnKrBJQlNjKDFn4bLdADuNB2Ii8ZLb8RSKuhnOK3NkUynAsiB9zDhSKv",
"type": "public-key"
},
{
"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": "U2FsdGVkX1-4t5sC3cBLbR2d4Sc8eEq_mMvDaDtXLTOVJCf2K0GokiISQlo68y1K-osvhu9jCLMhqJWqjIfq9D1tOnZpPNC2XPCRr33esk0hBB7KEFExrE8b1gt0vDEi",
"type": "public-key"
},
{
"id": "U2FsdGVkX1--7q5BHvUX3-ksdZhOTdl2q1bfLoOB_LSufrTyOYDcaPvvQDlICggP_D26DuGbiHLkuqLeAInNxMRo6eKTZgEhF9_pg0msypzqU0pznvDabRMUJVFvKiXd",
"type": "public-key"
},
{
"id": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"type": "public-key"
},
{
"id": "JVigN6QEf6vy5U6pAFTKAMoaGmQ",
"type": "public-key"
},
{
"id": "dF7MXlvzrDIcfjDoFUsU5g",
"type": "public-key"
},
{
"id": "U2FsdGVkX18k9mw-Gk-kTE8h517fvx98bvl0ZhOiGa4xkFmIufH4FMcr2aEVNvfZgvBuLPgNLXsJJcsIhqeR5pVDE87ONXWuLH1q1dhnqGaLcg_jazzXhv2G5qfyE75t",
"type": "public-key"
}
],
"extensions": {},
"userVerification": "required"
}
2024-08-23T22:05:20.502Z performAssertion sending assertion result to ISV:
{
"id": "U2FsdGVkX191Uyvu9WKRM5Q4pdsJJMc1KD9ZAa-m-gmflWxnCB8bq4Taxk3ccx-o_ph02EGfUnKrBJQlNjKDFn4bLdADuNB2Ii8ZLb8RSKuhnOK3NkUynAsiB9zDhSKv",
"rawId": "U2FsdGVkX191Uyvu9WKRM5Q4pdsJJMc1KD9ZAa-m-gmflWxnCB8bq4Taxk3ccx-o_ph02EGfUnKrBJQlNjKDFn4bLdADuNB2Ii8ZLb8RSKuhnOK3NkUynAsiB9zDhSKv",
"response": {
"clientDataJSON": "eyJvcmlnaW4iOiJodHRwczovL215aWRwLmljZS5pYm1jbG91ZC5jb20iLCJjaGFsbGVuZ2UiOiJ5dmFXaEN4UmVuUlBaaHBYaGJrWVR5dkM3T0RDLWxISTBKWEFFR1RuLXhZIiwidHlwZSI6IndlYmF1dGhuLmdldCJ9",
"authenticatorData": "3nnhzRfnGCswi_ioQBrsRV-UaxlFmtcq1EliDLdfmzwFAAAAAA",
"signature": "MEYCIQDy_haZvxEABvAfWxvA4py6T5i8tupUhSZtOTIKZzz2iQIhAId18T3pyK-5Fz0I8X962FiKtvuU0VDD28hTj_7k_VsT",
"userHandle": ""
},
"type": "public-key",
"getClientExtensionResults": {}
}
2024-08-23T22:05:21.080Z performAssertion got assertionResultResponse:
{
"id": "9b648246-5565-4fc2-aa94-4abd6ce9c2ea",
"userId": "50BCRQVNC0",
"type": "fido2",
"created": "2024-08-23T21:47:03.506Z",
"updated": "2024-08-23T21:47:03.506Z",
"attempted": "2024-08-23T22:05:20.905Z",
"enabled": true,
"validated": true,
"attributes": {
"attestationType": "Basic",
"attestationFormat": "packed",
"nickname": "NodeClient - 6cxDSlCbu03o-eQHrNEp7HzXjxQHj8lVHzh4y0DdiH0",
"userVerified": true,
"userPresent": true,
"credentialId": "U2FsdGVkX191Uyvu9WKRM5Q4pdsJJMc1KD9ZAa-m-gmflWxnCB8bq4Taxk3ccx-o_ph02EGfUnKrBJQlNjKDFn4bLdADuNB2Ii8ZLb8RSKuhnOK3NkUynAsiB9zDhSKv",
"credentialPublicKey": "v2EzYi03Yi0xAWItMlggjDClnq/1N5wqd1nKu6OQFjMV991vkOzQ1KFU8veSlaxhMQJhMyZiLTNYIO0Ry6DyxpB1bmFEwjDPZdS6MaAiROTTPJz/E6T3YAQT/w==",
"rpId": "myidp.ice.ibmcloud.com",
"counter": 0,
"aaGuid": "9C512BFC-F2EE-4C02-A1F1-CEE67250D888",
"icon": "",
"description": "FIDOTEST-PACKED",
"x5c": [
"-----BEGIN CERTIFICATE-----\nMIIB/DCCAaKgAwIBAgIUcTVBA0VpCRNXHTNyGvHo2vE1+50wCgYIKoZIzj0EAwIw\nLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1Qw\nIBcNMjQwODIzMjEzNDMwWhgPMjA1MjAxMDgyMTM0MzBaMFcxCzAJBgNVBAYTAlVT\nMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRp\nb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMB\nBwNCAATQ1AWDUIAz63bHLaAOu7ztBPSb/IcSA3AY+4wKUQE7xdtq7KIF8+qwMP+x\nhnEusJLyHrnnXNcOOQ+pXnzWpU4ao3MwcTAMBgNVHRMBAf8EAjAAMCEGCysGAQQB\nguUcAQEEBBIEEJxRK/zy7kwCofHO5nJQ2IgwHQYDVR0OBBYEFFY56HWDdb58S58n\nHrPJ4e02KsRyMB8GA1UdIwQYMBaAFPbQn0ADxglDAqwiZ+FiKA2uLR+jMAoGCCqG\nSM49BAMCA0gAMEUCIAdgde2OgVR61Grtcg+KBEtBi35kRyPSmoOFPybHaZ8YAiEA\nrl+Pci7JxTRmnS/XnzUauMFlKwIbGgjNnyhyOiCOQLo=\n-----END CERTIFICATE-----"
],
"transports": [],
"backupEligibility": false,
"backupState": false
},
"references": {
"rpUuid": "117b66c2-7e5b-41fa-a0ee-d321f1f5e6ce"
}
}
2024-08-23T22:05:21.082Z assertionResultResponses: [ <REDACTED - SAME content as above> ]
2024-08-23T22:05:21.082Z ==============================================================
2024-08-23T22:05:21.082Z authenticatorRecords:
{
"U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M": {
"rpId": "myidp.ice.ibmcloud.com",
"privateKeyHex": "9cda1332f3c4028363445bc9fa81e883cb9198dff21220731465ba0077cce44c",
"credentialID": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"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:
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.
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-08-23T22:10:14.289Z performAssertion sending assertion options to ISV:
{
"userVerification": "required"
}
2024-08-23T22:10:14.668Z performAssertion: assertionOptionsResponse:
{
"rpId": "myidp.ice.ibmcloud.com",
"timeout": 240000,
"challenge": "2q_xO-GUuSMrjW8jmecT5MAP1X83QguYzWm6AESXnas",
"userVerification": "required"
}
2024-08-23T22:10:14.721Z performAssertion sending assertion result to ISV:
{
"id": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"rawId": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"response": {
"clientDataJSON": "eyJvcmlnaW4iOiJodHRwczovL215aWRwLmljZS5pYm1jbG91ZC5jb20iLCJjaGFsbGVuZ2UiOiIycV94Ty1HVXVTTXJqVzhqbWVjVDVNQVAxWDgzUWd1WXpXbTZBRVNYbmFzIiwidHlwZSI6IndlYmF1dGhuLmdldCJ9",
"authenticatorData": "3nnhzRfnGCswi_ioQBrsRV-UaxlFmtcq1EliDLdfmzwFAAAAAA",
"signature": "MEUCIQCQMcjLZN6Ur_qH50xm44ecZRZyeu6Pe1RL4PDqQ4kIHwIgTb_q7D-995Nh7JlbPTB__lP_kpO4EGieOR8bDZz3LNY",
"userHandle": "NTBCQ1JRVk5DMA"
},
"type": "public-key",
"getClientExtensionResults": {}
}
2024-08-23T22:10:15.179Z performAssertion got assertionResultResponse:
{
"id": "e87b398b-890c-4ccd-895e-93eddb3ad72a",
"userId": "50BCRQVNC0",
"type": "fido2",
"created": "2024-08-23T22:05:19.890Z",
"updated": "2024-08-23T22:05:19.890Z",
"attempted": "2024-08-23T22:10:15.115Z",
"enabled": true,
"validated": true,
"attributes": {
"attestationType": "Basic",
"attestationFormat": "packed",
"nickname": "NodeClient - iP4zwdxW18HhFk_mzMvy4hwANdcFZJdzrIC7mGOaSFg",
"userVerified": true,
"userPresent": true,
"credentialId": "U2FsdGVkX19bg8bmysGHt3vpWdfg1DGG0zeQnaSpI38AmoxgjSFjB_Wr2KVcWPmzJCZjj7yGw5FfObyR3V43fGU1TYkwoK7bnS2GJ-h0PDFmS4pFH3bI66RYl1oesq8M",
"credentialPublicKey": "v2EzYi03Yi0xAWItMlggaO7lo9U/zdNKUS2rF2F3s1CRirJWvrPgHIoQ8wqKEdFhMQJhMyZiLTNYINn1ohNhm3srwHSBVpGYiBkOqVRftpf/ZBfXkyiOU8ki/w==",
"rpId": "myidp.ice.ibmcloud.com",
"counter": 0,
"aaGuid": "9C512BFC-F2EE-4C02-A1F1-CEE67250D888",
"icon": "",
"description": "FIDOTEST-PACKED",
"x5c": [
"-----BEGIN CERTIFICATE-----\nMIIB/DCCAaKgAwIBAgIUcTVBA0VpCRNXHTNyGvHo2vE1+50wCgYIKoZIzj0EAwIw\nLjELMAkGA1UEBhMCVVMxDDAKBgNVBAoMA0lCTTERMA8GA1UEAwwIRklET1RFU1Qw\nIBcNMjQwODIzMjEzNDMwWhgPMjA1MjAxMDgyMTM0MzBaMFcxCzAJBgNVBAYTAlVT\nMQwwCgYDVQQKDANJQk0xIjAgBgNVBAsMGUF1dGhlbnRpY2F0b3IgQXR0ZXN0YXRp\nb24xFjAUBgNVBAMMDVBBQ0tFRC1TSUdORVIwWTATBgcqhkjOPQIBBggqhkjOPQMB\nBwNCAATQ1AWDUIAz63bHLaAOu7ztBPSb/IcSA3AY+4wKUQE7xdtq7KIF8+qwMP+x\nhnEusJLyHrnnXNcOOQ+pXnzWpU4ao3MwcTAMBgNVHRMBAf8EAjAAMCEGCysGAQQB\nguUcAQEEBBIEEJxRK/zy7kwCofHO5nJQ2IgwHQYDVR0OBBYEFFY56HWDdb58S58n\nHrPJ4e02KsRyMB8GA1UdIwQYMBaAFPbQn0ADxglDAqwiZ+FiKA2uLR+jMAoGCCqG\nSM49BAMCA0gAMEUCIAdgde2OgVR61Grtcg+KBEtBi35kRyPSmoOFPybHaZ8YAiEA\nrl+Pci7JxTRmnS/XnzUauMFlKwIbGgjNnyhyOiCOQLo=\n-----END CERTIFICATE-----"
],
"transports": [],
"backupEligibility": false,
"backupState": false
},
"references": {
"rpUuid": "117b66c2-7e5b-41fa-a0ee-d321f1f5e6ce"
}
}
2024-08-23T22:10:15.179Z assertionResultResponse:
<REDACTED - SAME content as above>