@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.
- package/CHANGELOG.md +24 -0
- package/LICENSE +21 -0
- package/README.md +155 -0
- package/dist/PasskeyController.cjs +448 -0
- package/dist/PasskeyController.cjs.map +1 -0
- package/dist/PasskeyController.d.cts +168 -0
- package/dist/PasskeyController.d.cts.map +1 -0
- package/dist/PasskeyController.d.mts +168 -0
- package/dist/PasskeyController.d.mts.map +1 -0
- package/dist/PasskeyController.mjs +443 -0
- package/dist/PasskeyController.mjs.map +1 -0
- package/dist/ceremony-manager.cjs +134 -0
- package/dist/ceremony-manager.cjs.map +1 -0
- package/dist/ceremony-manager.d.cts +71 -0
- package/dist/ceremony-manager.d.cts.map +1 -0
- package/dist/ceremony-manager.d.mts +71 -0
- package/dist/ceremony-manager.d.mts.map +1 -0
- package/dist/ceremony-manager.mjs +130 -0
- package/dist/ceremony-manager.mjs.map +1 -0
- package/dist/constants.cjs +33 -0
- package/dist/constants.cjs.map +1 -0
- package/dist/constants.d.cts +30 -0
- package/dist/constants.d.cts.map +1 -0
- package/dist/constants.d.mts +30 -0
- package/dist/constants.d.mts.map +1 -0
- package/dist/constants.mjs +30 -0
- package/dist/constants.mjs.map +1 -0
- package/dist/errors.cjs +57 -0
- package/dist/errors.cjs.map +1 -0
- package/dist/errors.d.cts +34 -0
- package/dist/errors.d.cts.map +1 -0
- package/dist/errors.d.mts +34 -0
- package/dist/errors.d.mts.map +1 -0
- package/dist/errors.mjs +53 -0
- package/dist/errors.mjs.map +1 -0
- package/dist/index.cjs +19 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +9 -0
- package/dist/index.d.cts.map +1 -0
- package/dist/index.d.mts +9 -0
- package/dist/index.d.mts.map +1 -0
- package/dist/index.mjs +5 -0
- package/dist/index.mjs.map +1 -0
- package/dist/key-derivation.cjs +76 -0
- package/dist/key-derivation.cjs.map +1 -0
- package/dist/key-derivation.d.cts +43 -0
- package/dist/key-derivation.d.cts.map +1 -0
- package/dist/key-derivation.d.mts +43 -0
- package/dist/key-derivation.d.mts.map +1 -0
- package/dist/key-derivation.mjs +71 -0
- package/dist/key-derivation.mjs.map +1 -0
- package/dist/logger.cjs +9 -0
- package/dist/logger.cjs.map +1 -0
- package/dist/logger.d.cts +5 -0
- package/dist/logger.d.cts.map +1 -0
- package/dist/logger.d.mts +5 -0
- package/dist/logger.d.mts.map +1 -0
- package/dist/logger.mjs +6 -0
- package/dist/logger.mjs.map +1 -0
- package/dist/types.cjs +3 -0
- package/dist/types.cjs.map +1 -0
- package/dist/types.d.cts +92 -0
- package/dist/types.d.cts.map +1 -0
- package/dist/types.d.mts +92 -0
- package/dist/types.d.mts.map +1 -0
- package/dist/types.mjs +2 -0
- package/dist/types.mjs.map +1 -0
- package/dist/utils/crypto.cjs +55 -0
- package/dist/utils/crypto.cjs.map +1 -0
- package/dist/utils/crypto.d.cts +30 -0
- package/dist/utils/crypto.d.cts.map +1 -0
- package/dist/utils/crypto.d.mts +30 -0
- package/dist/utils/crypto.d.mts.map +1 -0
- package/dist/utils/crypto.mjs +49 -0
- package/dist/utils/crypto.mjs.map +1 -0
- package/dist/utils/encoding.cjs +42 -0
- package/dist/utils/encoding.cjs.map +1 -0
- package/dist/utils/encoding.d.cts +22 -0
- package/dist/utils/encoding.d.cts.map +1 -0
- package/dist/utils/encoding.d.mts +22 -0
- package/dist/utils/encoding.d.mts.map +1 -0
- package/dist/utils/encoding.mjs +36 -0
- package/dist/utils/encoding.mjs.map +1 -0
- package/dist/webauthn/constants.cjs +74 -0
- package/dist/webauthn/constants.cjs.map +1 -0
- package/dist/webauthn/constants.d.cts +68 -0
- package/dist/webauthn/constants.d.cts.map +1 -0
- package/dist/webauthn/constants.d.mts +68 -0
- package/dist/webauthn/constants.d.mts.map +1 -0
- package/dist/webauthn/constants.mjs +71 -0
- package/dist/webauthn/constants.mjs.map +1 -0
- package/dist/webauthn/decode-attestation-object.cjs +18 -0
- package/dist/webauthn/decode-attestation-object.cjs.map +1 -0
- package/dist/webauthn/decode-attestation-object.d.cts +10 -0
- package/dist/webauthn/decode-attestation-object.d.cts.map +1 -0
- package/dist/webauthn/decode-attestation-object.d.mts +10 -0
- package/dist/webauthn/decode-attestation-object.d.mts.map +1 -0
- package/dist/webauthn/decode-attestation-object.mjs +14 -0
- package/dist/webauthn/decode-attestation-object.mjs.map +1 -0
- package/dist/webauthn/decode-client-data-json.cjs +17 -0
- package/dist/webauthn/decode-client-data-json.cjs.map +1 -0
- package/dist/webauthn/decode-client-data-json.d.cts +9 -0
- package/dist/webauthn/decode-client-data-json.d.cts.map +1 -0
- package/dist/webauthn/decode-client-data-json.d.mts +9 -0
- package/dist/webauthn/decode-client-data-json.d.mts.map +1 -0
- package/dist/webauthn/decode-client-data-json.mjs +13 -0
- package/dist/webauthn/decode-client-data-json.mjs.map +1 -0
- package/dist/webauthn/match-expected-rp-id.cjs +43 -0
- package/dist/webauthn/match-expected-rp-id.cjs.map +1 -0
- package/dist/webauthn/match-expected-rp-id.d.cts +11 -0
- package/dist/webauthn/match-expected-rp-id.d.cts.map +1 -0
- package/dist/webauthn/match-expected-rp-id.d.mts +11 -0
- package/dist/webauthn/match-expected-rp-id.d.mts.map +1 -0
- package/dist/webauthn/match-expected-rp-id.mjs +39 -0
- package/dist/webauthn/match-expected-rp-id.mjs.map +1 -0
- package/dist/webauthn/parse-authenticator-data.cjs +69 -0
- package/dist/webauthn/parse-authenticator-data.cjs.map +1 -0
- package/dist/webauthn/parse-authenticator-data.d.cts +10 -0
- package/dist/webauthn/parse-authenticator-data.d.cts.map +1 -0
- package/dist/webauthn/parse-authenticator-data.d.mts +10 -0
- package/dist/webauthn/parse-authenticator-data.d.mts.map +1 -0
- package/dist/webauthn/parse-authenticator-data.mjs +65 -0
- package/dist/webauthn/parse-authenticator-data.mjs.map +1 -0
- package/dist/webauthn/types.cjs +3 -0
- package/dist/webauthn/types.cjs.map +1 -0
- package/dist/webauthn/types.d.cts +113 -0
- package/dist/webauthn/types.d.cts.map +1 -0
- package/dist/webauthn/types.d.mts +113 -0
- package/dist/webauthn/types.d.mts.map +1 -0
- package/dist/webauthn/types.mjs +2 -0
- package/dist/webauthn/types.mjs.map +1 -0
- package/dist/webauthn/verify-authentication-response.cjs +134 -0
- package/dist/webauthn/verify-authentication-response.cjs.map +1 -0
- package/dist/webauthn/verify-authentication-response.d.cts +63 -0
- package/dist/webauthn/verify-authentication-response.d.cts.map +1 -0
- package/dist/webauthn/verify-authentication-response.d.mts +63 -0
- package/dist/webauthn/verify-authentication-response.d.mts.map +1 -0
- package/dist/webauthn/verify-authentication-response.mjs +130 -0
- package/dist/webauthn/verify-authentication-response.mjs.map +1 -0
- package/dist/webauthn/verify-registration-response.cjs +205 -0
- package/dist/webauthn/verify-registration-response.cjs.map +1 -0
- package/dist/webauthn/verify-registration-response.d.cts +60 -0
- package/dist/webauthn/verify-registration-response.d.cts.map +1 -0
- package/dist/webauthn/verify-registration-response.d.mts +60 -0
- package/dist/webauthn/verify-registration-response.d.mts.map +1 -0
- package/dist/webauthn/verify-registration-response.mjs +201 -0
- package/dist/webauthn/verify-registration-response.mjs.map +1 -0
- package/dist/webauthn/verify-signature.cjs +176 -0
- package/dist/webauthn/verify-signature.cjs.map +1 -0
- package/dist/webauthn/verify-signature.d.cts +21 -0
- package/dist/webauthn/verify-signature.d.cts.map +1 -0
- package/dist/webauthn/verify-signature.d.mts +21 -0
- package/dist/webauthn/verify-signature.d.mts.map +1 -0
- package/dist/webauthn/verify-signature.mjs +172 -0
- package/dist/webauthn/verify-signature.mjs.map +1 -0
- 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":"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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 {\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"}
|