123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- /* eslint-disable @typescript-eslint/camelcase */
- // Copyright 2017-2019 @polkadot/app-123code authors & contributors
- // This software may be modified and distributed under the terms
- // of the Apache-2.0 license. See the LICENSE file for details.
- import { EthereumAddress, EcdsaSignature } from '@polkadot/types/interfaces';
- import secp256k1 from 'secp256k1/elliptic';
- import { createType } from '@polkadot/types';
- import { assert, hexToU8a, stringToU8a, u8aToBuffer, u8aConcat } from '@polkadot/util';
- import { keccakAsHex, keccakAsU8a } from '@polkadot/util-crypto';
- interface RecoveredSignature {
- error: Error | null;
- ethereumAddress: EthereumAddress | null;
- signature: EcdsaSignature | null;
- }
- interface SignatureParts {
- recovery: number;
- signature: Buffer;
- }
- // converts an Ethereum address to a checksum representation
- export function addrToChecksum (_address: string): string {
- const address = _address.toLowerCase();
- const hash = keccakAsHex(address.substr(2)).substr(2);
- let result = '0x';
- for (let n = 0; n < 40; n++) {
- result = `${result}${
- parseInt(hash[n], 16) > 7
- ? address[n + 2].toUpperCase()
- : address[n + 2]
- }`;
- }
- return result;
- }
- // convert a give public key to an Ethereum address (the last 20 bytes of an _exapnded_ key keccack)
- export function publicToAddr (publicKey: Uint8Array): string {
- return addrToChecksum(`0x${keccakAsHex(publicKey).slice(-40)}`);
- }
- // hash a message for use in signature recovery, adding the standard Ethereum header
- export function hashMessage (message: string): Buffer {
- const expanded = stringToU8a(`\x19Ethereum Signed Message:\n${message.length.toString()}${message}`);
- const hashed = keccakAsU8a(expanded);
- return u8aToBuffer(hashed);
- }
- // split is 65-byte signature into the r, s (combined) and recovery number (derived from v)
- export function sigToParts (_signature: string): SignatureParts {
- const signature = hexToU8a(_signature);
- assert(signature.length === 65, `Invalid signature length, expected 65 found ${signature.length}`);
- let v = signature[64];
- if (v < 27) {
- v += 27;
- }
- const recovery = v - 27;
- assert(recovery === 0 || recovery === 1, 'Invalid signature v value');
- return {
- recovery,
- signature: u8aToBuffer(signature.slice(0, 64))
- };
- }
- // recover an address from a given message and a recover/signature combination
- export function recoverAddress (message: string, { recovery, signature }: SignatureParts): string {
- const msgHash = hashMessage(message);
- const senderPubKey = secp256k1.recover(msgHash, signature, recovery);
- return publicToAddr(
- secp256k1.publicKeyConvert(senderPubKey, false).slice(1)
- );
- }
- // recover an address from a signature JSON (as supplied by e.g. MyCrypto)
- export function recoverFromJSON (signatureJson: string | null): RecoveredSignature {
- try {
- const { msg, sig } = JSON.parse(signatureJson || '{}');
- if (!msg || !sig) {
- throw new Error('Invalid signature object');
- }
- const parts = sigToParts(sig);
- return {
- error: null,
- ethereumAddress: createType('EthereumAddress', recoverAddress(msg, parts)),
- signature: createType('EcdsaSignature', u8aConcat(parts.signature, new Uint8Array([parts.recovery])))
- };
- } catch (error) {
- console.error(error);
- return {
- error,
- ethereumAddress: null,
- signature: null
- };
- }
- }
|