UNPKG

@libp2p/crypto

Version:
94 lines (80 loc) 2.54 kB
import { InvalidParametersError } from '@libp2p/interface' import { fromString as uint8ArrayFromString } from 'uint8arrays/from-string' import randomBytes from '../../random-bytes.js' import webcrypto from '../../webcrypto/index.js' import * as utils from './utils.js' import type { JWKKeyPair } from '../interface.js' import type { Uint8ArrayList } from 'uint8arraylist' export { utils } export async function generateRSAKey (bits: number): Promise<JWKKeyPair> { const pair = await webcrypto.get().subtle.generateKey( { name: 'RSASSA-PKCS1-v1_5', modulusLength: bits, publicExponent: new Uint8Array([0x01, 0x00, 0x01]), hash: { name: 'SHA-256' } }, true, ['sign', 'verify'] ) const keys = await exportKey(pair) return { privateKey: keys[0], publicKey: keys[1] } } export { randomBytes as getRandomValues } export async function hashAndSign (key: JsonWebKey, msg: Uint8Array | Uint8ArrayList): Promise<Uint8Array> { const privateKey = await webcrypto.get().subtle.importKey( 'jwk', key, { name: 'RSASSA-PKCS1-v1_5', hash: { name: 'SHA-256' } }, false, ['sign'] ) const sig = await webcrypto.get().subtle.sign( { name: 'RSASSA-PKCS1-v1_5' }, privateKey, msg instanceof Uint8Array ? msg : msg.subarray() ) return new Uint8Array(sig, 0, sig.byteLength) } export async function hashAndVerify (key: JsonWebKey, sig: Uint8Array, msg: Uint8Array | Uint8ArrayList): Promise<boolean> { const publicKey = await webcrypto.get().subtle.importKey( 'jwk', key, { name: 'RSASSA-PKCS1-v1_5', hash: { name: 'SHA-256' } }, false, ['verify'] ) return webcrypto.get().subtle.verify( { name: 'RSASSA-PKCS1-v1_5' }, publicKey, sig, msg instanceof Uint8Array ? msg : msg.subarray() ) } async function exportKey (pair: CryptoKeyPair): Promise<[JsonWebKey, JsonWebKey]> { if (pair.privateKey == null || pair.publicKey == null) { throw new InvalidParametersError('Private and public key are required') } return Promise.all([ webcrypto.get().subtle.exportKey('jwk', pair.privateKey), webcrypto.get().subtle.exportKey('jwk', pair.publicKey) ]) } export function rsaKeySize (jwk: JsonWebKey): number { if (jwk.kty !== 'RSA') { throw new InvalidParametersError('invalid key type') } else if (jwk.n == null) { throw new InvalidParametersError('invalid key modulus') } const bytes = uint8ArrayFromString(jwk.n, 'base64url') return bytes.length * 8 }