const PAYMENT_DETAILS = { total: {label: 'Total', amount: {value: '0.01', currency: 'USD'}} }; const AUTHENTICATOR_OPTS = { protocol: 'ctap2_1', transport: 'internal', hasResidentKey: true, hasUserVerification: true, isUserVerified: true, }; const ICON_URL = 'https://{{hosts[][www]}}:{{ports[https][0]}}/secure-payment-confirmation/troy.png'; const NONEXISTENT_ICON_URL = 'https://{{hosts[][www]}}:{{ports[https][0]}}/secure-payment-confirmation/nonexistent.png'; const ICON_DATA_URL = ''; const INVALID_ICON_DATA_URL = ''; // Creates and returns a WebAuthn credential, optionally with the payment // extension set. // // Assumes that a virtual authenticator has already been created. async function createCredential(set_payment_extension=true) { const challengeBytes = new Uint8Array(16); window.crypto.getRandomValues(challengeBytes); const publicKey = { challenge: challengeBytes, rp: { name: 'Acme', }, user: { id: new Uint8Array(16), name: 'jane.doe@example.com', displayName: 'Jane Doe', }, pubKeyCredParams: [{ type: 'public-key', alg: -7, // 'ES256' }], authenticatorSelection: { userVerification: 'required', residentKey: 'required', authenticatorAttachment: 'platform', }, timeout: 30000, }; if (set_payment_extension) { publicKey.extensions = { payment: { isPayment: true }, }; } return navigator.credentials.create({publicKey}); } // Creates a SPC credential in an iframe for the WPT 'alt' domain. Returns a // promise that resolves with the created credential id. // // Assumes that a virtual authenticator has already been created. async function createCredentialForAltDomain() { const frame = document.createElement('iframe'); frame.allow = 'payment'; frame.src = 'https://{{hosts[alt][]}}:{{ports[https][0]}}' + '/secure-payment-confirmation/resources/iframe-enroll.html'; // Wait for the iframe to load. const readyPromise = new Promise(resolve => { window.addEventListener('message', function handler(evt) { if (evt.source === frame.contentWindow && evt.data.type == 'loaded') { window.removeEventListener('message', handler); resolve(evt.data); } }); }); document.body.appendChild(frame); await readyPromise; // Setup the result promise, and then trigger credential creation. const resultPromise = new Promise(resolve => { window.addEventListener('message', function handler(evt) { if (evt.source === frame.contentWindow && evt.data.type == 'spc_result') { document.body.removeChild(frame); window.removeEventListener('message', handler); resolve(evt.data); } }); }); frame.contentWindow.postMessage({ userActivation: true }, '*'); return resultPromise; } function arrayBufferToString(buffer) { return String.fromCharCode(...new Uint8Array(buffer)); } function base64UrlEncode(data) { let result = btoa(data); return result.replace(/=+$/g, '').replace(/\+/g, "-").replace(/\//g, "_"); }