@thru/passkey 0.2.12 → 0.2.14
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +73 -90
- package/dist/auth.cjs +672 -0
- package/dist/auth.cjs.map +1 -0
- package/dist/auth.d.cts +60 -0
- package/dist/auth.d.ts +60 -0
- package/dist/auth.js +422 -0
- package/dist/auth.js.map +1 -0
- package/dist/chunk-2JHC7OOH.js +250 -0
- package/dist/chunk-2JHC7OOH.js.map +1 -0
- package/dist/chunk-75G2FPYW.js +54 -0
- package/dist/chunk-75G2FPYW.js.map +1 -0
- package/dist/chunk-B5SN7AS7.js +586 -0
- package/dist/chunk-B5SN7AS7.js.map +1 -0
- package/dist/chunk-LNDWK3FA.js +163 -0
- package/dist/chunk-LNDWK3FA.js.map +1 -0
- package/dist/index.cjs +27 -94
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +4 -187
- package/dist/index.d.ts +4 -187
- package/dist/index.js +47 -810
- package/dist/index.js.map +1 -1
- package/dist/mobile.cjs +301 -0
- package/dist/mobile.cjs.map +1 -0
- package/dist/mobile.d.cts +49 -0
- package/dist/mobile.d.ts +49 -0
- package/dist/mobile.js +41 -0
- package/dist/mobile.js.map +1 -0
- package/dist/popup.cjs +247 -0
- package/dist/popup.cjs.map +1 -0
- package/dist/popup.d.cts +22 -0
- package/dist/popup.d.ts +22 -0
- package/dist/popup.js +31 -0
- package/dist/popup.js.map +1 -0
- package/dist/server.cjs +351 -0
- package/dist/server.cjs.map +1 -0
- package/dist/server.d.cts +119 -0
- package/dist/server.d.ts +119 -0
- package/dist/server.js +340 -0
- package/dist/server.js.map +1 -0
- package/dist/types-_HRzmn-j.d.cts +125 -0
- package/dist/types-_HRzmn-j.d.ts +125 -0
- package/dist/web.cjs +758 -0
- package/dist/web.cjs.map +1 -0
- package/dist/web.d.cts +32 -0
- package/dist/web.d.ts +32 -0
- package/dist/web.js +60 -0
- package/dist/web.js.map +1 -0
- package/package.json +47 -2
- package/src/auth/execute-tx.ts +87 -0
- package/src/auth/index.ts +18 -0
- package/src/auth/types.ts +56 -0
- package/src/auth/use-passkey-auth.ts +428 -0
- package/src/index.ts +37 -39
- package/src/mobile/errors.ts +31 -0
- package/src/mobile/index.ts +33 -0
- package/src/mobile/passkey.ts +154 -0
- package/src/mobile/storage.ts +115 -0
- package/src/mobile/types.ts +24 -0
- package/src/popup-entry.ts +33 -0
- package/src/popup-service.ts +0 -103
- package/src/server/challenge.ts +26 -0
- package/src/server/create-wallet.ts +149 -0
- package/src/server/handlers.ts +93 -0
- package/src/server/index.ts +13 -0
- package/src/server/submit.ts +47 -0
- package/src/server/types.ts +70 -0
- package/src/server/utils.ts +69 -0
- package/src/types.ts +1 -0
- package/src/web.ts +51 -0
- package/tsconfig.json +6 -1
- package/tsup.config.ts +9 -1
|
@@ -0,0 +1,586 @@
|
|
|
1
|
+
import {
|
|
2
|
+
closePopup,
|
|
3
|
+
openPasskeyPopupWindow,
|
|
4
|
+
requestPasskeyPopup
|
|
5
|
+
} from "./chunk-LNDWK3FA.js";
|
|
6
|
+
|
|
7
|
+
// src/register.ts
|
|
8
|
+
import { arrayBufferToBase64Url, bytesToHex } from "@thru/passkey-manager";
|
|
9
|
+
|
|
10
|
+
// src/capabilities.ts
|
|
11
|
+
var DEBUG = typeof process !== "undefined" && process.env?.NEXT_PUBLIC_PASSKEY_DEBUG === "1";
|
|
12
|
+
var cachedClientCapabilities;
|
|
13
|
+
var clientCapabilitiesPromise = null;
|
|
14
|
+
function isWebAuthnSupported() {
|
|
15
|
+
const supported = typeof window !== "undefined" && typeof window.PublicKeyCredential !== "undefined" && typeof navigator.credentials !== "undefined";
|
|
16
|
+
if (DEBUG) {
|
|
17
|
+
console.log("[Passkey] WebAuthn support check:", {
|
|
18
|
+
window: typeof window !== "undefined",
|
|
19
|
+
PublicKeyCredential: typeof window !== "undefined" && typeof window.PublicKeyCredential !== "undefined",
|
|
20
|
+
credentials: typeof window !== "undefined" && typeof navigator !== "undefined" && typeof navigator.credentials !== "undefined",
|
|
21
|
+
supported
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
return supported;
|
|
25
|
+
}
|
|
26
|
+
async function fetchPasskeyClientCapabilities() {
|
|
27
|
+
if (typeof window === "undefined" || typeof window.PublicKeyCredential === "undefined") {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
const getClientCapabilities = window.PublicKeyCredential.getClientCapabilities;
|
|
31
|
+
if (typeof getClientCapabilities !== "function") {
|
|
32
|
+
return null;
|
|
33
|
+
}
|
|
34
|
+
try {
|
|
35
|
+
const capabilities = await getClientCapabilities.call(window.PublicKeyCredential);
|
|
36
|
+
if (DEBUG) {
|
|
37
|
+
console.log("[Passkey] WebAuthn client capabilities:", capabilities);
|
|
38
|
+
}
|
|
39
|
+
return capabilities ?? null;
|
|
40
|
+
} catch (error) {
|
|
41
|
+
if (DEBUG) {
|
|
42
|
+
console.warn("[Passkey] Failed to read client capabilities:", error);
|
|
43
|
+
}
|
|
44
|
+
return null;
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
function preloadPasskeyClientCapabilities() {
|
|
48
|
+
if (cachedClientCapabilities !== void 0 || clientCapabilitiesPromise) {
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
clientCapabilitiesPromise = fetchPasskeyClientCapabilities().then((capabilities) => {
|
|
52
|
+
cachedClientCapabilities = capabilities;
|
|
53
|
+
return capabilities;
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
async function getPasskeyClientCapabilities() {
|
|
57
|
+
if (cachedClientCapabilities !== void 0) {
|
|
58
|
+
return cachedClientCapabilities;
|
|
59
|
+
}
|
|
60
|
+
if (!clientCapabilitiesPromise) {
|
|
61
|
+
preloadPasskeyClientCapabilities();
|
|
62
|
+
}
|
|
63
|
+
if (!clientCapabilitiesPromise) {
|
|
64
|
+
cachedClientCapabilities = null;
|
|
65
|
+
return null;
|
|
66
|
+
}
|
|
67
|
+
const capabilities = await clientCapabilitiesPromise;
|
|
68
|
+
cachedClientCapabilities = capabilities;
|
|
69
|
+
return capabilities;
|
|
70
|
+
}
|
|
71
|
+
function getCachedPasskeyClientCapabilities() {
|
|
72
|
+
return cachedClientCapabilities;
|
|
73
|
+
}
|
|
74
|
+
function isInIframe() {
|
|
75
|
+
if (typeof window === "undefined") {
|
|
76
|
+
return false;
|
|
77
|
+
}
|
|
78
|
+
try {
|
|
79
|
+
return window.self !== window.top;
|
|
80
|
+
} catch {
|
|
81
|
+
return true;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
async function shouldUsePasskeyPopup(action) {
|
|
85
|
+
if (!isInIframe()) {
|
|
86
|
+
return false;
|
|
87
|
+
}
|
|
88
|
+
const mode = await getPasskeyPromptMode(action);
|
|
89
|
+
return mode === "popup";
|
|
90
|
+
}
|
|
91
|
+
function getPermissionsPolicyAllowsFeature(feature) {
|
|
92
|
+
if (typeof document === "undefined") {
|
|
93
|
+
return null;
|
|
94
|
+
}
|
|
95
|
+
const policy = document.permissionsPolicy;
|
|
96
|
+
const featurePolicy = document.featurePolicy;
|
|
97
|
+
const allowsFeature = policy?.allowsFeature || featurePolicy?.allowsFeature;
|
|
98
|
+
if (typeof allowsFeature !== "function") {
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
try {
|
|
102
|
+
return allowsFeature(feature);
|
|
103
|
+
} catch {
|
|
104
|
+
return null;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
function getCachedPromptMode(action) {
|
|
108
|
+
if (!isInIframe()) {
|
|
109
|
+
return "inline";
|
|
110
|
+
}
|
|
111
|
+
if (cachedClientCapabilities === void 0 && !clientCapabilitiesPromise) {
|
|
112
|
+
preloadPasskeyClientCapabilities();
|
|
113
|
+
}
|
|
114
|
+
const feature = action === "create" ? "publickey-credentials-create" : "publickey-credentials-get";
|
|
115
|
+
const policyAllows = getPermissionsPolicyAllowsFeature(feature);
|
|
116
|
+
const capabilities = getCachedPasskeyClientCapabilities();
|
|
117
|
+
const supportsInline = capabilities?.passkeyPlatformAuthenticator === true || capabilities?.userVerifyingPlatformAuthenticator === true;
|
|
118
|
+
if (policyAllows === false) {
|
|
119
|
+
return "popup";
|
|
120
|
+
}
|
|
121
|
+
if (capabilities === void 0) {
|
|
122
|
+
return "unknown";
|
|
123
|
+
}
|
|
124
|
+
if (!supportsInline) {
|
|
125
|
+
return "popup";
|
|
126
|
+
}
|
|
127
|
+
return "inline";
|
|
128
|
+
}
|
|
129
|
+
async function getPasskeyPromptMode(action) {
|
|
130
|
+
if (!isInIframe()) {
|
|
131
|
+
return "inline";
|
|
132
|
+
}
|
|
133
|
+
const feature = action === "create" ? "publickey-credentials-create" : "publickey-credentials-get";
|
|
134
|
+
const policyAllows = getPermissionsPolicyAllowsFeature(feature);
|
|
135
|
+
const capabilities = await getPasskeyClientCapabilities();
|
|
136
|
+
const supportsInline = capabilities?.passkeyPlatformAuthenticator === true || capabilities?.userVerifyingPlatformAuthenticator === true;
|
|
137
|
+
if (DEBUG) {
|
|
138
|
+
console.log("[Passkey] Prompt mode check:", {
|
|
139
|
+
action,
|
|
140
|
+
policyAllows,
|
|
141
|
+
supportsInline,
|
|
142
|
+
capabilities
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
if (!supportsInline) {
|
|
146
|
+
return "popup";
|
|
147
|
+
}
|
|
148
|
+
if (policyAllows === false) {
|
|
149
|
+
return "popup";
|
|
150
|
+
}
|
|
151
|
+
return "inline";
|
|
152
|
+
}
|
|
153
|
+
function maybePreopenPopup(action, openPopupFn) {
|
|
154
|
+
const cachedMode = getCachedPromptMode(action);
|
|
155
|
+
if (cachedMode !== "popup") {
|
|
156
|
+
return null;
|
|
157
|
+
}
|
|
158
|
+
try {
|
|
159
|
+
return openPopupFn();
|
|
160
|
+
} catch {
|
|
161
|
+
return null;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
function shouldFallbackToPopup(error) {
|
|
165
|
+
if (!isInIframe()) {
|
|
166
|
+
return false;
|
|
167
|
+
}
|
|
168
|
+
const name = error && typeof error === "object" && "name" in error ? String(error.name) : "";
|
|
169
|
+
const message = error && typeof error === "object" && "message" in error ? String(error.message) : "";
|
|
170
|
+
const normalized = `${name} ${message}`.toLowerCase();
|
|
171
|
+
if (normalized.includes("cancel") || normalized.includes("canceled") || normalized.includes("cancelled") || normalized.includes("user canceled") || normalized.includes("user cancelled") || normalized.includes("aborted")) {
|
|
172
|
+
return false;
|
|
173
|
+
}
|
|
174
|
+
if (normalized.includes("securityerror")) {
|
|
175
|
+
return true;
|
|
176
|
+
}
|
|
177
|
+
if (normalized.includes("notallowederror")) {
|
|
178
|
+
if (normalized.includes("permission") || normalized.includes("policy") || normalized.includes("iframe") || normalized.includes("frame")) {
|
|
179
|
+
return true;
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
return false;
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// src/register.ts
|
|
186
|
+
async function registerPasskey(alias, userId, rpId) {
|
|
187
|
+
if (!isWebAuthnSupported()) {
|
|
188
|
+
throw new Error("WebAuthn is not supported in this browser");
|
|
189
|
+
}
|
|
190
|
+
return runWithPromptMode(
|
|
191
|
+
"create",
|
|
192
|
+
() => registerPasskeyInline(alias, userId, rpId),
|
|
193
|
+
(preopenedPopup) => registerPasskeyViaPopup(alias, userId, rpId, preopenedPopup)
|
|
194
|
+
);
|
|
195
|
+
}
|
|
196
|
+
async function runWithPromptMode(action, inlineFn, popupFn) {
|
|
197
|
+
const preopenedPopup = maybePreopenPopup(action, openPasskeyPopupWindow);
|
|
198
|
+
const promptMode = await getPasskeyPromptMode(action);
|
|
199
|
+
if (promptMode === "popup") {
|
|
200
|
+
return popupFn(preopenedPopup);
|
|
201
|
+
}
|
|
202
|
+
closePopup(preopenedPopup);
|
|
203
|
+
try {
|
|
204
|
+
return await inlineFn();
|
|
205
|
+
} catch (error) {
|
|
206
|
+
if (shouldFallbackToPopup(error)) {
|
|
207
|
+
return popupFn();
|
|
208
|
+
}
|
|
209
|
+
throw error;
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
async function registerPasskeyInline(alias, userId, rpId) {
|
|
213
|
+
const rpName = "Thru Wallet";
|
|
214
|
+
const userIdBytes = new TextEncoder().encode(userId);
|
|
215
|
+
const userIdBuffer = userIdBytes.slice(0, 64);
|
|
216
|
+
const challenge = crypto.getRandomValues(new Uint8Array(32));
|
|
217
|
+
const createOptions = {
|
|
218
|
+
challenge,
|
|
219
|
+
rp: {
|
|
220
|
+
id: rpId,
|
|
221
|
+
name: rpName
|
|
222
|
+
},
|
|
223
|
+
user: {
|
|
224
|
+
id: userIdBuffer,
|
|
225
|
+
name: alias,
|
|
226
|
+
displayName: alias
|
|
227
|
+
},
|
|
228
|
+
pubKeyCredParams: [
|
|
229
|
+
{ type: "public-key", alg: -7 }
|
|
230
|
+
],
|
|
231
|
+
authenticatorSelection: {
|
|
232
|
+
authenticatorAttachment: "platform",
|
|
233
|
+
userVerification: "required",
|
|
234
|
+
residentKey: "required",
|
|
235
|
+
requireResidentKey: true
|
|
236
|
+
},
|
|
237
|
+
attestation: "none",
|
|
238
|
+
timeout: 6e4
|
|
239
|
+
};
|
|
240
|
+
const credential = await navigator.credentials.create({
|
|
241
|
+
publicKey: createOptions
|
|
242
|
+
});
|
|
243
|
+
if (!credential) {
|
|
244
|
+
throw new Error("Passkey registration was cancelled");
|
|
245
|
+
}
|
|
246
|
+
const response = credential.response;
|
|
247
|
+
const { x, y } = extractP256PublicKey(response);
|
|
248
|
+
return {
|
|
249
|
+
credentialId: arrayBufferToBase64Url(credential.rawId),
|
|
250
|
+
publicKeyX: bytesToHex(x),
|
|
251
|
+
publicKeyY: bytesToHex(y),
|
|
252
|
+
rpId
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
async function registerPasskeyViaPopup(alias, userId, rpId, preopenedPopup) {
|
|
256
|
+
const result = await requestPasskeyPopup(
|
|
257
|
+
"create",
|
|
258
|
+
{ alias, userId, rpId },
|
|
259
|
+
preopenedPopup
|
|
260
|
+
);
|
|
261
|
+
return result;
|
|
262
|
+
}
|
|
263
|
+
function extractP256PublicKey(response) {
|
|
264
|
+
if (typeof response.getPublicKey === "function") {
|
|
265
|
+
const spkiKey = response.getPublicKey();
|
|
266
|
+
if (spkiKey) {
|
|
267
|
+
return extractFromSpki(new Uint8Array(spkiKey));
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
if (typeof response.getAuthenticatorData === "function") {
|
|
271
|
+
const authData = new Uint8Array(response.getAuthenticatorData());
|
|
272
|
+
return extractFromAuthenticatorData(authData);
|
|
273
|
+
}
|
|
274
|
+
throw new Error("Unable to extract public key: browser does not support required WebAuthn methods");
|
|
275
|
+
}
|
|
276
|
+
function extractFromSpki(spki) {
|
|
277
|
+
const pointStart = spki.length - 65;
|
|
278
|
+
if (spki[pointStart] !== 4) {
|
|
279
|
+
throw new Error("Invalid SPKI format: expected uncompressed point");
|
|
280
|
+
}
|
|
281
|
+
const x = spki.slice(pointStart + 1, pointStart + 33);
|
|
282
|
+
const y = spki.slice(pointStart + 33, pointStart + 65);
|
|
283
|
+
if (x.length !== 32 || y.length !== 32) {
|
|
284
|
+
throw new Error("Invalid SPKI format: incorrect coordinate length");
|
|
285
|
+
}
|
|
286
|
+
return { x, y };
|
|
287
|
+
}
|
|
288
|
+
function extractFromAuthenticatorData(authData) {
|
|
289
|
+
const rpIdHashLength = 32;
|
|
290
|
+
const flagsLength = 1;
|
|
291
|
+
const counterLength = 4;
|
|
292
|
+
const offset = rpIdHashLength + flagsLength + counterLength;
|
|
293
|
+
const aaguidLength = 16;
|
|
294
|
+
const credIdLenOffset = offset + aaguidLength;
|
|
295
|
+
const credIdLength = authData[credIdLenOffset] << 8 | authData[credIdLenOffset + 1];
|
|
296
|
+
const coseKeyOffset = credIdLenOffset + 2 + credIdLength;
|
|
297
|
+
const coseKey = authData.slice(coseKeyOffset);
|
|
298
|
+
return extractFromCoseKey(coseKey);
|
|
299
|
+
}
|
|
300
|
+
function extractFromCoseKey(coseKey) {
|
|
301
|
+
const mapStart = coseKey[0];
|
|
302
|
+
if (mapStart !== 165 && mapStart !== 164) {
|
|
303
|
+
throw new Error("Invalid COSE key format");
|
|
304
|
+
}
|
|
305
|
+
let offset = 1;
|
|
306
|
+
let x = null;
|
|
307
|
+
let y = null;
|
|
308
|
+
while (offset < coseKey.length) {
|
|
309
|
+
const key = coseKey[offset++];
|
|
310
|
+
const valueType = coseKey[offset++];
|
|
311
|
+
if (key === 33) {
|
|
312
|
+
const length = valueType & 31;
|
|
313
|
+
x = coseKey.slice(offset, offset + length);
|
|
314
|
+
offset += length;
|
|
315
|
+
continue;
|
|
316
|
+
}
|
|
317
|
+
if (key === 34) {
|
|
318
|
+
const length = valueType & 31;
|
|
319
|
+
y = coseKey.slice(offset, offset + length);
|
|
320
|
+
offset += length;
|
|
321
|
+
continue;
|
|
322
|
+
}
|
|
323
|
+
if (valueType >= 64 && valueType <= 95) {
|
|
324
|
+
const length = valueType & 31;
|
|
325
|
+
offset += length;
|
|
326
|
+
continue;
|
|
327
|
+
}
|
|
328
|
+
if (valueType === 1 || valueType === 2 || valueType === 3) {
|
|
329
|
+
continue;
|
|
330
|
+
}
|
|
331
|
+
if (valueType >= 24 && valueType <= 27) {
|
|
332
|
+
const size = 1 << valueType - 24;
|
|
333
|
+
offset += size;
|
|
334
|
+
continue;
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
if (!x || !y) {
|
|
338
|
+
throw new Error("Failed to extract P-256 public key from COSE data");
|
|
339
|
+
}
|
|
340
|
+
if (x.length !== 32 || y.length !== 32) {
|
|
341
|
+
throw new Error("Invalid COSE key: incorrect coordinate length");
|
|
342
|
+
}
|
|
343
|
+
return { x, y };
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
// src/sign.ts
|
|
347
|
+
import {
|
|
348
|
+
arrayBufferToBase64Url as arrayBufferToBase64Url2,
|
|
349
|
+
base64UrlToArrayBuffer,
|
|
350
|
+
bytesToBase64Url,
|
|
351
|
+
base64UrlToBytes,
|
|
352
|
+
parseDerSignature,
|
|
353
|
+
normalizeLowS
|
|
354
|
+
} from "@thru/passkey-manager";
|
|
355
|
+
async function signWithPasskey(credentialId, challenge, rpId) {
|
|
356
|
+
if (!isWebAuthnSupported()) {
|
|
357
|
+
throw new Error("WebAuthn is not supported in this browser");
|
|
358
|
+
}
|
|
359
|
+
return runWithPromptMode2(
|
|
360
|
+
"get",
|
|
361
|
+
() => signWithPasskeyInline(credentialId, challenge, rpId),
|
|
362
|
+
(preopenedPopup) => signWithPasskeyViaPopup(credentialId, challenge, rpId, preopenedPopup)
|
|
363
|
+
);
|
|
364
|
+
}
|
|
365
|
+
async function signWithStoredPasskey(challenge, rpId, preferredPasskey, allPasskeys, context) {
|
|
366
|
+
if (!isWebAuthnSupported()) {
|
|
367
|
+
throw new Error("WebAuthn is not supported in this browser");
|
|
368
|
+
}
|
|
369
|
+
const preopenedPopup = maybePreopenPopup("get", openPasskeyPopupWindow);
|
|
370
|
+
const promptMode = await getPasskeyPromptMode("get");
|
|
371
|
+
const storedPasskey = preferredPasskey;
|
|
372
|
+
const canUsePopup = isInIframe();
|
|
373
|
+
if (promptMode === "popup" || canUsePopup && !storedPasskey) {
|
|
374
|
+
return requestStoredPasskeyPopup(challenge, preopenedPopup, context);
|
|
375
|
+
}
|
|
376
|
+
closePopup(preopenedPopup);
|
|
377
|
+
try {
|
|
378
|
+
if (storedPasskey) {
|
|
379
|
+
const result = await signWithPasskeyInline(
|
|
380
|
+
storedPasskey.credentialId,
|
|
381
|
+
challenge,
|
|
382
|
+
storedPasskey.rpId
|
|
383
|
+
);
|
|
384
|
+
return {
|
|
385
|
+
...result,
|
|
386
|
+
passkey: storedPasskey
|
|
387
|
+
};
|
|
388
|
+
}
|
|
389
|
+
const discoverable = await signWithDiscoverablePasskey(challenge, rpId);
|
|
390
|
+
const matchingPasskey = allPasskeys.find((p) => p.credentialId === discoverable.credentialId) ?? null;
|
|
391
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
392
|
+
const passkey = matchingPasskey ?? {
|
|
393
|
+
credentialId: discoverable.credentialId,
|
|
394
|
+
publicKeyX: "",
|
|
395
|
+
publicKeyY: "",
|
|
396
|
+
rpId: discoverable.rpId,
|
|
397
|
+
createdAt: now,
|
|
398
|
+
lastUsedAt: now
|
|
399
|
+
};
|
|
400
|
+
return {
|
|
401
|
+
signature: discoverable.signature,
|
|
402
|
+
authenticatorData: discoverable.authenticatorData,
|
|
403
|
+
clientDataJSON: discoverable.clientDataJSON,
|
|
404
|
+
signatureR: discoverable.signatureR,
|
|
405
|
+
signatureS: discoverable.signatureS,
|
|
406
|
+
passkey
|
|
407
|
+
};
|
|
408
|
+
} catch (error) {
|
|
409
|
+
if (canUsePopup && shouldFallbackToPopup(error)) {
|
|
410
|
+
return requestStoredPasskeyPopup(challenge, void 0, context);
|
|
411
|
+
}
|
|
412
|
+
throw error;
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
async function signWithDiscoverablePasskey(challenge, rpId) {
|
|
416
|
+
if (!isWebAuthnSupported()) {
|
|
417
|
+
throw new Error("WebAuthn is not supported in this browser");
|
|
418
|
+
}
|
|
419
|
+
const resolvedRpId = rpId;
|
|
420
|
+
const result = await signWithPasskeyAssertion(challenge, resolvedRpId);
|
|
421
|
+
return {
|
|
422
|
+
signature: result.signature,
|
|
423
|
+
authenticatorData: result.authenticatorData,
|
|
424
|
+
clientDataJSON: result.clientDataJSON,
|
|
425
|
+
signatureR: result.signatureR,
|
|
426
|
+
signatureS: result.signatureS,
|
|
427
|
+
credentialId: result.credentialId,
|
|
428
|
+
rpId: resolvedRpId
|
|
429
|
+
};
|
|
430
|
+
}
|
|
431
|
+
async function runWithPromptMode2(action, inlineFn, popupFn) {
|
|
432
|
+
const preopenedPopup = maybePreopenPopup(action, openPasskeyPopupWindow);
|
|
433
|
+
const promptMode = await getPasskeyPromptMode(action);
|
|
434
|
+
if (promptMode === "popup") {
|
|
435
|
+
return popupFn(preopenedPopup);
|
|
436
|
+
}
|
|
437
|
+
closePopup(preopenedPopup);
|
|
438
|
+
try {
|
|
439
|
+
return await inlineFn();
|
|
440
|
+
} catch (error) {
|
|
441
|
+
if (shouldFallbackToPopup(error)) {
|
|
442
|
+
return popupFn();
|
|
443
|
+
}
|
|
444
|
+
throw error;
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
async function signWithPasskeyInline(credentialId, challenge, rpId) {
|
|
448
|
+
const result = await signWithPasskeyAssertion(challenge, rpId, credentialId);
|
|
449
|
+
return {
|
|
450
|
+
signature: result.signature,
|
|
451
|
+
authenticatorData: result.authenticatorData,
|
|
452
|
+
clientDataJSON: result.clientDataJSON,
|
|
453
|
+
signatureR: result.signatureR,
|
|
454
|
+
signatureS: result.signatureS
|
|
455
|
+
};
|
|
456
|
+
}
|
|
457
|
+
async function signWithPasskeyAssertion(challenge, rpId, credentialId) {
|
|
458
|
+
const challengeBytes = new Uint8Array(challenge);
|
|
459
|
+
const getOptions = {
|
|
460
|
+
challenge: challengeBytes,
|
|
461
|
+
rpId,
|
|
462
|
+
userVerification: "required",
|
|
463
|
+
timeout: 6e4
|
|
464
|
+
};
|
|
465
|
+
if (credentialId) {
|
|
466
|
+
const credentialIdBuffer = base64UrlToArrayBuffer(credentialId);
|
|
467
|
+
getOptions.allowCredentials = [
|
|
468
|
+
{
|
|
469
|
+
type: "public-key",
|
|
470
|
+
id: credentialIdBuffer,
|
|
471
|
+
transports: ["internal", "hybrid", "usb", "ble", "nfc"]
|
|
472
|
+
}
|
|
473
|
+
];
|
|
474
|
+
}
|
|
475
|
+
const assertion = await navigator.credentials.get({
|
|
476
|
+
publicKey: getOptions
|
|
477
|
+
});
|
|
478
|
+
if (!assertion) {
|
|
479
|
+
throw new Error("Passkey authentication was cancelled");
|
|
480
|
+
}
|
|
481
|
+
const response = assertion.response;
|
|
482
|
+
const signature = new Uint8Array(response.signature);
|
|
483
|
+
let { r, s } = parseDerSignature(signature);
|
|
484
|
+
s = normalizeLowS(s);
|
|
485
|
+
return {
|
|
486
|
+
signature: new Uint8Array([...r, ...s]),
|
|
487
|
+
authenticatorData: new Uint8Array(response.authenticatorData),
|
|
488
|
+
clientDataJSON: new Uint8Array(response.clientDataJSON),
|
|
489
|
+
signatureR: r,
|
|
490
|
+
signatureS: s,
|
|
491
|
+
credentialId: arrayBufferToBase64Url2(assertion.rawId)
|
|
492
|
+
};
|
|
493
|
+
}
|
|
494
|
+
async function signWithPasskeyViaPopup(credentialId, challenge, rpId, preopenedPopup) {
|
|
495
|
+
const result = await requestPasskeyPopup(
|
|
496
|
+
"get",
|
|
497
|
+
{
|
|
498
|
+
credentialId,
|
|
499
|
+
challengeBase64Url: bytesToBase64Url(challenge),
|
|
500
|
+
rpId
|
|
501
|
+
},
|
|
502
|
+
preopenedPopup
|
|
503
|
+
);
|
|
504
|
+
return decodePopupSigningResult(result);
|
|
505
|
+
}
|
|
506
|
+
async function requestStoredPasskeyPopup(challenge, preopenedPopup, context) {
|
|
507
|
+
const result = await requestPasskeyPopup(
|
|
508
|
+
"getStored",
|
|
509
|
+
{
|
|
510
|
+
challengeBase64Url: bytesToBase64Url(challenge),
|
|
511
|
+
context
|
|
512
|
+
},
|
|
513
|
+
preopenedPopup
|
|
514
|
+
);
|
|
515
|
+
return decodePopupStoredSigningResult(result);
|
|
516
|
+
}
|
|
517
|
+
function decodePopupSigningResult(result) {
|
|
518
|
+
return {
|
|
519
|
+
signature: base64UrlToBytes(result.signatureBase64Url),
|
|
520
|
+
authenticatorData: base64UrlToBytes(result.authenticatorDataBase64Url),
|
|
521
|
+
clientDataJSON: base64UrlToBytes(result.clientDataJSONBase64Url),
|
|
522
|
+
signatureR: base64UrlToBytes(result.signatureRBase64Url),
|
|
523
|
+
signatureS: base64UrlToBytes(result.signatureSBase64Url)
|
|
524
|
+
};
|
|
525
|
+
}
|
|
526
|
+
function decodePopupStoredSigningResult(result) {
|
|
527
|
+
return {
|
|
528
|
+
...decodePopupSigningResult(result),
|
|
529
|
+
passkey: result.passkey,
|
|
530
|
+
accounts: result.accounts
|
|
531
|
+
};
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
// src/web.ts
|
|
535
|
+
import {
|
|
536
|
+
parseDerSignature as parseDerSignature2,
|
|
537
|
+
normalizeLowS as normalizeLowS2,
|
|
538
|
+
normalizeSignatureComponent,
|
|
539
|
+
P256_N,
|
|
540
|
+
P256_HALF_N,
|
|
541
|
+
bytesToBigIntBE,
|
|
542
|
+
bigIntToBytesBE
|
|
543
|
+
} from "@thru/passkey-manager";
|
|
544
|
+
import {
|
|
545
|
+
arrayBufferToBase64Url as arrayBufferToBase64Url3,
|
|
546
|
+
base64UrlToArrayBuffer as base64UrlToArrayBuffer2,
|
|
547
|
+
bytesToBase64,
|
|
548
|
+
bytesToBase64Url as bytesToBase64Url2,
|
|
549
|
+
base64UrlToBytes as base64UrlToBytes2,
|
|
550
|
+
bytesToHex as bytesToHex2,
|
|
551
|
+
hexToBytes,
|
|
552
|
+
bytesEqual,
|
|
553
|
+
compareBytes,
|
|
554
|
+
uniqueAccounts
|
|
555
|
+
} from "@thru/passkey-manager";
|
|
556
|
+
|
|
557
|
+
export {
|
|
558
|
+
isWebAuthnSupported,
|
|
559
|
+
preloadPasskeyClientCapabilities,
|
|
560
|
+
getPasskeyClientCapabilities,
|
|
561
|
+
getCachedPasskeyClientCapabilities,
|
|
562
|
+
isInIframe,
|
|
563
|
+
shouldUsePasskeyPopup,
|
|
564
|
+
registerPasskey,
|
|
565
|
+
signWithPasskey,
|
|
566
|
+
signWithStoredPasskey,
|
|
567
|
+
signWithDiscoverablePasskey,
|
|
568
|
+
parseDerSignature2 as parseDerSignature,
|
|
569
|
+
normalizeLowS2 as normalizeLowS,
|
|
570
|
+
normalizeSignatureComponent,
|
|
571
|
+
P256_N,
|
|
572
|
+
P256_HALF_N,
|
|
573
|
+
bytesToBigIntBE,
|
|
574
|
+
bigIntToBytesBE,
|
|
575
|
+
arrayBufferToBase64Url3 as arrayBufferToBase64Url,
|
|
576
|
+
base64UrlToArrayBuffer2 as base64UrlToArrayBuffer,
|
|
577
|
+
bytesToBase64,
|
|
578
|
+
bytesToBase64Url2 as bytesToBase64Url,
|
|
579
|
+
base64UrlToBytes2 as base64UrlToBytes,
|
|
580
|
+
bytesToHex2 as bytesToHex,
|
|
581
|
+
hexToBytes,
|
|
582
|
+
bytesEqual,
|
|
583
|
+
compareBytes,
|
|
584
|
+
uniqueAccounts
|
|
585
|
+
};
|
|
586
|
+
//# sourceMappingURL=chunk-B5SN7AS7.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/register.ts","../src/capabilities.ts","../src/sign.ts","../src/web.ts"],"sourcesContent":["import type { PasskeyRegistrationResult, PasskeyPopupRegistrationResult } from './types';\nimport { arrayBufferToBase64Url, bytesToHex } from '@thru/passkey-manager';\nimport {\n isWebAuthnSupported,\n getPasskeyPromptMode,\n maybePreopenPopup,\n shouldFallbackToPopup,\n type PasskeyPromptAction,\n} from './capabilities';\nimport { requestPasskeyPopup, openPasskeyPopupWindow, closePopup } from './popup';\n\n/**\n * Register a new passkey for a profile.\n */\nexport async function registerPasskey(\n alias: string,\n userId: string,\n rpId: string\n): Promise<PasskeyRegistrationResult> {\n if (!isWebAuthnSupported()) {\n throw new Error('WebAuthn is not supported in this browser');\n }\n\n return runWithPromptMode(\n 'create',\n () => registerPasskeyInline(alias, userId, rpId),\n (preopenedPopup) => registerPasskeyViaPopup(alias, userId, rpId, preopenedPopup)\n );\n}\n\nasync function runWithPromptMode<T>(\n action: PasskeyPromptAction,\n inlineFn: () => Promise<T>,\n popupFn: (preopenedPopup?: Window | null) => Promise<T>\n): Promise<T> {\n const preopenedPopup = maybePreopenPopup(action, openPasskeyPopupWindow);\n const promptMode = await getPasskeyPromptMode(action);\n if (promptMode === 'popup') {\n return popupFn(preopenedPopup);\n }\n\n closePopup(preopenedPopup);\n\n try {\n return await inlineFn();\n } catch (error) {\n if (shouldFallbackToPopup(error)) {\n return popupFn();\n }\n throw error;\n }\n}\n\nasync function registerPasskeyInline(\n alias: string,\n userId: string,\n rpId: string\n): Promise<PasskeyRegistrationResult> {\n const rpName = 'Thru Wallet';\n\n const userIdBytes = new TextEncoder().encode(userId);\n const userIdBuffer = userIdBytes.slice(0, 64);\n\n const challenge = crypto.getRandomValues(new Uint8Array(32));\n\n const createOptions: PublicKeyCredentialCreationOptions = {\n challenge,\n rp: {\n id: rpId,\n name: rpName,\n },\n user: {\n id: userIdBuffer,\n name: alias,\n displayName: alias,\n },\n pubKeyCredParams: [\n { type: 'public-key', alg: -7 },\n ],\n authenticatorSelection: {\n authenticatorAttachment: 'platform',\n userVerification: 'required',\n residentKey: 'required',\n requireResidentKey: true,\n },\n attestation: 'none',\n timeout: 60000,\n };\n\n const credential = (await navigator.credentials.create({\n publicKey: createOptions,\n })) as PublicKeyCredential | null;\n\n if (!credential) {\n throw new Error('Passkey registration was cancelled');\n }\n\n const response = credential.response as AuthenticatorAttestationResponse;\n const { x, y } = extractP256PublicKey(response);\n\n return {\n credentialId: arrayBufferToBase64Url(credential.rawId),\n publicKeyX: bytesToHex(x),\n publicKeyY: bytesToHex(y),\n rpId,\n };\n}\n\nasync function registerPasskeyViaPopup(\n alias: string,\n userId: string,\n rpId: string,\n preopenedPopup?: Window | null\n): Promise<PasskeyRegistrationResult> {\n const result = await requestPasskeyPopup<PasskeyPopupRegistrationResult>(\n 'create',\n { alias, userId, rpId },\n preopenedPopup\n );\n return result;\n}\n\n// Key extraction helpers\n\nfunction extractP256PublicKey(\n response: AuthenticatorAttestationResponse\n): { x: Uint8Array; y: Uint8Array } {\n if (typeof response.getPublicKey === 'function') {\n const spkiKey = response.getPublicKey();\n if (spkiKey) {\n return extractFromSpki(new Uint8Array(spkiKey));\n }\n }\n\n if (typeof response.getAuthenticatorData === 'function') {\n const authData = new Uint8Array(response.getAuthenticatorData());\n return extractFromAuthenticatorData(authData);\n }\n\n throw new Error('Unable to extract public key: browser does not support required WebAuthn methods');\n}\n\nfunction extractFromSpki(spki: Uint8Array): { x: Uint8Array; y: Uint8Array } {\n const pointStart = spki.length - 65;\n\n if (spki[pointStart] !== 0x04) {\n throw new Error('Invalid SPKI format: expected uncompressed point');\n }\n\n const x = spki.slice(pointStart + 1, pointStart + 33);\n const y = spki.slice(pointStart + 33, pointStart + 65);\n\n if (x.length !== 32 || y.length !== 32) {\n throw new Error('Invalid SPKI format: incorrect coordinate length');\n }\n\n return { x, y };\n}\n\nfunction extractFromAuthenticatorData(authData: Uint8Array): { x: Uint8Array; y: Uint8Array } {\n const rpIdHashLength = 32;\n const flagsLength = 1;\n const counterLength = 4;\n const offset = rpIdHashLength + flagsLength + counterLength;\n const aaguidLength = 16;\n const credIdLenOffset = offset + aaguidLength;\n const credIdLength = (authData[credIdLenOffset] << 8) | authData[credIdLenOffset + 1];\n const coseKeyOffset = credIdLenOffset + 2 + credIdLength;\n const coseKey = authData.slice(coseKeyOffset);\n\n return extractFromCoseKey(coseKey);\n}\n\nfunction extractFromCoseKey(coseKey: Uint8Array): { x: Uint8Array; y: Uint8Array } {\n const mapStart = coseKey[0];\n if (mapStart !== 0xa5 && mapStart !== 0xa4) {\n throw new Error('Invalid COSE key format');\n }\n\n let offset = 1;\n let x: Uint8Array | null = null;\n let y: Uint8Array | null = null;\n\n while (offset < coseKey.length) {\n const key = coseKey[offset++];\n const valueType = coseKey[offset++];\n\n if (key === 0x21) {\n const length = valueType & 0x1f;\n x = coseKey.slice(offset, offset + length);\n offset += length;\n continue;\n }\n\n if (key === 0x22) {\n const length = valueType & 0x1f;\n y = coseKey.slice(offset, offset + length);\n offset += length;\n continue;\n }\n\n if (valueType >= 0x40 && valueType <= 0x5f) {\n const length = valueType & 0x1f;\n offset += length;\n continue;\n }\n\n if (valueType === 0x01 || valueType === 0x02 || valueType === 0x03) {\n continue;\n }\n\n if (valueType >= 0x18 && valueType <= 0x1b) {\n const size = 1 << (valueType - 0x18);\n offset += size;\n continue;\n }\n }\n\n if (!x || !y) {\n throw new Error('Failed to extract P-256 public key from COSE data');\n }\n\n if (x.length !== 32 || y.length !== 32) {\n throw new Error('Invalid COSE key: incorrect coordinate length');\n }\n\n return { x, y };\n}\n","import type { PasskeyClientCapabilities } from './types';\n\nconst DEBUG = typeof process !== 'undefined' && process.env?.NEXT_PUBLIC_PASSKEY_DEBUG === '1';\n\nlet cachedClientCapabilities: PasskeyClientCapabilities | null | undefined;\nlet clientCapabilitiesPromise: Promise<PasskeyClientCapabilities | null> | null = null;\n\nexport function isWebAuthnSupported(): boolean {\n const supported =\n typeof window !== 'undefined' &&\n typeof window.PublicKeyCredential !== 'undefined' &&\n typeof navigator.credentials !== 'undefined';\n\n if (DEBUG) {\n console.log('[Passkey] WebAuthn support check:', {\n window: typeof window !== 'undefined',\n PublicKeyCredential:\n typeof window !== 'undefined' && typeof window.PublicKeyCredential !== 'undefined',\n credentials:\n typeof window !== 'undefined' &&\n typeof navigator !== 'undefined' &&\n typeof navigator.credentials !== 'undefined',\n supported,\n });\n }\n\n return supported;\n}\n\nasync function fetchPasskeyClientCapabilities(): Promise<PasskeyClientCapabilities | null> {\n if (typeof window === 'undefined' || typeof window.PublicKeyCredential === 'undefined') {\n return null;\n }\n\n const getClientCapabilities = (window.PublicKeyCredential as {\n getClientCapabilities?: () => Promise<PasskeyClientCapabilities>;\n }).getClientCapabilities;\n\n if (typeof getClientCapabilities !== 'function') {\n return null;\n }\n\n try {\n const capabilities = await getClientCapabilities.call(window.PublicKeyCredential);\n if (DEBUG) {\n console.log('[Passkey] WebAuthn client capabilities:', capabilities);\n }\n return capabilities ?? null;\n } catch (error) {\n if (DEBUG) {\n console.warn('[Passkey] Failed to read client capabilities:', error);\n }\n return null;\n }\n}\n\nexport function preloadPasskeyClientCapabilities(): void {\n if (cachedClientCapabilities !== undefined || clientCapabilitiesPromise) {\n return;\n }\n\n clientCapabilitiesPromise = fetchPasskeyClientCapabilities().then((capabilities) => {\n cachedClientCapabilities = capabilities;\n return capabilities;\n });\n}\n\nexport async function getPasskeyClientCapabilities(): Promise<PasskeyClientCapabilities | null> {\n if (cachedClientCapabilities !== undefined) {\n return cachedClientCapabilities;\n }\n\n if (!clientCapabilitiesPromise) {\n preloadPasskeyClientCapabilities();\n }\n\n if (!clientCapabilitiesPromise) {\n cachedClientCapabilities = null;\n return null;\n }\n\n const capabilities = await clientCapabilitiesPromise;\n cachedClientCapabilities = capabilities;\n return capabilities;\n}\n\nexport function getCachedPasskeyClientCapabilities(): PasskeyClientCapabilities | null | undefined {\n return cachedClientCapabilities;\n}\n\nexport function isInIframe(): boolean {\n if (typeof window === 'undefined') {\n return false;\n }\n try {\n return window.self !== window.top;\n } catch {\n return true;\n }\n}\n\nexport type PasskeyPromptAction = 'get' | 'create';\n\nexport async function shouldUsePasskeyPopup(action: PasskeyPromptAction): Promise<boolean> {\n if (!isInIframe()) {\n return false;\n }\n const mode = await getPasskeyPromptMode(action);\n return mode === 'popup';\n}\n\ntype PasskeyPromptMode = 'inline' | 'popup';\n\nfunction getPermissionsPolicyAllowsFeature(feature: string): boolean | null {\n if (typeof document === 'undefined') {\n return null;\n }\n\n const policy = (document as { permissionsPolicy?: { allowsFeature?: (name: string) => boolean } })\n .permissionsPolicy;\n const featurePolicy = (document as { featurePolicy?: { allowsFeature?: (name: string) => boolean } })\n .featurePolicy;\n const allowsFeature = policy?.allowsFeature || featurePolicy?.allowsFeature;\n\n if (typeof allowsFeature !== 'function') {\n return null;\n }\n\n try {\n return allowsFeature(feature);\n } catch {\n return null;\n }\n}\n\nfunction getCachedPromptMode(action: PasskeyPromptAction): PasskeyPromptMode | 'unknown' {\n if (!isInIframe()) {\n return 'inline';\n }\n\n if (cachedClientCapabilities === undefined && !clientCapabilitiesPromise) {\n preloadPasskeyClientCapabilities();\n }\n\n const feature =\n action === 'create' ? 'publickey-credentials-create' : 'publickey-credentials-get';\n const policyAllows = getPermissionsPolicyAllowsFeature(feature);\n const capabilities = getCachedPasskeyClientCapabilities();\n const supportsInline =\n capabilities?.passkeyPlatformAuthenticator === true ||\n capabilities?.userVerifyingPlatformAuthenticator === true;\n\n if (policyAllows === false) {\n return 'popup';\n }\n\n if (capabilities === undefined) {\n return 'unknown';\n }\n\n if (!supportsInline) {\n return 'popup';\n }\n\n return 'inline';\n}\n\nexport async function getPasskeyPromptMode(action: PasskeyPromptAction): Promise<PasskeyPromptMode> {\n if (!isInIframe()) {\n return 'inline';\n }\n\n const feature =\n action === 'create' ? 'publickey-credentials-create' : 'publickey-credentials-get';\n const policyAllows = getPermissionsPolicyAllowsFeature(feature);\n const capabilities = await getPasskeyClientCapabilities();\n const supportsInline =\n capabilities?.passkeyPlatformAuthenticator === true ||\n capabilities?.userVerifyingPlatformAuthenticator === true;\n\n if (DEBUG) {\n console.log('[Passkey] Prompt mode check:', {\n action,\n policyAllows,\n supportsInline,\n capabilities,\n });\n }\n\n if (!supportsInline) {\n return 'popup';\n }\n\n if (policyAllows === false) {\n return 'popup';\n }\n\n return 'inline';\n}\n\nexport function maybePreopenPopup(action: PasskeyPromptAction, openPopupFn: () => Window): Window | null {\n const cachedMode = getCachedPromptMode(action);\n if (cachedMode !== 'popup') {\n return null;\n }\n\n try {\n return openPopupFn();\n } catch {\n return null;\n }\n}\n\nexport function shouldFallbackToPopup(error: unknown): boolean {\n if (!isInIframe()) {\n return false;\n }\n\n const name =\n error && typeof error === 'object' && 'name' in error ? String((error as { name?: unknown }).name) : '';\n const message =\n error && typeof error === 'object' && 'message' in error\n ? String((error as { message?: unknown }).message)\n : '';\n const normalized = `${name} ${message}`.toLowerCase();\n\n if (\n normalized.includes('cancel') ||\n normalized.includes('canceled') ||\n normalized.includes('cancelled') ||\n normalized.includes('user canceled') ||\n normalized.includes('user cancelled') ||\n normalized.includes('aborted')\n ) {\n return false;\n }\n\n if (normalized.includes('securityerror')) {\n return true;\n }\n\n if (normalized.includes('notallowederror')) {\n if (\n normalized.includes('permission') ||\n normalized.includes('policy') ||\n normalized.includes('iframe') ||\n normalized.includes('frame')\n ) {\n return true;\n }\n }\n\n return false;\n}\n","import type {\n PasskeySigningResult,\n PasskeyStoredSigningResult,\n PasskeyDiscoverableSigningResult,\n PasskeyMetadata,\n PasskeyPopupContext,\n PasskeyPopupSigningResult,\n PasskeyPopupStoredSigningResult,\n} from './types';\nimport {\n arrayBufferToBase64Url,\n base64UrlToArrayBuffer,\n bytesToBase64Url,\n base64UrlToBytes,\n parseDerSignature,\n normalizeLowS,\n} from '@thru/passkey-manager';\nimport {\n isWebAuthnSupported,\n getPasskeyPromptMode,\n isInIframe,\n maybePreopenPopup,\n shouldFallbackToPopup,\n type PasskeyPromptAction,\n} from './capabilities';\nimport { requestPasskeyPopup, openPasskeyPopupWindow, closePopup } from './popup';\n\n/**\n * Sign a challenge with an existing passkey (by credential ID).\n */\nexport async function signWithPasskey(\n credentialId: string,\n challenge: Uint8Array,\n rpId: string\n): Promise<PasskeySigningResult> {\n if (!isWebAuthnSupported()) {\n throw new Error('WebAuthn is not supported in this browser');\n }\n\n return runWithPromptMode(\n 'get',\n () => signWithPasskeyInline(credentialId, challenge, rpId),\n (preopenedPopup) => signWithPasskeyViaPopup(credentialId, challenge, rpId, preopenedPopup)\n );\n}\n\n/**\n * Sign with stored passkey (for embedded/popup contexts).\n */\nexport async function signWithStoredPasskey(\n challenge: Uint8Array,\n rpId: string,\n preferredPasskey: PasskeyMetadata | null,\n allPasskeys: PasskeyMetadata[],\n context?: PasskeyPopupContext\n): Promise<PasskeyStoredSigningResult> {\n if (!isWebAuthnSupported()) {\n throw new Error('WebAuthn is not supported in this browser');\n }\n\n const preopenedPopup = maybePreopenPopup('get', openPasskeyPopupWindow);\n const promptMode = await getPasskeyPromptMode('get');\n const storedPasskey = preferredPasskey;\n const canUsePopup = isInIframe();\n\n if (promptMode === 'popup' || (canUsePopup && !storedPasskey)) {\n return requestStoredPasskeyPopup(challenge, preopenedPopup, context);\n }\n\n closePopup(preopenedPopup);\n\n try {\n if (storedPasskey) {\n const result = await signWithPasskeyInline(\n storedPasskey.credentialId,\n challenge,\n storedPasskey.rpId\n );\n return {\n ...result,\n passkey: storedPasskey,\n };\n }\n\n const discoverable = await signWithDiscoverablePasskey(challenge, rpId);\n const matchingPasskey = allPasskeys.find(p => p.credentialId === discoverable.credentialId) ?? null;\n const now = new Date().toISOString();\n const passkey = matchingPasskey ?? {\n credentialId: discoverable.credentialId,\n publicKeyX: '',\n publicKeyY: '',\n rpId: discoverable.rpId,\n createdAt: now,\n lastUsedAt: now,\n };\n\n return {\n signature: discoverable.signature,\n authenticatorData: discoverable.authenticatorData,\n clientDataJSON: discoverable.clientDataJSON,\n signatureR: discoverable.signatureR,\n signatureS: discoverable.signatureS,\n passkey,\n };\n } catch (error) {\n if (canUsePopup && shouldFallbackToPopup(error)) {\n return requestStoredPasskeyPopup(challenge, undefined, context);\n }\n\n throw error;\n }\n}\n\n/**\n * Sign with a discoverable passkey (no credential ID - browser prompts user to select).\n */\nexport async function signWithDiscoverablePasskey(\n challenge: Uint8Array,\n rpId: string\n): Promise<PasskeyDiscoverableSigningResult> {\n if (!isWebAuthnSupported()) {\n throw new Error('WebAuthn is not supported in this browser');\n }\n\n const resolvedRpId = rpId;\n const result = await signWithPasskeyAssertion(challenge, resolvedRpId);\n\n return {\n signature: result.signature,\n authenticatorData: result.authenticatorData,\n clientDataJSON: result.clientDataJSON,\n signatureR: result.signatureR,\n signatureS: result.signatureS,\n credentialId: result.credentialId,\n rpId: resolvedRpId,\n };\n}\n\n// Internal helpers\n\nasync function runWithPromptMode<T>(\n action: PasskeyPromptAction,\n inlineFn: () => Promise<T>,\n popupFn: (preopenedPopup?: Window | null) => Promise<T>\n): Promise<T> {\n const preopenedPopup = maybePreopenPopup(action, openPasskeyPopupWindow);\n const promptMode = await getPasskeyPromptMode(action);\n if (promptMode === 'popup') {\n return popupFn(preopenedPopup);\n }\n\n closePopup(preopenedPopup);\n\n try {\n return await inlineFn();\n } catch (error) {\n if (shouldFallbackToPopup(error)) {\n return popupFn();\n }\n throw error;\n }\n}\n\nasync function signWithPasskeyInline(\n credentialId: string,\n challenge: Uint8Array,\n rpId: string\n): Promise<PasskeySigningResult> {\n const result = await signWithPasskeyAssertion(challenge, rpId, credentialId);\n return {\n signature: result.signature,\n authenticatorData: result.authenticatorData,\n clientDataJSON: result.clientDataJSON,\n signatureR: result.signatureR,\n signatureS: result.signatureS,\n };\n}\n\nasync function signWithPasskeyAssertion(\n challenge: Uint8Array,\n rpId: string,\n credentialId?: string\n): Promise<PasskeySigningResult & { credentialId: string }> {\n const challengeBytes = new Uint8Array(challenge);\n const getOptions: PublicKeyCredentialRequestOptions = {\n challenge: challengeBytes,\n rpId,\n userVerification: 'required',\n timeout: 60000,\n };\n\n if (credentialId) {\n const credentialIdBuffer = base64UrlToArrayBuffer(credentialId);\n getOptions.allowCredentials = [\n {\n type: 'public-key',\n id: credentialIdBuffer,\n transports: ['internal', 'hybrid', 'usb', 'ble', 'nfc'],\n },\n ];\n }\n\n const assertion = (await navigator.credentials.get({\n publicKey: getOptions,\n })) as PublicKeyCredential | null;\n\n if (!assertion) {\n throw new Error('Passkey authentication was cancelled');\n }\n\n const response = assertion.response as AuthenticatorAssertionResponse;\n\n const signature = new Uint8Array(response.signature);\n let { r, s } = parseDerSignature(signature);\n s = normalizeLowS(s);\n\n return {\n signature: new Uint8Array([...r, ...s]),\n authenticatorData: new Uint8Array(response.authenticatorData),\n clientDataJSON: new Uint8Array(response.clientDataJSON),\n signatureR: r,\n signatureS: s,\n credentialId: arrayBufferToBase64Url(assertion.rawId),\n };\n}\n\nasync function signWithPasskeyViaPopup(\n credentialId: string,\n challenge: Uint8Array,\n rpId: string,\n preopenedPopup?: Window | null\n): Promise<PasskeySigningResult> {\n const result = await requestPasskeyPopup<PasskeyPopupSigningResult>(\n 'get',\n {\n credentialId,\n challengeBase64Url: bytesToBase64Url(challenge),\n rpId,\n },\n preopenedPopup\n );\n\n return decodePopupSigningResult(result);\n}\n\nasync function requestStoredPasskeyPopup(\n challenge: Uint8Array,\n preopenedPopup?: Window | null,\n context?: PasskeyPopupContext\n): Promise<PasskeyStoredSigningResult> {\n const result = await requestPasskeyPopup<PasskeyPopupStoredSigningResult>(\n 'getStored',\n {\n challengeBase64Url: bytesToBase64Url(challenge),\n context,\n },\n preopenedPopup\n );\n return decodePopupStoredSigningResult(result);\n}\n\nfunction decodePopupSigningResult(result: PasskeyPopupSigningResult): PasskeySigningResult {\n return {\n signature: base64UrlToBytes(result.signatureBase64Url),\n authenticatorData: base64UrlToBytes(result.authenticatorDataBase64Url),\n clientDataJSON: base64UrlToBytes(result.clientDataJSONBase64Url),\n signatureR: base64UrlToBytes(result.signatureRBase64Url),\n signatureS: base64UrlToBytes(result.signatureSBase64Url),\n };\n}\n\nfunction decodePopupStoredSigningResult(\n result: PasskeyPopupStoredSigningResult\n): PasskeyStoredSigningResult {\n return {\n ...decodePopupSigningResult(result),\n passkey: result.passkey,\n accounts: result.accounts,\n };\n}\n","export type {\n PasskeyRegistrationResult,\n PasskeySigningResult,\n PasskeyDiscoverableSigningResult,\n PasskeyStoredSigningResult,\n PasskeyMetadata,\n PasskeyClientCapabilities,\n PasskeyPopupContext,\n PasskeyPopupAccount,\n} from './types';\n\nexport { registerPasskey } from './register';\n\nexport {\n signWithPasskey,\n signWithStoredPasskey,\n signWithDiscoverablePasskey,\n} from './sign';\n\nexport {\n parseDerSignature,\n normalizeLowS,\n normalizeSignatureComponent,\n P256_N,\n P256_HALF_N,\n bytesToBigIntBE,\n bigIntToBytesBE,\n} from '@thru/passkey-manager';\n\nexport {\n isWebAuthnSupported,\n preloadPasskeyClientCapabilities,\n getPasskeyClientCapabilities,\n getCachedPasskeyClientCapabilities,\n shouldUsePasskeyPopup,\n isInIframe,\n type PasskeyPromptAction,\n} from './capabilities';\n\nexport {\n arrayBufferToBase64Url,\n base64UrlToArrayBuffer,\n bytesToBase64,\n bytesToBase64Url,\n base64UrlToBytes,\n bytesToHex,\n hexToBytes,\n bytesEqual,\n compareBytes,\n uniqueAccounts,\n} from '@thru/passkey-manager';\n"],"mappings":"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","names":["arrayBufferToBase64Url","runWithPromptMode","arrayBufferToBase64Url","parseDerSignature","normalizeLowS","arrayBufferToBase64Url","base64UrlToArrayBuffer","bytesToBase64Url","base64UrlToBytes","bytesToHex"]}
|