UNPKG

ox

Version:

Ethereum Standard Library

423 lines (392 loc) 11.4 kB
import { secp256k1 } from '@noble/curves/secp256k1' import * as Address from './Address.js' import * as Bytes from './Bytes.js' import type * as Errors from './Errors.js' import * as Hex from './Hex.js' import * as Entropy from './internal/entropy.js' import type { OneOf } from './internal/types.js' import * as PublicKey from './PublicKey.js' import type * as Signature from './Signature.js' /** Re-export of noble/curves secp256k1 utilities. */ export const noble = secp256k1 /** * Creates a new secp256k1 ECDSA key pair consisting of a private key and its corresponding public key. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const { privateKey, publicKey } = Secp256k1.createKeyPair() * ``` * * @param options - The options to generate the key pair. * @returns The generated key pair containing both private and public keys. */ export function createKeyPair<as extends 'Hex' | 'Bytes' = 'Hex'>( options: createKeyPair.Options<as> = {}, ): createKeyPair.ReturnType<as> { const { as = 'Hex' } = options const privateKey = randomPrivateKey({ as }) const publicKey = getPublicKey({ privateKey }) return { privateKey: privateKey as never, publicKey, } } export declare namespace createKeyPair { type Options<as extends 'Hex' | 'Bytes' = 'Hex'> = { /** * Format of the returned private key. * @default 'Hex' */ as?: as | 'Hex' | 'Bytes' | undefined } type ReturnType<as extends 'Hex' | 'Bytes'> = { privateKey: | (as extends 'Bytes' ? Bytes.Bytes : never) | (as extends 'Hex' ? Hex.Hex : never) publicKey: PublicKey.PublicKey } type ErrorType = | Hex.fromBytes.ErrorType | PublicKey.from.ErrorType | Errors.GlobalErrorType } /** * Computes the secp256k1 ECDSA public key from a provided private key. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const publicKey = Secp256k1.getPublicKey({ privateKey: '0x...' }) * ``` * * @param options - The options to compute the public key. * @returns The computed public key. */ export function getPublicKey( options: getPublicKey.Options, ): PublicKey.PublicKey { const { privateKey } = options const point = secp256k1.ProjectivePoint.fromPrivateKey( Hex.from(privateKey).slice(2), ) return PublicKey.from(point) } export declare namespace getPublicKey { type Options = { /** * Private key to compute the public key from. */ privateKey: Hex.Hex | Bytes.Bytes } type ErrorType = | Hex.from.ErrorType | PublicKey.from.ErrorType | Errors.GlobalErrorType } /** * Computes a shared secret using ECDH (Elliptic Curve Diffie-Hellman) between a private key and a public key. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const { privateKey: privateKeyA } = Secp256k1.createKeyPair() * const { publicKey: publicKeyB } = Secp256k1.createKeyPair() * * const sharedSecret = Secp256k1.getSharedSecret({ * privateKey: privateKeyA, * publicKey: publicKeyB * }) * ``` * * @param options - The options to compute the shared secret. * @returns The computed shared secret. */ export function getSharedSecret<as extends 'Hex' | 'Bytes' = 'Hex'>( options: getSharedSecret.Options<as>, ): getSharedSecret.ReturnType<as> { const { as = 'Hex', privateKey, publicKey } = options const point = secp256k1.ProjectivePoint.fromHex( PublicKey.toHex(publicKey).slice(2), ) const sharedPoint = point.multiply( secp256k1.utils.normPrivateKeyToScalar(Hex.from(privateKey).slice(2)), ) const sharedSecret = sharedPoint.toRawBytes(true) // compressed format if (as === 'Hex') return Hex.fromBytes(sharedSecret) as never return sharedSecret as never } export declare namespace getSharedSecret { type Options<as extends 'Hex' | 'Bytes' = 'Hex'> = { /** * Format of the returned shared secret. * @default 'Hex' */ as?: as | 'Hex' | 'Bytes' | undefined /** * Private key to use for the shared secret computation. */ privateKey: Hex.Hex | Bytes.Bytes /** * Public key to use for the shared secret computation. */ publicKey: PublicKey.PublicKey<boolean> } type ReturnType<as extends 'Hex' | 'Bytes'> = | (as extends 'Bytes' ? Bytes.Bytes : never) | (as extends 'Hex' ? Hex.Hex : never) type ErrorType = | Hex.from.ErrorType | PublicKey.toHex.ErrorType | Hex.fromBytes.ErrorType | Errors.GlobalErrorType } /** * Generates a random ECDSA private key on the secp256k1 curve. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const privateKey = Secp256k1.randomPrivateKey() * ``` * * @param options - The options to generate the private key. * @returns The generated private key. */ export function randomPrivateKey<as extends 'Hex' | 'Bytes' = 'Hex'>( options: randomPrivateKey.Options<as> = {}, ): randomPrivateKey.ReturnType<as> { const { as = 'Hex' } = options const bytes = secp256k1.utils.randomPrivateKey() if (as === 'Hex') return Hex.fromBytes(bytes) as never return bytes as never } export declare namespace randomPrivateKey { type Options<as extends 'Hex' | 'Bytes' = 'Hex'> = { /** * Format of the returned private key. * @default 'Hex' */ as?: as | 'Hex' | 'Bytes' | undefined } type ReturnType<as extends 'Hex' | 'Bytes'> = | (as extends 'Bytes' ? Bytes.Bytes : never) | (as extends 'Hex' ? Hex.Hex : never) type ErrorType = Hex.fromBytes.ErrorType | Errors.GlobalErrorType } /** * Recovers the signing address from the signed payload and signature. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const signature = Secp256k1.sign({ payload: '0xdeadbeef', privateKey: '0x...' }) * * const address = Secp256k1.recoverAddress({ // [!code focus] * payload: '0xdeadbeef', // [!code focus] * signature, // [!code focus] * }) // [!code focus] * ``` * * @param options - The recovery options. * @returns The recovered address. */ export function recoverAddress( options: recoverAddress.Options, ): recoverAddress.ReturnType { return Address.fromPublicKey(recoverPublicKey(options)) } export declare namespace recoverAddress { type Options = { /** Payload that was signed. */ payload: Hex.Hex | Bytes.Bytes /** Signature of the payload. */ signature: Signature.Signature } type ReturnType = Address.Address type ErrorType = | Address.fromPublicKey.ErrorType | recoverPublicKey.ErrorType | Errors.GlobalErrorType } /** * Recovers the signing public key from the signed payload and signature. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const signature = Secp256k1.sign({ payload: '0xdeadbeef', privateKey: '0x...' }) * * const publicKey = Secp256k1.recoverPublicKey({ // [!code focus] * payload: '0xdeadbeef', // [!code focus] * signature, // [!code focus] * }) // [!code focus] * ``` * * @param options - The recovery options. * @returns The recovered public key. */ export function recoverPublicKey( options: recoverPublicKey.Options, ): PublicKey.PublicKey { const { payload, signature } = options const { r, s, yParity } = signature const signature_ = new secp256k1.Signature( BigInt(r), BigInt(s), ).addRecoveryBit(yParity) const point = signature_.recoverPublicKey(Hex.from(payload).substring(2)) return PublicKey.from(point) } export declare namespace recoverPublicKey { type Options = { /** Payload that was signed. */ payload: Hex.Hex | Bytes.Bytes /** Signature of the payload. */ signature: Signature.Signature } type ErrorType = | PublicKey.from.ErrorType | Hex.from.ErrorType | Errors.GlobalErrorType } /** * Signs the payload with the provided private key. * * @example * ```ts twoslash * import { Secp256k1 } from 'ox' * * const signature = Secp256k1.sign({ // [!code focus] * payload: '0xdeadbeef', // [!code focus] * privateKey: '0x...' // [!code focus] * }) // [!code focus] * ``` * * @param options - The signing options. * @returns The ECDSA {@link ox#Signature.Signature}. */ export function sign(options: sign.Options): Signature.Signature { const { extraEntropy = Entropy.extraEntropy, hash, payload, privateKey, } = options const { r, s, recovery } = secp256k1.sign( Bytes.from(payload), Bytes.from(privateKey), { extraEntropy: typeof extraEntropy === 'boolean' ? extraEntropy : Hex.from(extraEntropy).slice(2), lowS: true, ...(hash ? { prehash: true } : {}), }, ) return { r, s, yParity: recovery, } } export declare namespace sign { type Options = { /** * Extra entropy to add to the signing process. Setting to `false` will disable it. * @default true */ extraEntropy?: boolean | Hex.Hex | Bytes.Bytes | undefined /** * If set to `true`, the payload will be hashed (sha256) before being signed. */ hash?: boolean | undefined /** * Payload to sign. */ payload: Hex.Hex | Bytes.Bytes /** * ECDSA private key. */ privateKey: Hex.Hex | Bytes.Bytes } type ErrorType = Bytes.from.ErrorType | Errors.GlobalErrorType } /** * Verifies a payload was signed by the provided address. * * @example * ### Verify with Ethereum Address * * ```ts twoslash * import { Secp256k1 } from 'ox' * * const signature = Secp256k1.sign({ payload: '0xdeadbeef', privateKey: '0x...' }) * * const verified = Secp256k1.verify({ // [!code focus] * address: '0xf39fd6e51aad88f6f4ce6ab8827279cfffb92266', // [!code focus] * payload: '0xdeadbeef', // [!code focus] * signature, // [!code focus] * }) // [!code focus] * ``` * * @example * ### Verify with Public Key * * ```ts twoslash * import { Secp256k1 } from 'ox' * * const privateKey = '0x...' * const publicKey = Secp256k1.getPublicKey({ privateKey }) * const signature = Secp256k1.sign({ payload: '0xdeadbeef', privateKey }) * * const verified = Secp256k1.verify({ // [!code focus] * publicKey, // [!code focus] * payload: '0xdeadbeef', // [!code focus] * signature, // [!code focus] * }) // [!code focus] * ``` * * @param options - The verification options. * @returns Whether the payload was signed by the provided address. */ export function verify(options: verify.Options): boolean { const { address, hash, payload, publicKey, signature } = options if (address) return Address.isEqual(address, recoverAddress({ payload, signature })) return secp256k1.verify( signature, Bytes.from(payload), PublicKey.toBytes(publicKey), ...(hash ? [{ prehash: true, lowS: true }] : []), ) } export declare namespace verify { type Options = { /** If set to `true`, the payload will be hashed (sha256) before being verified. */ hash?: boolean | undefined /** Payload that was signed. */ payload: Hex.Hex | Bytes.Bytes } & OneOf< | { /** Address that signed the payload. */ address: Address.Address /** Signature of the payload. */ signature: Signature.Signature } | { /** Public key that signed the payload. */ publicKey: PublicKey.PublicKey<boolean> /** Signature of the payload. */ signature: Signature.Signature<false> } > type ErrorType = Errors.GlobalErrorType }