123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254 |
- import { sha256 } from "@noble/hashes/sha256";
- import { mod } from "@noble/curves/abstract/modular";
- import { secp256k1 } from "./secp256k1.js";
- import { assertBool, assertBytes, hexToBytes, toHex } from "./utils.js";
- // Use `secp256k1` module directly.
- // This is a legacy compatibility layer for the npm package `secp256k1` via noble-secp256k1
- const Point = secp256k1.ProjectivePoint;
- function hexToNumber(hex) {
- if (typeof hex !== "string") {
- throw new TypeError("hexToNumber: expected string, got " + typeof hex);
- }
- return BigInt(`0x${hex}`);
- }
- // Copy-paste from secp256k1, maybe export it?
- const bytesToNumber = (bytes) => hexToNumber(toHex(bytes));
- const numberToHex = (num) => num.toString(16).padStart(64, "0");
- const numberToBytes = (num) => hexToBytes(numberToHex(num));
- const ORDER = secp256k1.CURVE.n;
- function output(out = (len) => new Uint8Array(len), length, value) {
- if (typeof out === "function") {
- out = out(length);
- }
- assertBytes(out, length);
- if (value) {
- out.set(value);
- }
- return out;
- }
- function getSignature(signature) {
- assertBytes(signature, 64);
- return secp256k1.Signature.fromCompact(signature);
- }
- export function createPrivateKeySync() {
- return secp256k1.utils.randomPrivateKey();
- }
- export async function createPrivateKey() {
- return createPrivateKeySync();
- }
- export function privateKeyVerify(privateKey) {
- assertBytes(privateKey, 32);
- return secp256k1.utils.isValidPrivateKey(privateKey);
- }
- export function publicKeyCreate(privateKey, compressed = true, out) {
- assertBytes(privateKey, 32);
- assertBool(compressed);
- const res = secp256k1.getPublicKey(privateKey, compressed);
- return output(out, compressed ? 33 : 65, res);
- }
- export function publicKeyVerify(publicKey) {
- assertBytes(publicKey, 33, 65);
- try {
- Point.fromHex(publicKey);
- return true;
- }
- catch (e) {
- return false;
- }
- }
- export function publicKeyConvert(publicKey, compressed = true, out) {
- assertBytes(publicKey, 33, 65);
- assertBool(compressed);
- const res = Point.fromHex(publicKey).toRawBytes(compressed);
- return output(out, compressed ? 33 : 65, res);
- }
- export function ecdsaSign(msgHash, privateKey, options = { noncefn: undefined, data: undefined }, out) {
- assertBytes(msgHash, 32);
- assertBytes(privateKey, 32);
- if (typeof options !== "object" || options === null) {
- throw new TypeError("secp256k1.ecdsaSign: options should be object");
- }
- // noble-secp256k1 uses hmac instead of hmac-drbg here
- if (options &&
- (options.noncefn !== undefined || options.data !== undefined)) {
- throw new Error("Secp256k1: noncefn && data is unsupported");
- }
- const sig = secp256k1.sign(msgHash, privateKey);
- const recid = sig.recovery;
- return { signature: output(out, 64, sig.toCompactRawBytes()), recid };
- }
- export function ecdsaRecover(signature, recid, msgHash, compressed = true, out) {
- assertBytes(msgHash, 32);
- assertBool(compressed);
- const sign = getSignature(signature);
- const point = sign.addRecoveryBit(recid).recoverPublicKey(msgHash);
- return output(out, compressed ? 33 : 65, point.toRawBytes(compressed));
- }
- export function ecdsaVerify(signature, msgHash, publicKey) {
- assertBytes(signature, 64);
- assertBytes(msgHash, 32);
- assertBytes(publicKey, 33, 65);
- assertBytes(signature, 64);
- const r = bytesToNumber(signature.slice(0, 32));
- const s = bytesToNumber(signature.slice(32, 64));
- if (r >= ORDER || s >= ORDER) {
- throw new Error("Cannot parse signature");
- }
- const pub = Point.fromHex(publicKey); // can throw error
- pub; // typescript
- let sig;
- try {
- sig = getSignature(signature);
- }
- catch (error) {
- return false;
- }
- return secp256k1.verify(sig, msgHash, publicKey);
- }
- export function privateKeyTweakAdd(privateKey, tweak) {
- assertBytes(privateKey, 32);
- assertBytes(tweak, 32);
- let t = bytesToNumber(tweak);
- if (t === 0n) {
- throw new Error("Tweak must not be zero");
- }
- if (t >= ORDER) {
- throw new Error("Tweak bigger than curve order");
- }
- t += bytesToNumber(privateKey);
- if (t >= ORDER) {
- t -= ORDER;
- }
- if (t === 0n) {
- throw new Error("The tweak was out of range or the resulted private key is invalid");
- }
- privateKey.set(hexToBytes(numberToHex(t)));
- return privateKey;
- }
- export function privateKeyNegate(privateKey) {
- assertBytes(privateKey, 32);
- const bn = mod(-bytesToNumber(privateKey), ORDER);
- privateKey.set(hexToBytes(numberToHex(bn)));
- return privateKey;
- }
- export function publicKeyNegate(publicKey, compressed = true, out) {
- assertBytes(publicKey, 33, 65);
- assertBool(compressed);
- const point = Point.fromHex(publicKey).negate();
- return output(out, compressed ? 33 : 65, point.toRawBytes(compressed));
- }
- export function publicKeyCombine(publicKeys, compressed = true, out) {
- if (!Array.isArray(publicKeys) || !publicKeys.length) {
- throw new TypeError(`Expected array with one or more items, not ${publicKeys}`);
- }
- for (const publicKey of publicKeys) {
- assertBytes(publicKey, 33, 65);
- }
- assertBool(compressed);
- const combined = publicKeys
- .map((pub) => Point.fromHex(pub))
- .reduce((res, curr) => res.add(curr), Point.ZERO);
- // Prohibit returning ZERO point
- if (combined.equals(Point.ZERO)) {
- throw new Error("Combined result must not be zero");
- }
- return output(out, compressed ? 33 : 65, combined.toRawBytes(compressed));
- }
- export function publicKeyTweakAdd(publicKey, tweak, compressed = true, out) {
- assertBytes(publicKey, 33, 65);
- assertBytes(tweak, 32);
- assertBool(compressed);
- const p1 = Point.fromHex(publicKey);
- const p2 = Point.fromPrivateKey(tweak);
- const point = p1.add(p2);
- if (p2.equals(Point.ZERO) || point.equals(Point.ZERO)) {
- throw new Error("Tweak must not be zero");
- }
- return output(out, compressed ? 33 : 65, point.toRawBytes(compressed));
- }
- export function publicKeyTweakMul(publicKey, tweak, compressed = true, out) {
- assertBytes(publicKey, 33, 65);
- assertBytes(tweak, 32);
- assertBool(compressed);
- const bn = bytesToNumber(tweak);
- if (bn === 0n) {
- throw new Error("Tweak must not be zero");
- }
- if (bn <= 1 || bn >= ORDER) {
- throw new Error("Tweak is zero or bigger than curve order");
- }
- const point = Point.fromHex(publicKey).multiply(bn);
- return output(out, compressed ? 33 : 65, point.toRawBytes(compressed));
- }
- export function privateKeyTweakMul(privateKey, tweak) {
- assertBytes(privateKey, 32);
- assertBytes(tweak, 32);
- const bn = bytesToNumber(tweak);
- if (bn <= 1 || bn >= ORDER) {
- throw new Error("Tweak is zero or bigger than curve order");
- }
- const res = mod(bn * bytesToNumber(privateKey), ORDER);
- if (res === 0n) {
- throw new Error("The tweak was out of range or the resulted private key is invalid");
- }
- privateKey.set(hexToBytes(numberToHex(res)));
- return privateKey;
- }
- // internal -> DER
- export function signatureExport(signature, out) {
- const res = getSignature(signature).toDERRawBytes();
- return output(out, 72, res.slice()).slice(0, res.length);
- }
- // DER -> internal
- export function signatureImport(signature, out) {
- assertBytes(signature);
- const sig = secp256k1.Signature.fromDER(signature);
- return output(out, 64, hexToBytes(sig.toCompactHex()));
- }
- export function signatureNormalize(signature) {
- const res = getSignature(signature);
- if (res.s > ORDER / 2n) {
- signature.set(numberToBytes(ORDER - res.s), 32);
- }
- return signature;
- }
- export function ecdh(publicKey, privateKey, options = {}, out) {
- assertBytes(publicKey, 33, 65);
- assertBytes(privateKey, 32);
- if (typeof options !== "object" || options === null) {
- throw new TypeError("secp256k1.ecdh: options should be object");
- }
- if (options.data !== undefined) {
- assertBytes(options.data);
- }
- const point = Point.fromHex(secp256k1.getSharedSecret(privateKey, publicKey));
- if (options.hashfn === undefined) {
- return output(out, 32, sha256(point.toRawBytes(true)));
- }
- if (typeof options.hashfn !== "function") {
- throw new TypeError("secp256k1.ecdh: options.hashfn should be function");
- }
- if (options.xbuf !== undefined) {
- assertBytes(options.xbuf, 32);
- }
- if (options.ybuf !== undefined) {
- assertBytes(options.ybuf, 32);
- }
- assertBytes(out, 32);
- const { x, y } = point.toAffine();
- const xbuf = options.xbuf || new Uint8Array(32);
- xbuf.set(numberToBytes(x));
- const ybuf = options.ybuf || new Uint8Array(32);
- ybuf.set(numberToBytes(y));
- const hash = options.hashfn(xbuf, ybuf, options.data);
- if (!(hash instanceof Uint8Array) || hash.length !== 32) {
- throw new Error("secp256k1.ecdh: invalid options.hashfn output");
- }
- return output(out, 32, hash);
- }
- export function contextRandomize(seed) {
- if (seed !== null) {
- assertBytes(seed, 32);
- }
- // There is no context to randomize
- }
|