| 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 = ''; |
| |
| const PAYMENT_ENTITY_LOGO_URL = 'https://{{hosts[][www]}}:{{ports[https][0]}}/secure-payment-confirmation/sync-network-logo.png'; |
| const NONEXISTENT_PAYMENT_ENTITY_LOGO_URL = 'https://{{hosts[][www]}}:{{ports[https][0]}}/secure-payment-confirmation/nonexistent.png'; |
| |
| // Creates and returns a WebAuthn credential, optionally with the payment |
| // extension set. |
| // |
| // `options` is object containing additional options for creating the request. |
| // The options include the following: |
| // |
| // * "browserBoundPubKeyCredParams": The credential creation parameters to set |
| // on the payment extension. set_payment_extension must be set to true in |
| // order for this option to apply. |
| // |
| // Assumes that a virtual authenticator has already been created. |
| async function createCredential(set_payment_extension = true, options = {}) { |
| options = Object.assign({browserBoundPubKeyCredParams: []}, options); |
| 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, |
| browserBoundPubKeyCredParams: options.browserBoundPubKeyCredParams |
| }, |
| }; |
| } |
| |
| 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, "_"); |
| } |