@metamask-previews/passkey-controller 0.0.0-preview-4c0846313

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.
Files changed (156) hide show
  1. package/CHANGELOG.md +24 -0
  2. package/LICENSE +21 -0
  3. package/README.md +155 -0
  4. package/dist/PasskeyController.cjs +448 -0
  5. package/dist/PasskeyController.cjs.map +1 -0
  6. package/dist/PasskeyController.d.cts +168 -0
  7. package/dist/PasskeyController.d.cts.map +1 -0
  8. package/dist/PasskeyController.d.mts +168 -0
  9. package/dist/PasskeyController.d.mts.map +1 -0
  10. package/dist/PasskeyController.mjs +443 -0
  11. package/dist/PasskeyController.mjs.map +1 -0
  12. package/dist/ceremony-manager.cjs +134 -0
  13. package/dist/ceremony-manager.cjs.map +1 -0
  14. package/dist/ceremony-manager.d.cts +71 -0
  15. package/dist/ceremony-manager.d.cts.map +1 -0
  16. package/dist/ceremony-manager.d.mts +71 -0
  17. package/dist/ceremony-manager.d.mts.map +1 -0
  18. package/dist/ceremony-manager.mjs +130 -0
  19. package/dist/ceremony-manager.mjs.map +1 -0
  20. package/dist/constants.cjs +33 -0
  21. package/dist/constants.cjs.map +1 -0
  22. package/dist/constants.d.cts +30 -0
  23. package/dist/constants.d.cts.map +1 -0
  24. package/dist/constants.d.mts +30 -0
  25. package/dist/constants.d.mts.map +1 -0
  26. package/dist/constants.mjs +30 -0
  27. package/dist/constants.mjs.map +1 -0
  28. package/dist/errors.cjs +57 -0
  29. package/dist/errors.cjs.map +1 -0
  30. package/dist/errors.d.cts +34 -0
  31. package/dist/errors.d.cts.map +1 -0
  32. package/dist/errors.d.mts +34 -0
  33. package/dist/errors.d.mts.map +1 -0
  34. package/dist/errors.mjs +53 -0
  35. package/dist/errors.mjs.map +1 -0
  36. package/dist/index.cjs +19 -0
  37. package/dist/index.cjs.map +1 -0
  38. package/dist/index.d.cts +9 -0
  39. package/dist/index.d.cts.map +1 -0
  40. package/dist/index.d.mts +9 -0
  41. package/dist/index.d.mts.map +1 -0
  42. package/dist/index.mjs +5 -0
  43. package/dist/index.mjs.map +1 -0
  44. package/dist/key-derivation.cjs +76 -0
  45. package/dist/key-derivation.cjs.map +1 -0
  46. package/dist/key-derivation.d.cts +43 -0
  47. package/dist/key-derivation.d.cts.map +1 -0
  48. package/dist/key-derivation.d.mts +43 -0
  49. package/dist/key-derivation.d.mts.map +1 -0
  50. package/dist/key-derivation.mjs +71 -0
  51. package/dist/key-derivation.mjs.map +1 -0
  52. package/dist/logger.cjs +9 -0
  53. package/dist/logger.cjs.map +1 -0
  54. package/dist/logger.d.cts +5 -0
  55. package/dist/logger.d.cts.map +1 -0
  56. package/dist/logger.d.mts +5 -0
  57. package/dist/logger.d.mts.map +1 -0
  58. package/dist/logger.mjs +6 -0
  59. package/dist/logger.mjs.map +1 -0
  60. package/dist/types.cjs +3 -0
  61. package/dist/types.cjs.map +1 -0
  62. package/dist/types.d.cts +92 -0
  63. package/dist/types.d.cts.map +1 -0
  64. package/dist/types.d.mts +92 -0
  65. package/dist/types.d.mts.map +1 -0
  66. package/dist/types.mjs +2 -0
  67. package/dist/types.mjs.map +1 -0
  68. package/dist/utils/crypto.cjs +55 -0
  69. package/dist/utils/crypto.cjs.map +1 -0
  70. package/dist/utils/crypto.d.cts +30 -0
  71. package/dist/utils/crypto.d.cts.map +1 -0
  72. package/dist/utils/crypto.d.mts +30 -0
  73. package/dist/utils/crypto.d.mts.map +1 -0
  74. package/dist/utils/crypto.mjs +49 -0
  75. package/dist/utils/crypto.mjs.map +1 -0
  76. package/dist/utils/encoding.cjs +42 -0
  77. package/dist/utils/encoding.cjs.map +1 -0
  78. package/dist/utils/encoding.d.cts +22 -0
  79. package/dist/utils/encoding.d.cts.map +1 -0
  80. package/dist/utils/encoding.d.mts +22 -0
  81. package/dist/utils/encoding.d.mts.map +1 -0
  82. package/dist/utils/encoding.mjs +36 -0
  83. package/dist/utils/encoding.mjs.map +1 -0
  84. package/dist/webauthn/constants.cjs +74 -0
  85. package/dist/webauthn/constants.cjs.map +1 -0
  86. package/dist/webauthn/constants.d.cts +68 -0
  87. package/dist/webauthn/constants.d.cts.map +1 -0
  88. package/dist/webauthn/constants.d.mts +68 -0
  89. package/dist/webauthn/constants.d.mts.map +1 -0
  90. package/dist/webauthn/constants.mjs +71 -0
  91. package/dist/webauthn/constants.mjs.map +1 -0
  92. package/dist/webauthn/decode-attestation-object.cjs +18 -0
  93. package/dist/webauthn/decode-attestation-object.cjs.map +1 -0
  94. package/dist/webauthn/decode-attestation-object.d.cts +10 -0
  95. package/dist/webauthn/decode-attestation-object.d.cts.map +1 -0
  96. package/dist/webauthn/decode-attestation-object.d.mts +10 -0
  97. package/dist/webauthn/decode-attestation-object.d.mts.map +1 -0
  98. package/dist/webauthn/decode-attestation-object.mjs +14 -0
  99. package/dist/webauthn/decode-attestation-object.mjs.map +1 -0
  100. package/dist/webauthn/decode-client-data-json.cjs +17 -0
  101. package/dist/webauthn/decode-client-data-json.cjs.map +1 -0
  102. package/dist/webauthn/decode-client-data-json.d.cts +9 -0
  103. package/dist/webauthn/decode-client-data-json.d.cts.map +1 -0
  104. package/dist/webauthn/decode-client-data-json.d.mts +9 -0
  105. package/dist/webauthn/decode-client-data-json.d.mts.map +1 -0
  106. package/dist/webauthn/decode-client-data-json.mjs +13 -0
  107. package/dist/webauthn/decode-client-data-json.mjs.map +1 -0
  108. package/dist/webauthn/match-expected-rp-id.cjs +43 -0
  109. package/dist/webauthn/match-expected-rp-id.cjs.map +1 -0
  110. package/dist/webauthn/match-expected-rp-id.d.cts +11 -0
  111. package/dist/webauthn/match-expected-rp-id.d.cts.map +1 -0
  112. package/dist/webauthn/match-expected-rp-id.d.mts +11 -0
  113. package/dist/webauthn/match-expected-rp-id.d.mts.map +1 -0
  114. package/dist/webauthn/match-expected-rp-id.mjs +39 -0
  115. package/dist/webauthn/match-expected-rp-id.mjs.map +1 -0
  116. package/dist/webauthn/parse-authenticator-data.cjs +69 -0
  117. package/dist/webauthn/parse-authenticator-data.cjs.map +1 -0
  118. package/dist/webauthn/parse-authenticator-data.d.cts +10 -0
  119. package/dist/webauthn/parse-authenticator-data.d.cts.map +1 -0
  120. package/dist/webauthn/parse-authenticator-data.d.mts +10 -0
  121. package/dist/webauthn/parse-authenticator-data.d.mts.map +1 -0
  122. package/dist/webauthn/parse-authenticator-data.mjs +65 -0
  123. package/dist/webauthn/parse-authenticator-data.mjs.map +1 -0
  124. package/dist/webauthn/types.cjs +3 -0
  125. package/dist/webauthn/types.cjs.map +1 -0
  126. package/dist/webauthn/types.d.cts +113 -0
  127. package/dist/webauthn/types.d.cts.map +1 -0
  128. package/dist/webauthn/types.d.mts +113 -0
  129. package/dist/webauthn/types.d.mts.map +1 -0
  130. package/dist/webauthn/types.mjs +2 -0
  131. package/dist/webauthn/types.mjs.map +1 -0
  132. package/dist/webauthn/verify-authentication-response.cjs +134 -0
  133. package/dist/webauthn/verify-authentication-response.cjs.map +1 -0
  134. package/dist/webauthn/verify-authentication-response.d.cts +63 -0
  135. package/dist/webauthn/verify-authentication-response.d.cts.map +1 -0
  136. package/dist/webauthn/verify-authentication-response.d.mts +63 -0
  137. package/dist/webauthn/verify-authentication-response.d.mts.map +1 -0
  138. package/dist/webauthn/verify-authentication-response.mjs +130 -0
  139. package/dist/webauthn/verify-authentication-response.mjs.map +1 -0
  140. package/dist/webauthn/verify-registration-response.cjs +205 -0
  141. package/dist/webauthn/verify-registration-response.cjs.map +1 -0
  142. package/dist/webauthn/verify-registration-response.d.cts +60 -0
  143. package/dist/webauthn/verify-registration-response.d.cts.map +1 -0
  144. package/dist/webauthn/verify-registration-response.d.mts +60 -0
  145. package/dist/webauthn/verify-registration-response.d.mts.map +1 -0
  146. package/dist/webauthn/verify-registration-response.mjs +201 -0
  147. package/dist/webauthn/verify-registration-response.mjs.map +1 -0
  148. package/dist/webauthn/verify-signature.cjs +176 -0
  149. package/dist/webauthn/verify-signature.cjs.map +1 -0
  150. package/dist/webauthn/verify-signature.d.cts +21 -0
  151. package/dist/webauthn/verify-signature.d.cts.map +1 -0
  152. package/dist/webauthn/verify-signature.d.mts +21 -0
  153. package/dist/webauthn/verify-signature.d.mts.map +1 -0
  154. package/dist/webauthn/verify-signature.mjs +172 -0
  155. package/dist/webauthn/verify-signature.mjs.map +1 -0
  156. package/package.json +78 -0
@@ -0,0 +1,130 @@
1
+ import { decodePartialCBOR } from "@levischuck/tiny-cbor";
2
+ import { concatBytes } from "@metamask/utils";
3
+ import { sha256 } from "@noble/hashes/sha2";
4
+ import { base64URLToBytes } from "../utils/encoding.mjs";
5
+ import { decodeClientDataJSON } from "./decode-client-data-json.mjs";
6
+ import { matchExpectedRPID } from "./match-expected-rp-id.mjs";
7
+ import { parseAuthenticatorData } from "./parse-authenticator-data.mjs";
8
+ import { verifySignature } from "./verify-signature.mjs";
9
+ /**
10
+ * Verifies a WebAuthn authentication (assertion) response per
11
+ * W3C WebAuthn Level 3 §7.2.
12
+ *
13
+ * Performs the following checks in order:
14
+ * 1. Credential ID presence, base64url consistency, and type.
15
+ * 2. `clientDataJSON` -- type is `"webauthn.get"`, challenge and origin
16
+ * match.
17
+ * 3. `authenticatorData` -- RP ID hash matches, user-presence flag is
18
+ * set, and optional user-verification flag is checked.
19
+ * 4. Signature verification -- `signature` is verified over
20
+ * `authData || SHA-256(clientDataJSON)` using the stored credential
21
+ * public key (COSE-encoded).
22
+ * 5. Counter monotonicity -- if either the stored or returned counter
23
+ * is non-zero, the new counter must exceed the stored value.
24
+ *
25
+ * @param opts - Verification options.
26
+ * @param opts.response - The `PublicKeyCredential` result from
27
+ * `navigator.credentials.get()`, serialized as JSON.
28
+ * @param opts.expectedChallenge - The base64url challenge that was issued
29
+ * for this ceremony.
30
+ * @param opts.expectedOrigin - One or more acceptable origins.
31
+ * @param opts.expectedRPID - The Relying Party ID domain.
32
+ * @param opts.credential - The stored credential record to verify against.
33
+ * @param opts.credential.id - The credential ID (base64url).
34
+ * @param opts.credential.publicKey - The COSE-encoded public key bytes
35
+ * persisted during registration.
36
+ * @param opts.credential.counter - The last known signature counter value.
37
+ * @param opts.credential.transports - Optional authenticator transports.
38
+ * @param opts.requireUserVerification - When `true`, verification fails
39
+ * if the UV flag is not set. Defaults to `false`.
40
+ * @returns Verification result containing `verified` status and parsed
41
+ * authentication info (new counter, origin, RP ID).
42
+ */
43
+ export async function verifyAuthenticationResponse(opts) {
44
+ const { response, expectedChallenge, expectedOrigin, expectedRPID, credential, requireUserVerification = false, } = opts;
45
+ const { id, rawId, type: credentialType, response: assertionResponse, } = response;
46
+ // Ensure credential specified an ID
47
+ if (!id) {
48
+ throw new Error('Missing credential ID');
49
+ }
50
+ // Ensure ID is base64url-encoded
51
+ if (id !== rawId) {
52
+ throw new Error('Credential ID was not base64url-encoded');
53
+ }
54
+ // Make sure credential type is public-key
55
+ if (credentialType !== 'public-key') {
56
+ throw new Error(`Unexpected credential type ${String(credentialType)}, expected "public-key"`);
57
+ }
58
+ if (typeof assertionResponse?.clientDataJSON !== 'string') {
59
+ throw new Error('Credential response clientDataJSON was not a string');
60
+ }
61
+ const clientDataJSON = decodeClientDataJSON(assertionResponse.clientDataJSON);
62
+ const { type, challenge, origin, tokenBinding } = clientDataJSON;
63
+ // Make sure we're handling an authentication
64
+ if (type !== 'webauthn.get') {
65
+ throw new Error(`Unexpected authentication response type: ${type}`);
66
+ }
67
+ // Ensure the device provided the challenge we gave it
68
+ if (challenge !== expectedChallenge) {
69
+ throw new Error(`Unexpected authentication response challenge "${challenge}", expected "${expectedChallenge}"`);
70
+ }
71
+ // Check that the origin is our site
72
+ const expectedOrigins = Array.isArray(expectedOrigin)
73
+ ? expectedOrigin
74
+ : [expectedOrigin];
75
+ if (!expectedOrigins.includes(origin)) {
76
+ throw new Error(`Unexpected authentication response origin "${origin}", expected one of: ${expectedOrigins.join(', ')}`);
77
+ }
78
+ if (assertionResponse.userHandle &&
79
+ typeof assertionResponse.userHandle !== 'string') {
80
+ throw new Error('Credential response userHandle was not a string');
81
+ }
82
+ if (tokenBinding) {
83
+ if (typeof tokenBinding !== 'object') {
84
+ throw new Error('ClientDataJSON tokenBinding was not an object');
85
+ }
86
+ if (!['present', 'supported', 'not-supported'].includes(tokenBinding.status)) {
87
+ throw new Error(`Unexpected tokenBinding status ${tokenBinding.status}`);
88
+ }
89
+ }
90
+ const authDataBuffer = base64URLToBytes(assertionResponse.authenticatorData);
91
+ const parsedAuthData = parseAuthenticatorData(authDataBuffer);
92
+ const { rpIdHash, flags, counter } = parsedAuthData;
93
+ // Make sure the response's RP ID is ours
94
+ const matchedRPID = matchExpectedRPID(rpIdHash, [expectedRPID]);
95
+ // WebAuthn only requires the user presence flag be true
96
+ if (!flags.up) {
97
+ throw new Error('User not present during authentication');
98
+ }
99
+ // Enforce user verification if required
100
+ if (requireUserVerification && !flags.uv) {
101
+ throw new Error('User verification required, but user could not be verified');
102
+ }
103
+ const clientDataHash = sha256(base64URLToBytes(assertionResponse.clientDataJSON));
104
+ const signatureBase = concatBytes([authDataBuffer, clientDataHash]);
105
+ const signature = base64URLToBytes(assertionResponse.signature);
106
+ const cosePublicKey = decodePartialCBOR(new Uint8Array(credential.publicKey), 0)[0];
107
+ const verified = await verifySignature({
108
+ cosePublicKey,
109
+ signature,
110
+ data: signatureBase,
111
+ });
112
+ if (!verified) {
113
+ return { verified: false };
114
+ }
115
+ if ((counter > 0 || credential.counter > 0) &&
116
+ counter <= credential.counter) {
117
+ throw new Error(`Response counter value ${counter} was lower than expected ${credential.counter}`);
118
+ }
119
+ return {
120
+ verified: true,
121
+ authenticationInfo: {
122
+ credentialId: credential.id,
123
+ newCounter: counter,
124
+ userVerified: flags.uv,
125
+ origin: clientDataJSON.origin,
126
+ rpID: matchedRPID,
127
+ },
128
+ };
129
+ }
130
+ //# sourceMappingURL=verify-authentication-response.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify-authentication-response.mjs","sourceRoot":"","sources":["../../src/webauthn/verify-authentication-response.ts"],"names":[],"mappings":"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","sourcesContent":["import { decodePartialCBOR } from '@levischuck/tiny-cbor';\nimport { concatBytes } from '@metamask/utils';\nimport { sha256 } from '@noble/hashes/sha2';\n\nimport type { AuthenticatorTransportFuture } from '../types';\nimport { base64URLToBytes } from '../utils/encoding';\nimport { decodeClientDataJSON } from './decode-client-data-json';\nimport { matchExpectedRPID } from './match-expected-rp-id';\nimport { parseAuthenticatorData } from './parse-authenticator-data';\nimport type { ParsedAuthenticatorData } from './types';\nimport type { PasskeyAuthenticationResponse } from './types';\nimport { verifySignature } from './verify-signature';\n\nexport type VerifiedAuthenticationResponse =\n | { verified: false; authenticationInfo?: never }\n | {\n verified: true;\n authenticationInfo: {\n credentialId: string;\n newCounter: number;\n userVerified: boolean;\n origin: string;\n rpID: string;\n };\n };\n\n/**\n * Verifies a WebAuthn authentication (assertion) response per\n * W3C WebAuthn Level 3 §7.2.\n *\n * Performs the following checks in order:\n * 1. Credential ID presence, base64url consistency, and type.\n * 2. `clientDataJSON` -- type is `\"webauthn.get\"`, challenge and origin\n * match.\n * 3. `authenticatorData` -- RP ID hash matches, user-presence flag is\n * set, and optional user-verification flag is checked.\n * 4. Signature verification -- `signature` is verified over\n * `authData || SHA-256(clientDataJSON)` using the stored credential\n * public key (COSE-encoded).\n * 5. Counter monotonicity -- if either the stored or returned counter\n * is non-zero, the new counter must exceed the stored value.\n *\n * @param opts - Verification options.\n * @param opts.response - The `PublicKeyCredential` result from\n * `navigator.credentials.get()`, serialized as JSON.\n * @param opts.expectedChallenge - The base64url challenge that was issued\n * for this ceremony.\n * @param opts.expectedOrigin - One or more acceptable origins.\n * @param opts.expectedRPID - The Relying Party ID domain.\n * @param opts.credential - The stored credential record to verify against.\n * @param opts.credential.id - The credential ID (base64url).\n * @param opts.credential.publicKey - The COSE-encoded public key bytes\n * persisted during registration.\n * @param opts.credential.counter - The last known signature counter value.\n * @param opts.credential.transports - Optional authenticator transports.\n * @param opts.requireUserVerification - When `true`, verification fails\n * if the UV flag is not set. Defaults to `false`.\n * @returns Verification result containing `verified` status and parsed\n * authentication info (new counter, origin, RP ID).\n */\nexport async function verifyAuthenticationResponse(opts: {\n response: PasskeyAuthenticationResponse;\n expectedChallenge: string;\n expectedOrigin: string | string[];\n expectedRPID: string;\n credential: {\n id: string;\n publicKey: Uint8Array;\n counter: number;\n transports?: AuthenticatorTransportFuture[];\n };\n requireUserVerification?: boolean;\n}): Promise<VerifiedAuthenticationResponse> {\n const {\n response,\n expectedChallenge,\n expectedOrigin,\n expectedRPID,\n credential,\n requireUserVerification = false,\n } = opts;\n\n const {\n id,\n rawId,\n type: credentialType,\n response: assertionResponse,\n } = response;\n\n // Ensure credential specified an ID\n if (!id) {\n throw new Error('Missing credential ID');\n }\n\n // Ensure ID is base64url-encoded\n if (id !== rawId) {\n throw new Error('Credential ID was not base64url-encoded');\n }\n\n // Make sure credential type is public-key\n if (credentialType !== 'public-key') {\n throw new Error(\n `Unexpected credential type ${String(credentialType)}, expected \"public-key\"`,\n );\n }\n\n if (typeof assertionResponse?.clientDataJSON !== 'string') {\n throw new Error('Credential response clientDataJSON was not a string');\n }\n\n const clientDataJSON = decodeClientDataJSON(assertionResponse.clientDataJSON);\n const { type, challenge, origin, tokenBinding } = clientDataJSON;\n\n // Make sure we're handling an authentication\n if (type !== 'webauthn.get') {\n throw new Error(`Unexpected authentication response type: ${type}`);\n }\n\n // Ensure the device provided the challenge we gave it\n if (challenge !== expectedChallenge) {\n throw new Error(\n `Unexpected authentication response challenge \"${challenge}\", expected \"${expectedChallenge}\"`,\n );\n }\n\n // Check that the origin is our site\n const expectedOrigins = Array.isArray(expectedOrigin)\n ? expectedOrigin\n : [expectedOrigin];\n if (!expectedOrigins.includes(origin)) {\n throw new Error(\n `Unexpected authentication response origin \"${origin}\", expected one of: ${expectedOrigins.join(', ')}`,\n );\n }\n\n if (\n assertionResponse.userHandle &&\n typeof assertionResponse.userHandle !== 'string'\n ) {\n throw new Error('Credential response userHandle was not a string');\n }\n\n if (tokenBinding) {\n if (typeof tokenBinding !== 'object') {\n throw new Error('ClientDataJSON tokenBinding was not an object');\n }\n\n if (\n !['present', 'supported', 'not-supported'].includes(tokenBinding.status)\n ) {\n throw new Error(`Unexpected tokenBinding status ${tokenBinding.status}`);\n }\n }\n\n const authDataBuffer = base64URLToBytes(assertionResponse.authenticatorData);\n const parsedAuthData: ParsedAuthenticatorData =\n parseAuthenticatorData(authDataBuffer);\n const { rpIdHash, flags, counter } = parsedAuthData;\n\n // Make sure the response's RP ID is ours\n const matchedRPID = matchExpectedRPID(rpIdHash, [expectedRPID]);\n\n // WebAuthn only requires the user presence flag be true\n if (!flags.up) {\n throw new Error('User not present during authentication');\n }\n\n // Enforce user verification if required\n if (requireUserVerification && !flags.uv) {\n throw new Error(\n 'User verification required, but user could not be verified',\n );\n }\n\n const clientDataHash = sha256(\n base64URLToBytes(assertionResponse.clientDataJSON),\n );\n const signatureBase = concatBytes([authDataBuffer, clientDataHash]);\n\n const signature = base64URLToBytes(assertionResponse.signature);\n\n const cosePublicKey = decodePartialCBOR(\n new Uint8Array(credential.publicKey),\n 0,\n )[0] as Map<number, number | Uint8Array>;\n\n const verified = await verifySignature({\n cosePublicKey,\n signature,\n data: signatureBase,\n });\n\n if (!verified) {\n return { verified: false };\n }\n\n if (\n (counter > 0 || credential.counter > 0) &&\n counter <= credential.counter\n ) {\n throw new Error(\n `Response counter value ${counter} was lower than expected ${credential.counter}`,\n );\n }\n\n return {\n verified: true,\n authenticationInfo: {\n credentialId: credential.id,\n newCounter: counter,\n userVerified: flags.uv,\n origin: clientDataJSON.origin,\n rpID: matchedRPID,\n },\n };\n}\n"]}
@@ -0,0 +1,205 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.verifyRegistrationResponse = void 0;
4
+ const tiny_cbor_1 = require("@levischuck/tiny-cbor");
5
+ const utils_1 = require("@metamask/utils");
6
+ const sha2_1 = require("@noble/hashes/sha2");
7
+ const encoding_1 = require("../utils/encoding.cjs");
8
+ const constants_1 = require("./constants.cjs");
9
+ const decode_attestation_object_1 = require("./decode-attestation-object.cjs");
10
+ const decode_client_data_json_1 = require("./decode-client-data-json.cjs");
11
+ const match_expected_rp_id_1 = require("./match-expected-rp-id.cjs");
12
+ const parse_authenticator_data_1 = require("./parse-authenticator-data.cjs");
13
+ const verify_signature_1 = require("./verify-signature.cjs");
14
+ /**
15
+ * Verifies a WebAuthn registration (attestation) response per
16
+ * W3C WebAuthn Level 3 §7.1.
17
+ *
18
+ * Performs the following checks in order:
19
+ * 1. Credential ID presence and base64url consistency (`id === rawId`), and
20
+ * that `id` matches the credential id inside parsed authenticator data.
21
+ * 2. Credential type is `"public-key"`.
22
+ * 3. `clientDataJSON` -- type is `"webauthn.create"`, challenge and origin
23
+ * match the expected values.
24
+ * 4. Attestation object -- CBOR-decodes and parses `authData` to verify
25
+ * the RP ID hash, user-presence flag, optional user-verification flag,
26
+ * and the attested credential public key algorithm.
27
+ * 5. Attestation statement -- supports `"none"` (no signature) and
28
+ * `"packed"` self-attestation (signature verified against the
29
+ * credential's own public key).
30
+ *
31
+ * @param opts - Verification options.
32
+ * @param opts.response - The `PublicKeyCredential` result from
33
+ * `navigator.credentials.create()`, serialized as JSON.
34
+ * @param opts.expectedChallenge - The base64url challenge that was passed
35
+ * to the authenticator (must match `clientDataJSON.challenge`).
36
+ * @param opts.expectedOrigin - One or more acceptable origins (e.g.
37
+ * `"chrome-extension://..."` or `"https://metamask.io"`).
38
+ * @param opts.expectedRPID - The Relying Party ID domain. The
39
+ * authenticator's `rpIdHash` is compared against `SHA-256(expectedRPID)`.
40
+ * @param opts.requireUserVerification - When `true`, verification fails
41
+ * if the UV flag is not set. Defaults to `false`.
42
+ * @param opts.supportedAlgorithmIDs - COSE algorithm identifiers accepted
43
+ * for the credential public key. Defaults to EdDSA, ES256, and RS256.
44
+ * @returns On success, `{ verified: true, registrationInfo }` with the
45
+ * parsed credential ID, public key, counter, AAGUID, and transport
46
+ * hints. On failure, `{ verified: false }`.
47
+ */
48
+ async function verifyRegistrationResponse(opts) {
49
+ const { response, expectedChallenge, expectedOrigin, expectedRPID, requireUserVerification = false, supportedAlgorithmIDs = [constants_1.COSEALG.EdDSA, constants_1.COSEALG.ES256, constants_1.COSEALG.RS256], } = opts;
50
+ const { id, rawId, type: credentialType, response: attestationResponse, } = response;
51
+ // Ensure credential specified an ID
52
+ if (!id) {
53
+ throw new Error('Missing credential ID');
54
+ }
55
+ // Ensure ID is base64url-encoded
56
+ if (id !== rawId) {
57
+ throw new Error('Credential ID was not base64url-encoded');
58
+ }
59
+ // Make sure credential type is public-key
60
+ if (credentialType !== 'public-key') {
61
+ throw new Error(`Unexpected credential type ${String(credentialType)}, expected "public-key"`);
62
+ }
63
+ const clientDataJSON = (0, decode_client_data_json_1.decodeClientDataJSON)(attestationResponse.clientDataJSON);
64
+ const { type, challenge, origin, tokenBinding } = clientDataJSON;
65
+ // Make sure we're handling an registration
66
+ if (type !== 'webauthn.create') {
67
+ throw new Error(`Unexpected registration response type: ${type}`);
68
+ }
69
+ // Ensure the device provided the challenge we gave it
70
+ if (challenge !== expectedChallenge) {
71
+ throw new Error(`Unexpected registration response challenge "${challenge}", expected "${expectedChallenge}"`);
72
+ }
73
+ // Check that the origin is our site
74
+ const expectedOrigins = Array.isArray(expectedOrigin)
75
+ ? expectedOrigin
76
+ : [expectedOrigin];
77
+ if (!expectedOrigins.includes(origin)) {
78
+ throw new Error(`Unexpected registration response origin "${origin}", expected one of: ${expectedOrigins.join(', ')}`);
79
+ }
80
+ if (tokenBinding) {
81
+ if (typeof tokenBinding !== 'object') {
82
+ throw new Error('ClientDataJSON tokenBinding was not an object');
83
+ }
84
+ if (!['present', 'supported', 'not-supported'].includes(tokenBinding.status)) {
85
+ throw new Error(`Unexpected tokenBinding.status value of "${tokenBinding.status}"`);
86
+ }
87
+ }
88
+ const attestationObjectBytes = (0, encoding_1.base64URLToBytes)(attestationResponse.attestationObject);
89
+ const decodedAttObj = (0, decode_attestation_object_1.decodeAttestationObject)(attestationObjectBytes);
90
+ const fmt = decodedAttObj.get('fmt');
91
+ const authData = decodedAttObj.get('authData');
92
+ const attStmt = decodedAttObj.get('attStmt');
93
+ const parsedAuthData = (0, parse_authenticator_data_1.parseAuthenticatorData)(authData);
94
+ const { rpIdHash, flags, counter, credentialID, credentialPublicKey, aaguid, } = parsedAuthData;
95
+ (0, match_expected_rp_id_1.matchExpectedRPID)(rpIdHash, [expectedRPID]);
96
+ // Make sure someone was physically present
97
+ if (!flags.up) {
98
+ throw new Error('User presence was required, but user was not present');
99
+ }
100
+ // Enforce user verification if specified
101
+ if (requireUserVerification && !flags.uv) {
102
+ throw new Error('User verification was required, but user could not be verified');
103
+ }
104
+ if (!credentialID) {
105
+ throw new Error('No credential ID was provided by authenticator');
106
+ }
107
+ const attestedCredentialId = (0, encoding_1.bytesToBase64URL)(credentialID);
108
+ if (id !== attestedCredentialId) {
109
+ throw new Error('Credential id does not match the credential id in authenticator data');
110
+ }
111
+ if (!credentialPublicKey) {
112
+ throw new Error('No public key was provided by authenticator');
113
+ }
114
+ if (!aaguid) {
115
+ throw new Error('No AAGUID was present during registration');
116
+ }
117
+ const decodedPublicKey = (0, tiny_cbor_1.decodePartialCBOR)(new Uint8Array(credentialPublicKey), 0)[0];
118
+ const alg = decodedPublicKey.get(constants_1.COSEKEYS.Alg);
119
+ if (typeof alg !== 'number') {
120
+ throw new Error('Credential public key was missing numeric alg');
121
+ }
122
+ // Make sure the key algorithm is one we specified within the registration options
123
+ if (!supportedAlgorithmIDs.includes(alg)) {
124
+ throw new Error(`Unexpected public key alg "${alg}", expected one of "${supportedAlgorithmIDs.join(', ')}"`);
125
+ }
126
+ let verified = false;
127
+ if (fmt === 'none') {
128
+ if (attStmt.size > 0) {
129
+ throw new Error('None attestation had unexpected attestation statement');
130
+ }
131
+ verified = true;
132
+ }
133
+ else if (fmt === 'packed') {
134
+ verified = await verifyPackedAttestation(attStmt, authData, attestationResponse.clientDataJSON, decodedPublicKey);
135
+ }
136
+ else {
137
+ throw new Error(`Unsupported attestation format: ${fmt}`);
138
+ }
139
+ if (!verified) {
140
+ return { verified: false };
141
+ }
142
+ const aaguidHex = (0, encoding_1.bytesToHex)(aaguid);
143
+ const aaguidStr = [
144
+ aaguidHex.slice(0, 8),
145
+ aaguidHex.slice(8, 12),
146
+ aaguidHex.slice(12, 16),
147
+ aaguidHex.slice(16, 20),
148
+ aaguidHex.slice(20),
149
+ ].join('-');
150
+ return {
151
+ verified: true,
152
+ registrationInfo: {
153
+ credentialId: attestedCredentialId,
154
+ publicKey: credentialPublicKey,
155
+ counter,
156
+ transports: attestationResponse.transports,
157
+ aaguid: aaguidStr,
158
+ attestationFormat: fmt,
159
+ userVerified: flags.uv,
160
+ },
161
+ };
162
+ }
163
+ exports.verifyRegistrationResponse = verifyRegistrationResponse;
164
+ /**
165
+ * Verify packed self-attestation per WebAuthn §8.2: no x5c certificate
166
+ * chain, signature over `authData || SHA-256(clientDataJSON)` verified
167
+ * with the credential's own public key, and `alg` in the attestation
168
+ * statement must match the credential key's algorithm.
169
+ *
170
+ * @param attStmt - The attestation statement map from the attestation
171
+ * object.
172
+ * @param attStmt.get - Accessor to retrieve statement fields by key.
173
+ * @param attStmt.size - Number of entries in the statement.
174
+ * @param authData - Raw authenticator data bytes.
175
+ * @param clientDataJSONB64url - Base64url-encoded clientDataJSON.
176
+ * @param cosePublicKey - Decoded COSE public key map from authenticator
177
+ * data.
178
+ * @returns Whether the packed attestation signature is valid.
179
+ */
180
+ async function verifyPackedAttestation(attStmt, authData, clientDataJSONB64url, cosePublicKey) {
181
+ const attStmtAlg = attStmt.get('alg');
182
+ const signature = attStmt.get('sig');
183
+ const x5c = attStmt.get('x5c');
184
+ if (typeof attStmtAlg !== 'number') {
185
+ throw new Error('Packed attestation statement missing alg');
186
+ }
187
+ if (!signature) {
188
+ throw new Error('Packed attestation missing signature');
189
+ }
190
+ if (x5c && x5c.length > 0) {
191
+ throw new Error('Packed attestation with certificate chain (x5c) is not supported; only self-attestation is accepted');
192
+ }
193
+ const credAlg = cosePublicKey.get(constants_1.COSEKEYS.Alg);
194
+ if (attStmtAlg !== credAlg) {
195
+ throw new Error(`Packed attestation alg ${attStmtAlg} does not match credential alg ${credAlg}`);
196
+ }
197
+ const clientDataHash = (0, sha2_1.sha256)((0, encoding_1.base64URLToBytes)(clientDataJSONB64url));
198
+ const signatureBase = (0, utils_1.concatBytes)([authData, clientDataHash]);
199
+ return (0, verify_signature_1.verifySignature)({
200
+ cosePublicKey,
201
+ signature,
202
+ data: signatureBase,
203
+ });
204
+ }
205
+ //# sourceMappingURL=verify-registration-response.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify-registration-response.cjs","sourceRoot":"","sources":["../../src/webauthn/verify-registration-response.ts"],"names":[],"mappings":";;;AAAA,qDAA0D;AAC1D,2CAA8C;AAC9C,6CAA4C;AAG5C,oDAI2B;AAC3B,+CAAgD;AAChD,+EAAsE;AACtE,2EAAiE;AACjE,qEAA2D;AAC3D,6EAAoE;AAEpE,6DAAqD;AAiBrD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiCG;AACI,KAAK,UAAU,0BAA0B,CAAC,IAOhD;IACC,MAAM,EACJ,QAAQ,EACR,iBAAiB,EACjB,cAAc,EACd,YAAY,EACZ,uBAAuB,GAAG,KAAK,EAC/B,qBAAqB,GAAG,CAAC,mBAAO,CAAC,KAAK,EAAE,mBAAO,CAAC,KAAK,EAAE,mBAAO,CAAC,KAAK,CAAC,GACtE,GAAG,IAAI,CAAC;IAET,MAAM,EACJ,EAAE,EACF,KAAK,EACL,IAAI,EAAE,cAAc,EACpB,QAAQ,EAAE,mBAAmB,GAC9B,GAAG,QAAQ,CAAC;IAEb,oCAAoC;IACpC,IAAI,CAAC,EAAE,EAAE,CAAC;QACR,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;IAC3C,CAAC;IAED,iCAAiC;IACjC,IAAI,EAAE,KAAK,KAAK,EAAE,CAAC;QACjB,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;IAC7D,CAAC;IAED,0CAA0C;IAC1C,IAAI,cAAc,KAAK,YAAY,EAAE,CAAC;QACpC,MAAM,IAAI,KAAK,CACb,8BAA8B,MAAM,CAAC,cAAc,CAAC,yBAAyB,CAC9E,CAAC;IACJ,CAAC;IAED,MAAM,cAAc,GAAG,IAAA,8CAAoB,EACzC,mBAAmB,CAAC,cAAc,CACnC,CAAC;IACF,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,YAAY,EAAE,GAAG,cAAc,CAAC;IAEjE,2CAA2C;IAC3C,IAAI,IAAI,KAAK,iBAAiB,EAAE,CAAC;QAC/B,MAAM,IAAI,KAAK,CAAC,0CAA0C,IAAI,EAAE,CAAC,CAAC;IACpE,CAAC;IAED,sDAAsD;IACtD,IAAI,SAAS,KAAK,iBAAiB,EAAE,CAAC;QACpC,MAAM,IAAI,KAAK,CACb,+CAA+C,SAAS,gBAAgB,iBAAiB,GAAG,CAC7F,CAAC;IACJ,CAAC;IAED,oCAAoC;IACpC,MAAM,eAAe,GAAG,KAAK,CAAC,OAAO,CAAC,cAAc,CAAC;QACnD,CAAC,CAAC,cAAc;QAChB,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC;IACrB,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;QACtC,MAAM,IAAI,KAAK,CACb,4CAA4C,MAAM,uBAAuB,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACtG,CAAC;IACJ,CAAC;IAED,IAAI,YAAY,EAAE,CAAC;QACjB,IAAI,OAAO,YAAY,KAAK,QAAQ,EAAE,CAAC;YACrC,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;QACnE,CAAC;QAED,IACE,CAAC,CAAC,SAAS,EAAE,WAAW,EAAE,eAAe,CAAC,CAAC,QAAQ,CAAC,YAAY,CAAC,MAAM,CAAC,EACxE,CAAC;YACD,MAAM,IAAI,KAAK,CACb,4CAA4C,YAAY,CAAC,MAAM,GAAG,CACnE,CAAC;QACJ,CAAC;IACH,CAAC;IAED,MAAM,sBAAsB,GAAG,IAAA,2BAAgB,EAC7C,mBAAmB,CAAC,iBAAiB,CACtC,CAAC;IACF,MAAM,aAAa,GAAG,IAAA,mDAAuB,EAAC,sBAAsB,CAAC,CAAC;IACtE,MAAM,GAAG,GAAG,aAAa,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACrC,MAAM,QAAQ,GAAG,aAAa,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;IAC/C,MAAM,OAAO,GAAG,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAE7C,MAAM,cAAc,GAAG,IAAA,iDAAsB,EAAC,QAAQ,CAAC,CAAC;IACxD,MAAM,EACJ,QAAQ,EACR,KAAK,EACL,OAAO,EACP,YAAY,EACZ,mBAAmB,EACnB,MAAM,GACP,GAAG,cAAc,CAAC;IAEnB,IAAA,wCAAiB,EAAC,QAAQ,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC;IAE5C,2CAA2C;IAC3C,IAAI,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC;QACd,MAAM,IAAI,KAAK,CAAC,sDAAsD,CAAC,CAAC;IAC1E,CAAC;IAED,yCAAyC;IACzC,IAAI,uBAAuB,IAAI,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC;QACzC,MAAM,IAAI,KAAK,CACb,gEAAgE,CACjE,CAAC;IACJ,CAAC;IAED,IAAI,CAAC,YAAY,EAAE,CAAC;QAClB,MAAM,IAAI,KAAK,CAAC,gDAAgD,CAAC,CAAC;IACpE,CAAC;IAED,MAAM,oBAAoB,GAAG,IAAA,2BAAgB,EAAC,YAAY,CAAC,CAAC;IAC5D,IAAI,EAAE,KAAK,oBAAoB,EAAE,CAAC;QAChC,MAAM,IAAI,KAAK,CACb,sEAAsE,CACvE,CAAC;IACJ,CAAC;IAED,IAAI,CAAC,mBAAmB,EAAE,CAAC;QACzB,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;IACjE,CAAC;IACD,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CAAC,2CAA2C,CAAC,CAAC;IAC/D,CAAC;IAED,MAAM,gBAAgB,GAAG,IAAA,6BAAiB,EACxC,IAAI,UAAU,CAAC,mBAAmB,CAAC,EACnC,CAAC,CACF,CAAC,CAAC,CAAqC,CAAC;IACzC,MAAM,GAAG,GAAG,gBAAgB,CAAC,GAAG,CAAC,oBAAQ,CAAC,GAAG,CAAC,CAAC;IAE/C,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;QAC5B,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;IACnE,CAAC;IAED,kFAAkF;IAClF,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;QACzC,MAAM,IAAI,KAAK,CACb,8BAA8B,GAAG,uBAAuB,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAC5F,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,GAAG,KAAK,CAAC;IACrB,IAAI,GAAG,KAAK,MAAM,EAAE,CAAC;QACnB,IAAI,OAAO,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;YACrB,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;QAC3E,CAAC;QACD,QAAQ,GAAG,IAAI,CAAC;IAClB,CAAC;SAAM,IAAI,GAAG,KAAK,QAAQ,EAAE,CAAC;QAC5B,QAAQ,GAAG,MAAM,uBAAuB,CACtC,OAAO,EACP,QAAQ,EACR,mBAAmB,CAAC,cAAc,EAClC,gBAAgB,CACjB,CAAC;IACJ,CAAC;SAAM,CAAC;QACN,MAAM,IAAI,KAAK,CAAC,mCAAmC,GAAG,EAAE,CAAC,CAAC;IAC5D,CAAC;IAED,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;IAC7B,CAAC;IAED,MAAM,SAAS,GAAG,IAAA,qBAAU,EAAC,MAAM,CAAC,CAAC;IACrC,MAAM,SAAS,GAAG;QAChB,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;QACrB,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;QACtB,SAAS,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,CAAC;QACvB,SAAS,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,CAAC;QACvB,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC;KACpB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAEZ,OAAO;QACL,QAAQ,EAAE,IAAI;QACd,gBAAgB,EAAE;YAChB,YAAY,EAAE,oBAAoB;YAClC,SAAS,EAAE,mBAAmB;YAC9B,OAAO;YACP,UAAU,EACR,mBAAmB,CAAC,UAA4C;YAClE,MAAM,EAAE,SAAS;YACjB,iBAAiB,EAAE,GAAG;YACtB,YAAY,EAAE,KAAK,CAAC,EAAE;SACvB;KACF,CAAC;AACJ,CAAC;AAhMD,gEAgMC;AAED;;;;;;;;;;;;;;;GAeG;AACH,KAAK,UAAU,uBAAuB,CACpC,OAAoD,EACpD,QAAoB,EACpB,oBAA4B,EAC5B,aAA+C;IAE/C,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,KAAK,CAAuB,CAAC;IAC5D,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,KAAK,CAA2B,CAAC;IAC/D,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,KAAK,CAA6B,CAAC;IAE3D,IAAI,OAAO,UAAU,KAAK,QAAQ,EAAE,CAAC;QACnC,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;IAC9D,CAAC;IAED,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,MAAM,IAAI,KAAK,CAAC,sCAAsC,CAAC,CAAC;IAC1D,CAAC;IAED,IAAI,GAAG,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC1B,MAAM,IAAI,KAAK,CACb,qGAAqG,CACtG,CAAC;IACJ,CAAC;IAED,MAAM,OAAO,GAAG,aAAa,CAAC,GAAG,CAAC,oBAAQ,CAAC,GAAG,CAAW,CAAC;IAC1D,IAAI,UAAU,KAAK,OAAO,EAAE,CAAC;QAC3B,MAAM,IAAI,KAAK,CACb,0BAA0B,UAAU,kCAAkC,OAAO,EAAE,CAChF,CAAC;IACJ,CAAC;IAED,MAAM,cAAc,GAAG,IAAA,aAAM,EAAC,IAAA,2BAAgB,EAAC,oBAAoB,CAAC,CAAC,CAAC;IACtE,MAAM,aAAa,GAAG,IAAA,mBAAW,EAAC,CAAC,QAAQ,EAAE,cAAc,CAAC,CAAC,CAAC;IAE9D,OAAO,IAAA,kCAAe,EAAC;QACrB,aAAa;QACb,SAAS;QACT,IAAI,EAAE,aAAa;KACpB,CAAC,CAAC;AACL,CAAC","sourcesContent":["import { decodePartialCBOR } from '@levischuck/tiny-cbor';\nimport { concatBytes } from '@metamask/utils';\nimport { sha256 } from '@noble/hashes/sha2';\n\nimport type { AuthenticatorTransportFuture } from '../types';\nimport {\n base64URLToBytes,\n bytesToBase64URL,\n bytesToHex,\n} from '../utils/encoding';\nimport { COSEALG, COSEKEYS } from './constants';\nimport { decodeAttestationObject } from './decode-attestation-object';\nimport { decodeClientDataJSON } from './decode-client-data-json';\nimport { matchExpectedRPID } from './match-expected-rp-id';\nimport { parseAuthenticatorData } from './parse-authenticator-data';\nimport type { PasskeyRegistrationResponse } from './types';\nimport { verifySignature } from './verify-signature';\n\nexport type VerifiedRegistrationResponse =\n | { verified: false; registrationInfo?: never }\n | {\n verified: true;\n registrationInfo: {\n credentialId: string;\n publicKey: Uint8Array;\n counter: number;\n transports?: AuthenticatorTransportFuture[];\n aaguid: string;\n attestationFormat: string;\n userVerified: boolean;\n };\n };\n\n/**\n * Verifies a WebAuthn registration (attestation) response per\n * W3C WebAuthn Level 3 §7.1.\n *\n * Performs the following checks in order:\n * 1. Credential ID presence and base64url consistency (`id === rawId`), and\n * that `id` matches the credential id inside parsed authenticator data.\n * 2. Credential type is `\"public-key\"`.\n * 3. `clientDataJSON` -- type is `\"webauthn.create\"`, challenge and origin\n * match the expected values.\n * 4. Attestation object -- CBOR-decodes and parses `authData` to verify\n * the RP ID hash, user-presence flag, optional user-verification flag,\n * and the attested credential public key algorithm.\n * 5. Attestation statement -- supports `\"none\"` (no signature) and\n * `\"packed\"` self-attestation (signature verified against the\n * credential's own public key).\n *\n * @param opts - Verification options.\n * @param opts.response - The `PublicKeyCredential` result from\n * `navigator.credentials.create()`, serialized as JSON.\n * @param opts.expectedChallenge - The base64url challenge that was passed\n * to the authenticator (must match `clientDataJSON.challenge`).\n * @param opts.expectedOrigin - One or more acceptable origins (e.g.\n * `\"chrome-extension://...\"` or `\"https://metamask.io\"`).\n * @param opts.expectedRPID - The Relying Party ID domain. The\n * authenticator's `rpIdHash` is compared against `SHA-256(expectedRPID)`.\n * @param opts.requireUserVerification - When `true`, verification fails\n * if the UV flag is not set. Defaults to `false`.\n * @param opts.supportedAlgorithmIDs - COSE algorithm identifiers accepted\n * for the credential public key. Defaults to EdDSA, ES256, and RS256.\n * @returns On success, `{ verified: true, registrationInfo }` with the\n * parsed credential ID, public key, counter, AAGUID, and transport\n * hints. On failure, `{ verified: false }`.\n */\nexport async function verifyRegistrationResponse(opts: {\n response: PasskeyRegistrationResponse;\n expectedChallenge: string;\n expectedOrigin: string | string[];\n expectedRPID: string;\n requireUserVerification?: boolean;\n supportedAlgorithmIDs?: number[];\n}): Promise<VerifiedRegistrationResponse> {\n const {\n response,\n expectedChallenge,\n expectedOrigin,\n expectedRPID,\n requireUserVerification = false,\n supportedAlgorithmIDs = [COSEALG.EdDSA, COSEALG.ES256, COSEALG.RS256],\n } = opts;\n\n const {\n id,\n rawId,\n type: credentialType,\n response: attestationResponse,\n } = response;\n\n // Ensure credential specified an ID\n if (!id) {\n throw new Error('Missing credential ID');\n }\n\n // Ensure ID is base64url-encoded\n if (id !== rawId) {\n throw new Error('Credential ID was not base64url-encoded');\n }\n\n // Make sure credential type is public-key\n if (credentialType !== 'public-key') {\n throw new Error(\n `Unexpected credential type ${String(credentialType)}, expected \"public-key\"`,\n );\n }\n\n const clientDataJSON = decodeClientDataJSON(\n attestationResponse.clientDataJSON,\n );\n const { type, challenge, origin, tokenBinding } = clientDataJSON;\n\n // Make sure we're handling an registration\n if (type !== 'webauthn.create') {\n throw new Error(`Unexpected registration response type: ${type}`);\n }\n\n // Ensure the device provided the challenge we gave it\n if (challenge !== expectedChallenge) {\n throw new Error(\n `Unexpected registration response challenge \"${challenge}\", expected \"${expectedChallenge}\"`,\n );\n }\n\n // Check that the origin is our site\n const expectedOrigins = Array.isArray(expectedOrigin)\n ? expectedOrigin\n : [expectedOrigin];\n if (!expectedOrigins.includes(origin)) {\n throw new Error(\n `Unexpected registration response origin \"${origin}\", expected one of: ${expectedOrigins.join(', ')}`,\n );\n }\n\n if (tokenBinding) {\n if (typeof tokenBinding !== 'object') {\n throw new Error('ClientDataJSON tokenBinding was not an object');\n }\n\n if (\n !['present', 'supported', 'not-supported'].includes(tokenBinding.status)\n ) {\n throw new Error(\n `Unexpected tokenBinding.status value of \"${tokenBinding.status}\"`,\n );\n }\n }\n\n const attestationObjectBytes = base64URLToBytes(\n attestationResponse.attestationObject,\n );\n const decodedAttObj = decodeAttestationObject(attestationObjectBytes);\n const fmt = decodedAttObj.get('fmt');\n const authData = decodedAttObj.get('authData');\n const attStmt = decodedAttObj.get('attStmt');\n\n const parsedAuthData = parseAuthenticatorData(authData);\n const {\n rpIdHash,\n flags,\n counter,\n credentialID,\n credentialPublicKey,\n aaguid,\n } = parsedAuthData;\n\n matchExpectedRPID(rpIdHash, [expectedRPID]);\n\n // Make sure someone was physically present\n if (!flags.up) {\n throw new Error('User presence was required, but user was not present');\n }\n\n // Enforce user verification if specified\n if (requireUserVerification && !flags.uv) {\n throw new Error(\n 'User verification was required, but user could not be verified',\n );\n }\n\n if (!credentialID) {\n throw new Error('No credential ID was provided by authenticator');\n }\n\n const attestedCredentialId = bytesToBase64URL(credentialID);\n if (id !== attestedCredentialId) {\n throw new Error(\n 'Credential id does not match the credential id in authenticator data',\n );\n }\n\n if (!credentialPublicKey) {\n throw new Error('No public key was provided by authenticator');\n }\n if (!aaguid) {\n throw new Error('No AAGUID was present during registration');\n }\n\n const decodedPublicKey = decodePartialCBOR(\n new Uint8Array(credentialPublicKey),\n 0,\n )[0] as Map<number, number | Uint8Array>;\n const alg = decodedPublicKey.get(COSEKEYS.Alg);\n\n if (typeof alg !== 'number') {\n throw new Error('Credential public key was missing numeric alg');\n }\n\n // Make sure the key algorithm is one we specified within the registration options\n if (!supportedAlgorithmIDs.includes(alg)) {\n throw new Error(\n `Unexpected public key alg \"${alg}\", expected one of \"${supportedAlgorithmIDs.join(', ')}\"`,\n );\n }\n\n let verified = false;\n if (fmt === 'none') {\n if (attStmt.size > 0) {\n throw new Error('None attestation had unexpected attestation statement');\n }\n verified = true;\n } else if (fmt === 'packed') {\n verified = await verifyPackedAttestation(\n attStmt,\n authData,\n attestationResponse.clientDataJSON,\n decodedPublicKey,\n );\n } else {\n throw new Error(`Unsupported attestation format: ${fmt}`);\n }\n\n if (!verified) {\n return { verified: false };\n }\n\n const aaguidHex = bytesToHex(aaguid);\n const aaguidStr = [\n aaguidHex.slice(0, 8),\n aaguidHex.slice(8, 12),\n aaguidHex.slice(12, 16),\n aaguidHex.slice(16, 20),\n aaguidHex.slice(20),\n ].join('-');\n\n return {\n verified: true,\n registrationInfo: {\n credentialId: attestedCredentialId,\n publicKey: credentialPublicKey,\n counter,\n transports:\n attestationResponse.transports as AuthenticatorTransportFuture[],\n aaguid: aaguidStr,\n attestationFormat: fmt,\n userVerified: flags.uv,\n },\n };\n}\n\n/**\n * Verify packed self-attestation per WebAuthn §8.2: no x5c certificate\n * chain, signature over `authData || SHA-256(clientDataJSON)` verified\n * with the credential's own public key, and `alg` in the attestation\n * statement must match the credential key's algorithm.\n *\n * @param attStmt - The attestation statement map from the attestation\n * object.\n * @param attStmt.get - Accessor to retrieve statement fields by key.\n * @param attStmt.size - Number of entries in the statement.\n * @param authData - Raw authenticator data bytes.\n * @param clientDataJSONB64url - Base64url-encoded clientDataJSON.\n * @param cosePublicKey - Decoded COSE public key map from authenticator\n * data.\n * @returns Whether the packed attestation signature is valid.\n */\nasync function verifyPackedAttestation(\n attStmt: { get(key: string): unknown; size: number },\n authData: Uint8Array,\n clientDataJSONB64url: string,\n cosePublicKey: Map<number, number | Uint8Array>,\n): Promise<boolean> {\n const attStmtAlg = attStmt.get('alg') as number | undefined;\n const signature = attStmt.get('sig') as Uint8Array | undefined;\n const x5c = attStmt.get('x5c') as Uint8Array[] | undefined;\n\n if (typeof attStmtAlg !== 'number') {\n throw new Error('Packed attestation statement missing alg');\n }\n\n if (!signature) {\n throw new Error('Packed attestation missing signature');\n }\n\n if (x5c && x5c.length > 0) {\n throw new Error(\n 'Packed attestation with certificate chain (x5c) is not supported; only self-attestation is accepted',\n );\n }\n\n const credAlg = cosePublicKey.get(COSEKEYS.Alg) as number;\n if (attStmtAlg !== credAlg) {\n throw new Error(\n `Packed attestation alg ${attStmtAlg} does not match credential alg ${credAlg}`,\n );\n }\n\n const clientDataHash = sha256(base64URLToBytes(clientDataJSONB64url));\n const signatureBase = concatBytes([authData, clientDataHash]);\n\n return verifySignature({\n cosePublicKey,\n signature,\n data: signatureBase,\n });\n}\n"]}
@@ -0,0 +1,60 @@
1
+ import type { AuthenticatorTransportFuture } from "../types.cjs";
2
+ import type { PasskeyRegistrationResponse } from "./types.cjs";
3
+ export type VerifiedRegistrationResponse = {
4
+ verified: false;
5
+ registrationInfo?: never;
6
+ } | {
7
+ verified: true;
8
+ registrationInfo: {
9
+ credentialId: string;
10
+ publicKey: Uint8Array;
11
+ counter: number;
12
+ transports?: AuthenticatorTransportFuture[];
13
+ aaguid: string;
14
+ attestationFormat: string;
15
+ userVerified: boolean;
16
+ };
17
+ };
18
+ /**
19
+ * Verifies a WebAuthn registration (attestation) response per
20
+ * W3C WebAuthn Level 3 §7.1.
21
+ *
22
+ * Performs the following checks in order:
23
+ * 1. Credential ID presence and base64url consistency (`id === rawId`), and
24
+ * that `id` matches the credential id inside parsed authenticator data.
25
+ * 2. Credential type is `"public-key"`.
26
+ * 3. `clientDataJSON` -- type is `"webauthn.create"`, challenge and origin
27
+ * match the expected values.
28
+ * 4. Attestation object -- CBOR-decodes and parses `authData` to verify
29
+ * the RP ID hash, user-presence flag, optional user-verification flag,
30
+ * and the attested credential public key algorithm.
31
+ * 5. Attestation statement -- supports `"none"` (no signature) and
32
+ * `"packed"` self-attestation (signature verified against the
33
+ * credential's own public key).
34
+ *
35
+ * @param opts - Verification options.
36
+ * @param opts.response - The `PublicKeyCredential` result from
37
+ * `navigator.credentials.create()`, serialized as JSON.
38
+ * @param opts.expectedChallenge - The base64url challenge that was passed
39
+ * to the authenticator (must match `clientDataJSON.challenge`).
40
+ * @param opts.expectedOrigin - One or more acceptable origins (e.g.
41
+ * `"chrome-extension://..."` or `"https://metamask.io"`).
42
+ * @param opts.expectedRPID - The Relying Party ID domain. The
43
+ * authenticator's `rpIdHash` is compared against `SHA-256(expectedRPID)`.
44
+ * @param opts.requireUserVerification - When `true`, verification fails
45
+ * if the UV flag is not set. Defaults to `false`.
46
+ * @param opts.supportedAlgorithmIDs - COSE algorithm identifiers accepted
47
+ * for the credential public key. Defaults to EdDSA, ES256, and RS256.
48
+ * @returns On success, `{ verified: true, registrationInfo }` with the
49
+ * parsed credential ID, public key, counter, AAGUID, and transport
50
+ * hints. On failure, `{ verified: false }`.
51
+ */
52
+ export declare function verifyRegistrationResponse(opts: {
53
+ response: PasskeyRegistrationResponse;
54
+ expectedChallenge: string;
55
+ expectedOrigin: string | string[];
56
+ expectedRPID: string;
57
+ requireUserVerification?: boolean;
58
+ supportedAlgorithmIDs?: number[];
59
+ }): Promise<VerifiedRegistrationResponse>;
60
+ //# sourceMappingURL=verify-registration-response.d.cts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify-registration-response.d.cts","sourceRoot":"","sources":["../../src/webauthn/verify-registration-response.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,4BAA4B,EAAE,qBAAiB;AAW7D,OAAO,KAAK,EAAE,2BAA2B,EAAE,oBAAgB;AAG3D,MAAM,MAAM,4BAA4B,GACpC;IAAE,QAAQ,EAAE,KAAK,CAAC;IAAC,gBAAgB,CAAC,EAAE,KAAK,CAAA;CAAE,GAC7C;IACE,QAAQ,EAAE,IAAI,CAAC;IACf,gBAAgB,EAAE;QAChB,YAAY,EAAE,MAAM,CAAC;QACrB,SAAS,EAAE,UAAU,CAAC;QACtB,OAAO,EAAE,MAAM,CAAC;QAChB,UAAU,CAAC,EAAE,4BAA4B,EAAE,CAAC;QAC5C,MAAM,EAAE,MAAM,CAAC;QACf,iBAAiB,EAAE,MAAM,CAAC;QAC1B,YAAY,EAAE,OAAO,CAAC;KACvB,CAAC;CACH,CAAC;AAEN;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiCG;AACH,wBAAsB,0BAA0B,CAAC,IAAI,EAAE;IACrD,QAAQ,EAAE,2BAA2B,CAAC;IACtC,iBAAiB,EAAE,MAAM,CAAC;IAC1B,cAAc,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;IAClC,YAAY,EAAE,MAAM,CAAC;IACrB,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAClC,qBAAqB,CAAC,EAAE,MAAM,EAAE,CAAC;CAClC,GAAG,OAAO,CAAC,4BAA4B,CAAC,CAyLxC"}
@@ -0,0 +1,60 @@
1
+ import type { AuthenticatorTransportFuture } from "../types.mjs";
2
+ import type { PasskeyRegistrationResponse } from "./types.mjs";
3
+ export type VerifiedRegistrationResponse = {
4
+ verified: false;
5
+ registrationInfo?: never;
6
+ } | {
7
+ verified: true;
8
+ registrationInfo: {
9
+ credentialId: string;
10
+ publicKey: Uint8Array;
11
+ counter: number;
12
+ transports?: AuthenticatorTransportFuture[];
13
+ aaguid: string;
14
+ attestationFormat: string;
15
+ userVerified: boolean;
16
+ };
17
+ };
18
+ /**
19
+ * Verifies a WebAuthn registration (attestation) response per
20
+ * W3C WebAuthn Level 3 §7.1.
21
+ *
22
+ * Performs the following checks in order:
23
+ * 1. Credential ID presence and base64url consistency (`id === rawId`), and
24
+ * that `id` matches the credential id inside parsed authenticator data.
25
+ * 2. Credential type is `"public-key"`.
26
+ * 3. `clientDataJSON` -- type is `"webauthn.create"`, challenge and origin
27
+ * match the expected values.
28
+ * 4. Attestation object -- CBOR-decodes and parses `authData` to verify
29
+ * the RP ID hash, user-presence flag, optional user-verification flag,
30
+ * and the attested credential public key algorithm.
31
+ * 5. Attestation statement -- supports `"none"` (no signature) and
32
+ * `"packed"` self-attestation (signature verified against the
33
+ * credential's own public key).
34
+ *
35
+ * @param opts - Verification options.
36
+ * @param opts.response - The `PublicKeyCredential` result from
37
+ * `navigator.credentials.create()`, serialized as JSON.
38
+ * @param opts.expectedChallenge - The base64url challenge that was passed
39
+ * to the authenticator (must match `clientDataJSON.challenge`).
40
+ * @param opts.expectedOrigin - One or more acceptable origins (e.g.
41
+ * `"chrome-extension://..."` or `"https://metamask.io"`).
42
+ * @param opts.expectedRPID - The Relying Party ID domain. The
43
+ * authenticator's `rpIdHash` is compared against `SHA-256(expectedRPID)`.
44
+ * @param opts.requireUserVerification - When `true`, verification fails
45
+ * if the UV flag is not set. Defaults to `false`.
46
+ * @param opts.supportedAlgorithmIDs - COSE algorithm identifiers accepted
47
+ * for the credential public key. Defaults to EdDSA, ES256, and RS256.
48
+ * @returns On success, `{ verified: true, registrationInfo }` with the
49
+ * parsed credential ID, public key, counter, AAGUID, and transport
50
+ * hints. On failure, `{ verified: false }`.
51
+ */
52
+ export declare function verifyRegistrationResponse(opts: {
53
+ response: PasskeyRegistrationResponse;
54
+ expectedChallenge: string;
55
+ expectedOrigin: string | string[];
56
+ expectedRPID: string;
57
+ requireUserVerification?: boolean;
58
+ supportedAlgorithmIDs?: number[];
59
+ }): Promise<VerifiedRegistrationResponse>;
60
+ //# sourceMappingURL=verify-registration-response.d.mts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify-registration-response.d.mts","sourceRoot":"","sources":["../../src/webauthn/verify-registration-response.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,4BAA4B,EAAE,qBAAiB;AAW7D,OAAO,KAAK,EAAE,2BAA2B,EAAE,oBAAgB;AAG3D,MAAM,MAAM,4BAA4B,GACpC;IAAE,QAAQ,EAAE,KAAK,CAAC;IAAC,gBAAgB,CAAC,EAAE,KAAK,CAAA;CAAE,GAC7C;IACE,QAAQ,EAAE,IAAI,CAAC;IACf,gBAAgB,EAAE;QAChB,YAAY,EAAE,MAAM,CAAC;QACrB,SAAS,EAAE,UAAU,CAAC;QACtB,OAAO,EAAE,MAAM,CAAC;QAChB,UAAU,CAAC,EAAE,4BAA4B,EAAE,CAAC;QAC5C,MAAM,EAAE,MAAM,CAAC;QACf,iBAAiB,EAAE,MAAM,CAAC;QAC1B,YAAY,EAAE,OAAO,CAAC;KACvB,CAAC;CACH,CAAC;AAEN;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiCG;AACH,wBAAsB,0BAA0B,CAAC,IAAI,EAAE;IACrD,QAAQ,EAAE,2BAA2B,CAAC;IACtC,iBAAiB,EAAE,MAAM,CAAC;IAC1B,cAAc,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;IAClC,YAAY,EAAE,MAAM,CAAC;IACrB,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAClC,qBAAqB,CAAC,EAAE,MAAM,EAAE,CAAC;CAClC,GAAG,OAAO,CAAC,4BAA4B,CAAC,CAyLxC"}