Consolidate signature verification logic

This commit is contained in:
Franck Royer 2022-05-20 12:06:14 +10:00
parent 2ed3580584
commit c85c39a427
No known key found for this signature in database
GPG Key ID: A82ED75A8DFC50A4
6 changed files with 24 additions and 38 deletions

View File

@ -80,3 +80,19 @@ export function compressPublicKey(publicKey: Uint8Array): Uint8Array {
const point = secp.Point.fromHex(publicKey);
return point.toRawBytes(true);
}
/**
* Verify an ECDSA signature.
*/
export function verifySignature(
signature: Uint8Array,
message: Uint8Array | string,
publicKey: Uint8Array
): boolean {
try {
const _signature = secp.Signature.fromCompact(signature.slice(0, 64));
return secp.verify(_signature, message, publicKey);
} catch {
return false;
}
}

View File

@ -1,8 +1,7 @@
import * as secp from "@noble/secp256k1";
import * as base32 from "hi-base32";
import { fromString } from "uint8arrays/from-string";
import { keccak256 } from "../crypto";
import { keccak256, verifySignature } from "../crypto";
import { ENR } from "../enr";
import { utf8ToBytes } from "../utils";
@ -47,17 +46,11 @@ export class ENRTree {
64
);
let isVerified;
try {
const _sig = secp.Signature.fromCompact(signatureBuffer.slice(0, 64));
isVerified = secp.verify(
_sig,
const isVerified = verifySignature(
signatureBuffer,
keccak256(signedComponentBuffer),
new Uint8Array(decodedPublicKey)
);
} catch {
isVerified = false;
}
if (!isVerified) throw new Error("Unable to verify ENRTree root signature");

View File

@ -9,7 +9,7 @@ import { fromString } from "uint8arrays/from-string";
import { toString } from "uint8arrays/to-string";
import { encode as varintEncode } from "varint";
import { compressPublicKey } from "../crypto";
import { compressPublicKey, keccak256, verifySignature } from "../crypto";
import { bytesToHex, bytesToUtf8, hexToBytes, utf8ToBytes } from "../utils";
import { ERR_INVALID_ID, ERR_NO_SIGNATURE, MAX_RECORD_SIZE } from "./constants";
@ -472,7 +472,7 @@ export class ENR extends Map<ENRKey, ENRValue> {
if (!this.publicKey) {
throw new Error("Failed to verify ENR: No public key");
}
return v4.verify(this.publicKey, data, signature);
return verifySignature(signature, keccak256(data), this.publicKey);
}
async sign(data: Uint8Array, privateKey: Uint8Array): Promise<Uint8Array> {

View File

@ -49,15 +49,6 @@ export class Secp256k1Keypair implements IKeypair {
return true;
}
verify(msg: Uint8Array, sig: Uint8Array): boolean {
try {
const _sig = secp.Signature.fromCompact(sig.slice(0, 64));
return secp.verify(_sig, msg, this.publicKey);
} catch {
return false;
}
}
get privateKey(): Uint8Array {
if (!this._privateKey) {
throw new Error();

View File

@ -10,6 +10,5 @@ export interface IKeypair {
publicKey: Uint8Array;
privateKeyVerify(): boolean;
publicKeyVerify(): boolean;
verify(msg: Uint8Array, sig: Uint8Array): boolean;
hasPrivateKey(): boolean;
}

View File

@ -14,19 +14,6 @@ export async function sign(
});
}
export function verify(
pubKey: Uint8Array,
msg: Uint8Array,
sig: Uint8Array
): boolean {
try {
const _sig = secp.Signature.fromCompact(sig.slice(0, 64));
return secp.verify(_sig, keccak256(msg), pubKey);
} catch {
return false;
}
}
export function nodeId(pubKey: Uint8Array): NodeId {
const publicKey = secp.Point.fromHex(pubKey);
const uncompressedPubkey = publicKey.toRawBytes(false);