1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
|
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, "_");
}
|