2023-01-13 05:40:33 +01:00
|
|
|
import * as crypto from "node:crypto";
|
|
|
|
import config from "@/config/index.js";
|
2023-11-26 21:33:46 +01:00
|
|
|
import * as jsrsasign from "jsrsasign";
|
2019-07-03 13:18:07 +02:00
|
|
|
|
|
|
|
const ECC_PRELUDE = Buffer.from([0x04]);
|
|
|
|
const NULL_BYTE = Buffer.from([0]);
|
|
|
|
const PEM_PRELUDE = Buffer.from(
|
2023-01-13 05:40:33 +01:00
|
|
|
"3059301306072a8648ce3d020106082a8648ce3d030107034200",
|
|
|
|
"hex",
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
// Android Safetynet attestations are signed with this cert:
|
|
|
|
const GSR2 = `-----BEGIN CERTIFICATE-----
|
|
|
|
MIIDujCCAqKgAwIBAgILBAAAAAABD4Ym5g0wDQYJKoZIhvcNAQEFBQAwTDEgMB4G
|
|
|
|
A1UECxMXR2xvYmFsU2lnbiBSb290IENBIC0gUjIxEzARBgNVBAoTCkdsb2JhbFNp
|
|
|
|
Z24xEzARBgNVBAMTCkdsb2JhbFNpZ24wHhcNMDYxMjE1MDgwMDAwWhcNMjExMjE1
|
|
|
|
MDgwMDAwWjBMMSAwHgYDVQQLExdHbG9iYWxTaWduIFJvb3QgQ0EgLSBSMjETMBEG
|
|
|
|
A1UEChMKR2xvYmFsU2lnbjETMBEGA1UEAxMKR2xvYmFsU2lnbjCCASIwDQYJKoZI
|
|
|
|
hvcNAQEBBQADggEPADCCAQoCggEBAKbPJA6+Lm8omUVCxKs+IVSbC9N/hHD6ErPL
|
|
|
|
v4dfxn+G07IwXNb9rfF73OX4YJYJkhD10FPe+3t+c4isUoh7SqbKSaZeqKeMWhG8
|
|
|
|
eoLrvozps6yWJQeXSpkqBy+0Hne/ig+1AnwblrjFuTosvNYSuetZfeLQBoZfXklq
|
|
|
|
tTleiDTsvHgMCJiEbKjNS7SgfQx5TfC4LcshytVsW33hoCmEofnTlEnLJGKRILzd
|
|
|
|
C9XZzPnqJworc5HGnRusyMvo4KD0L5CLTfuwNhv2GXqF4G3yYROIXJ/gkwpRl4pa
|
|
|
|
zq+r1feqCapgvdzZX99yqWATXgAByUr6P6TqBwMhAo6CygPCm48CAwEAAaOBnDCB
|
|
|
|
mTAOBgNVHQ8BAf8EBAMCAQYwDwYDVR0TAQH/BAUwAwEB/zAdBgNVHQ4EFgQUm+IH
|
|
|
|
V2ccHsBqBt5ZtJot39wZhi4wNgYDVR0fBC8wLTAroCmgJ4YlaHR0cDovL2NybC5n
|
|
|
|
bG9iYWxzaWduLm5ldC9yb290LXIyLmNybDAfBgNVHSMEGDAWgBSb4gdXZxwewGoG
|
|
|
|
3lm0mi3f3BmGLjANBgkqhkiG9w0BAQUFAAOCAQEAmYFThxxol4aR7OBKuEQLq4Gs
|
|
|
|
J0/WwbgcQ3izDJr86iw8bmEbTUsp9Z8FHSbBuOmDAGJFtqkIk7mpM0sYmsL4h4hO
|
|
|
|
291xNBrBVNpGP+DTKqttVCL1OmLNIG+6KYnX3ZHu01yiPqFbQfXf5WRDLenVOavS
|
|
|
|
ot+3i9DAgBkcRcAtjOj4LaR0VknFBbVPFd5uRHg5h6h+u/N5GJG79G+dwfCMNYxd
|
|
|
|
AfvDbbnvRG15RjF+Cv6pgsH/76tuIMRQyV+dTZsXjAzlAcmgQWpzU/qlULRuJQ/7
|
|
|
|
TBj0/VLZjmmx6BEP3ojY+x1J96relc8geMJgEtslQIxq/H5COEBkEveegeGTLg==
|
|
|
|
-----END CERTIFICATE-----\n`;
|
|
|
|
|
|
|
|
function base64URLDecode(source: string) {
|
2023-01-13 05:40:33 +01:00
|
|
|
return Buffer.from(source.replace(/\-/g, "+").replace(/_/g, "/"), "base64");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function getCertSubject(certificate: string) {
|
|
|
|
const subjectCert = new jsrsasign.X509();
|
|
|
|
subjectCert.readCertPEM(certificate);
|
|
|
|
|
|
|
|
const subjectString = subjectCert.getSubjectString();
|
2023-01-13 05:40:33 +01:00
|
|
|
const subjectFields = subjectString.slice(1).split("/");
|
2019-07-03 13:18:07 +02:00
|
|
|
|
|
|
|
const fields = {} as Record<string, string>;
|
|
|
|
for (const field of subjectFields) {
|
2023-01-13 05:40:33 +01:00
|
|
|
const eqIndex = field.indexOf("=");
|
2019-07-03 13:18:07 +02:00
|
|
|
fields[field.substring(0, eqIndex)] = field.substring(eqIndex + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
function verifyCertificateChain(certificates: string[]) {
|
|
|
|
let valid = true;
|
|
|
|
|
|
|
|
for (let i = 0; i < certificates.length; i++) {
|
|
|
|
const Cert = certificates[i];
|
|
|
|
const certificate = new jsrsasign.X509();
|
|
|
|
certificate.readCertPEM(Cert);
|
|
|
|
|
|
|
|
const CACert = i + 1 >= certificates.length ? Cert : certificates[i + 1];
|
|
|
|
|
2019-07-04 19:00:54 +02:00
|
|
|
const certStruct = jsrsasign.ASN1HEX.getTLVbyList(certificate.hex!, 0, [0]);
|
2019-07-03 13:18:07 +02:00
|
|
|
const algorithm = certificate.getSignatureAlgorithmField();
|
|
|
|
const signatureHex = certificate.getSignatureValueHex();
|
|
|
|
|
|
|
|
// Verify against CA
|
2021-11-13 11:10:14 +01:00
|
|
|
const Signature = new jsrsasign.KJUR.crypto.Signature({ alg: algorithm });
|
2019-07-03 13:18:07 +02:00
|
|
|
Signature.init(CACert);
|
|
|
|
Signature.updateHex(certStruct);
|
2019-07-04 19:00:54 +02:00
|
|
|
valid = valid && !!Signature.verify(signatureHex); // true if CA signed the certificate
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
function PEMString(pemBuffer: Buffer, type = "CERTIFICATE") {
|
2020-04-04 01:46:54 +02:00
|
|
|
if (pemBuffer.length === 65 && pemBuffer[0] === 0x04) {
|
2019-07-03 13:18:07 +02:00
|
|
|
pemBuffer = Buffer.concat([PEM_PRELUDE, pemBuffer], 91);
|
2023-01-13 05:40:33 +01:00
|
|
|
type = "PUBLIC KEY";
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
const cert = pemBuffer.toString("base64");
|
2019-07-03 13:18:07 +02:00
|
|
|
|
|
|
|
const keyParts = [];
|
|
|
|
const max = Math.ceil(cert.length / 64);
|
|
|
|
let start = 0;
|
|
|
|
for (let i = 0; i < max; i++) {
|
|
|
|
keyParts.push(cert.substring(start, start + 64));
|
|
|
|
start += 64;
|
|
|
|
}
|
|
|
|
|
2023-01-16 20:19:20 +01:00
|
|
|
return `-----BEGIN ${type}-----\n${keyParts.join(
|
|
|
|
"\n",
|
|
|
|
)}\n-----END ${type}-----\n`;
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export function hash(data: Buffer) {
|
2023-01-13 05:40:33 +01:00
|
|
|
return crypto.createHash("sha256").update(data).digest();
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export function verifyLogin({
|
|
|
|
publicKey,
|
|
|
|
authenticatorData,
|
|
|
|
clientDataJSON,
|
|
|
|
clientData,
|
|
|
|
signature,
|
2021-12-09 15:58:30 +01:00
|
|
|
challenge,
|
2019-07-03 13:18:07 +02:00
|
|
|
}: {
|
2023-01-13 05:40:33 +01:00
|
|
|
publicKey: Buffer;
|
|
|
|
authenticatorData: Buffer;
|
|
|
|
clientDataJSON: Buffer;
|
|
|
|
clientData: any;
|
|
|
|
signature: Buffer;
|
|
|
|
challenge: string;
|
2019-07-03 13:18:07 +02:00
|
|
|
}) {
|
2023-01-13 05:40:33 +01:00
|
|
|
if (clientData.type !== "webauthn.get") {
|
|
|
|
throw new Error("type is not webauthn.get");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (hash(clientData.challenge).toString("hex") !== challenge) {
|
|
|
|
throw new Error("challenge mismatch");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
if (clientData.origin !== `${config.scheme}://${config.host}`) {
|
|
|
|
throw new Error("origin mismatch");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const verificationData = Buffer.concat(
|
|
|
|
[authenticatorData, hash(clientDataJSON)],
|
2021-11-13 11:10:14 +01:00
|
|
|
32 + authenticatorData.length,
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
return crypto
|
2023-01-13 05:40:33 +01:00
|
|
|
.createVerify("SHA256")
|
2019-07-03 13:18:07 +02:00
|
|
|
.update(verificationData)
|
|
|
|
.verify(PEMString(publicKey), signature);
|
|
|
|
}
|
|
|
|
|
|
|
|
export const procedures = {
|
|
|
|
none: {
|
2022-04-23 05:48:26 +02:00
|
|
|
verify({ publicKey }: { publicKey: Map<number, Buffer> }) {
|
2019-07-03 13:18:07 +02:00
|
|
|
const negTwo = publicKey.get(-2);
|
|
|
|
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negTwo || negTwo.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -2 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
const negThree = publicKey.get(-3);
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negThree || negThree.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -3 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const publicKeyU2F = Buffer.concat(
|
|
|
|
[ECC_PRELUDE, negTwo, negThree],
|
2021-11-13 11:10:14 +01:00
|
|
|
1 + 32 + 32,
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
return {
|
|
|
|
publicKey: publicKeyU2F,
|
2021-11-13 11:10:14 +01:00
|
|
|
valid: true,
|
2019-07-03 13:18:07 +02:00
|
|
|
};
|
2021-11-13 11:10:14 +01:00
|
|
|
},
|
2019-07-03 13:18:07 +02:00
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
"android-key": {
|
2019-07-03 13:18:07 +02:00
|
|
|
verify({
|
|
|
|
attStmt,
|
|
|
|
authenticatorData,
|
|
|
|
clientDataHash,
|
|
|
|
publicKey,
|
|
|
|
rpIdHash,
|
2021-11-13 11:10:14 +01:00
|
|
|
credentialId,
|
2019-07-03 13:18:07 +02:00
|
|
|
}: {
|
2023-01-13 05:40:33 +01:00
|
|
|
attStmt: any;
|
|
|
|
authenticatorData: Buffer;
|
|
|
|
clientDataHash: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
publicKey: Map<number, any>;
|
2023-01-13 05:40:33 +01:00
|
|
|
rpIdHash: Buffer;
|
|
|
|
credentialId: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
}) {
|
2022-04-03 08:33:22 +02:00
|
|
|
if (attStmt.alg !== -7) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("alg mismatch");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const verificationData = Buffer.concat([
|
|
|
|
authenticatorData,
|
2021-11-13 11:10:14 +01:00
|
|
|
clientDataHash,
|
2019-07-03 13:18:07 +02:00
|
|
|
]);
|
|
|
|
|
|
|
|
const attCert: Buffer = attStmt.x5c[0];
|
|
|
|
|
|
|
|
const negTwo = publicKey.get(-2);
|
|
|
|
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negTwo || negTwo.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -2 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
const negThree = publicKey.get(-3);
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negThree || negThree.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -3 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const publicKeyData = Buffer.concat(
|
|
|
|
[ECC_PRELUDE, negTwo, negThree],
|
2021-11-13 11:10:14 +01:00
|
|
|
1 + 32 + 32,
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
if (!attCert.equals(publicKeyData)) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("public key mismatch");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const isValid = crypto
|
2023-01-13 05:40:33 +01:00
|
|
|
.createVerify("SHA256")
|
2019-07-03 13:18:07 +02:00
|
|
|
.update(verificationData)
|
|
|
|
.verify(PEMString(attCert), attStmt.sig);
|
|
|
|
|
|
|
|
// TODO: Check 'attestationChallenge' field in extension of cert matches hash(clientDataJSON)
|
|
|
|
|
|
|
|
return {
|
|
|
|
valid: isValid,
|
2021-11-13 11:10:14 +01:00
|
|
|
publicKey: publicKeyData,
|
2019-07-03 13:18:07 +02:00
|
|
|
};
|
2021-11-13 11:10:14 +01:00
|
|
|
},
|
2019-07-03 13:18:07 +02:00
|
|
|
},
|
|
|
|
// what a stupid attestation
|
2023-01-13 05:40:33 +01:00
|
|
|
"android-safetynet": {
|
2019-07-03 13:18:07 +02:00
|
|
|
verify({
|
|
|
|
attStmt,
|
|
|
|
authenticatorData,
|
|
|
|
clientDataHash,
|
|
|
|
publicKey,
|
|
|
|
rpIdHash,
|
2021-11-13 11:10:14 +01:00
|
|
|
credentialId,
|
2019-07-03 13:18:07 +02:00
|
|
|
}: {
|
2023-01-13 05:40:33 +01:00
|
|
|
attStmt: any;
|
|
|
|
authenticatorData: Buffer;
|
|
|
|
clientDataHash: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
publicKey: Map<number, any>;
|
2023-01-13 05:40:33 +01:00
|
|
|
rpIdHash: Buffer;
|
|
|
|
credentialId: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
}) {
|
|
|
|
const verificationData = hash(
|
2021-11-13 11:10:14 +01:00
|
|
|
Buffer.concat([authenticatorData, clientDataHash]),
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const jwsParts = attStmt.response.toString("utf-8").split(".");
|
2019-07-03 13:18:07 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const header = JSON.parse(base64URLDecode(jwsParts[0]).toString("utf-8"));
|
2019-07-03 13:18:07 +02:00
|
|
|
const response = JSON.parse(
|
2023-01-13 05:40:33 +01:00
|
|
|
base64URLDecode(jwsParts[1]).toString("utf-8"),
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
const signature = jwsParts[2];
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (!verificationData.equals(Buffer.from(response.nonce, "base64"))) {
|
|
|
|
throw new Error("invalid nonce");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const certificateChain = header.x5c
|
2019-07-04 19:00:54 +02:00
|
|
|
.map((key: any) => PEMString(key))
|
2019-07-03 13:18:07 +02:00
|
|
|
.concat([GSR2]);
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (getCertSubject(certificateChain[0]).CN !== "attest.android.com") {
|
|
|
|
throw new Error("invalid common name");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!verifyCertificateChain(certificateChain)) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("Invalid certificate chain!");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const signatureBase = Buffer.from(
|
2023-01-13 05:40:33 +01:00
|
|
|
`${jwsParts[0]}.${jwsParts[1]}`,
|
|
|
|
"utf-8",
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const valid = crypto
|
2023-01-13 05:40:33 +01:00
|
|
|
.createVerify("sha256")
|
2019-07-03 13:18:07 +02:00
|
|
|
.update(signatureBase)
|
|
|
|
.verify(certificateChain[0], base64URLDecode(signature));
|
|
|
|
|
|
|
|
const negTwo = publicKey.get(-2);
|
|
|
|
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negTwo || negTwo.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -2 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
const negThree = publicKey.get(-3);
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negThree || negThree.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -3 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const publicKeyData = Buffer.concat(
|
|
|
|
[ECC_PRELUDE, negTwo, negThree],
|
2021-11-13 11:10:14 +01:00
|
|
|
1 + 32 + 32,
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
return {
|
|
|
|
valid,
|
2021-11-13 11:10:14 +01:00
|
|
|
publicKey: publicKeyData,
|
2019-07-03 13:18:07 +02:00
|
|
|
};
|
2021-11-13 11:10:14 +01:00
|
|
|
},
|
2019-07-03 13:18:07 +02:00
|
|
|
},
|
|
|
|
packed: {
|
|
|
|
verify({
|
|
|
|
attStmt,
|
|
|
|
authenticatorData,
|
|
|
|
clientDataHash,
|
|
|
|
publicKey,
|
|
|
|
rpIdHash,
|
2021-11-13 11:10:14 +01:00
|
|
|
credentialId,
|
2019-07-03 13:18:07 +02:00
|
|
|
}: {
|
2023-01-13 05:40:33 +01:00
|
|
|
attStmt: any;
|
|
|
|
authenticatorData: Buffer;
|
|
|
|
clientDataHash: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
publicKey: Map<number, any>;
|
2023-01-13 05:40:33 +01:00
|
|
|
rpIdHash: Buffer;
|
|
|
|
credentialId: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
}) {
|
|
|
|
const verificationData = Buffer.concat([
|
|
|
|
authenticatorData,
|
2021-11-13 11:10:14 +01:00
|
|
|
clientDataHash,
|
2019-07-03 13:18:07 +02:00
|
|
|
]);
|
|
|
|
|
|
|
|
if (attStmt.x5c) {
|
|
|
|
const attCert = attStmt.x5c[0];
|
|
|
|
|
|
|
|
const validSignature = crypto
|
2023-01-13 05:40:33 +01:00
|
|
|
.createVerify("SHA256")
|
2019-07-03 13:18:07 +02:00
|
|
|
.update(verificationData)
|
|
|
|
.verify(PEMString(attCert), attStmt.sig);
|
|
|
|
|
|
|
|
const negTwo = publicKey.get(-2);
|
|
|
|
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negTwo || negTwo.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -2 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
const negThree = publicKey.get(-3);
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negThree || negThree.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -3 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const publicKeyData = Buffer.concat(
|
|
|
|
[ECC_PRELUDE, negTwo, negThree],
|
2021-11-13 11:10:14 +01:00
|
|
|
1 + 32 + 32,
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
return {
|
|
|
|
valid: validSignature,
|
2021-11-13 11:10:14 +01:00
|
|
|
publicKey: publicKeyData,
|
2019-07-03 13:18:07 +02:00
|
|
|
};
|
|
|
|
} else if (attStmt.ecdaaKeyId) {
|
|
|
|
// https://fidoalliance.org/specs/fido-v2.0-id-20180227/fido-ecdaa-algorithm-v2.0-id-20180227.html#ecdaa-verify-operation
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("ECDAA-Verify is not supported");
|
2019-07-03 13:18:07 +02:00
|
|
|
} else {
|
2023-01-13 05:40:33 +01:00
|
|
|
if (attStmt.alg !== -7) throw new Error("alg mismatch");
|
2019-07-03 13:18:07 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("self attestation is not supported");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
2021-11-13 11:10:14 +01:00
|
|
|
},
|
2019-07-03 13:18:07 +02:00
|
|
|
},
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
"fido-u2f": {
|
2019-07-03 13:18:07 +02:00
|
|
|
verify({
|
|
|
|
attStmt,
|
|
|
|
authenticatorData,
|
|
|
|
clientDataHash,
|
|
|
|
publicKey,
|
|
|
|
rpIdHash,
|
2021-11-13 11:10:14 +01:00
|
|
|
credentialId,
|
2019-07-03 13:18:07 +02:00
|
|
|
}: {
|
2023-01-13 05:40:33 +01:00
|
|
|
attStmt: any;
|
|
|
|
authenticatorData: Buffer;
|
|
|
|
clientDataHash: Buffer;
|
|
|
|
publicKey: Map<number, any>;
|
|
|
|
rpIdHash: Buffer;
|
|
|
|
credentialId: Buffer;
|
2019-07-03 13:18:07 +02:00
|
|
|
}) {
|
|
|
|
const x5c: Buffer[] = attStmt.x5c;
|
2022-04-03 08:33:22 +02:00
|
|
|
if (x5c.length !== 1) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("x5c length does not match expectation");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const attCert = x5c[0];
|
|
|
|
|
|
|
|
// TODO: make sure attCert is an Elliptic Curve (EC) public key over the P-256 curve
|
|
|
|
|
|
|
|
const negTwo: Buffer = publicKey.get(-2);
|
|
|
|
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negTwo || negTwo.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -2 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
const negThree: Buffer = publicKey.get(-3);
|
2022-04-03 08:33:22 +02:00
|
|
|
if (!negThree || negThree.length !== 32) {
|
2023-01-13 05:40:33 +01:00
|
|
|
throw new Error("invalid or no -3 key given");
|
2019-07-03 13:18:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const publicKeyU2F = Buffer.concat(
|
|
|
|
[ECC_PRELUDE, negTwo, negThree],
|
2021-11-13 11:10:14 +01:00
|
|
|
1 + 32 + 32,
|
2019-07-03 13:18:07 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const verificationData = Buffer.concat([
|
|
|
|
NULL_BYTE,
|
|
|
|
rpIdHash,
|
|
|
|
clientDataHash,
|
|
|
|
credentialId,
|
2021-11-13 11:10:14 +01:00
|
|
|
publicKeyU2F,
|
2019-07-03 13:18:07 +02:00
|
|
|
]);
|
|
|
|
|
|
|
|
const validSignature = crypto
|
2023-01-13 05:40:33 +01:00
|
|
|
.createVerify("SHA256")
|
2019-07-03 13:18:07 +02:00
|
|
|
.update(verificationData)
|
|
|
|
.verify(PEMString(attCert), attStmt.sig);
|
|
|
|
|
|
|
|
return {
|
|
|
|
valid: validSignature,
|
2021-11-13 11:10:14 +01:00
|
|
|
publicKey: publicKeyU2F,
|
2019-07-03 13:18:07 +02:00
|
|
|
};
|
2021-11-13 11:10:14 +01:00
|
|
|
},
|
|
|
|
},
|
2019-07-03 13:18:07 +02:00
|
|
|
};
|