ethers.js/src.ts/wallet/hdnode.ts

258 lines
7.8 KiB
TypeScript
Raw Normal View History

2018-06-13 22:39:39 +03:00
'use strict';
// See: https://github.com/bitcoin/bips/blob/master/bip-0032.mediawiki
// See: https://github.com/bitcoin/bips/blob/master/bip-0039.mediawiki
2018-06-15 11:18:17 +03:00
import { KeyPair, N } from './secp256k1';
2018-06-13 22:39:39 +03:00
2018-06-15 11:18:17 +03:00
import { getWord, getWordIndex } from './words';
2018-06-13 22:39:39 +03:00
2018-06-15 11:18:17 +03:00
import { arrayify, Arrayish, hexlify } from '../utils/convert';
2018-06-13 22:39:39 +03:00
import { bigNumberify } from '../utils/bignumber';
import { toUtf8Bytes, UnicodeNormalizationForm } from '../utils/utf8';
import { pbkdf2 } from '../utils/pbkdf2';
import { createSha512Hmac } from '../utils/hmac';
import { sha256 } from '../utils/sha2';
import * as errors from '../utils/errors';
2018-06-13 22:39:39 +03:00
// "Bitcoin seed"
var MasterSecret = toUtf8Bytes('Bitcoin seed');
var HardenedBit = 0x80000000;
// Returns a byte with the MSB bits set
function getUpperMask(bits: number): number {
return ((1 << bits) - 1) << (8 - bits);
}
// Returns a byte with the LSB bits set
function getLowerMask(bits: number): number {
return (1 << bits) - 1;
}
export class HDNode {
2018-06-15 11:18:17 +03:00
private readonly keyPair: KeyPair;
2018-06-13 22:39:39 +03:00
readonly privateKey: string;
readonly publicKey: string;
readonly mnemonic: string;
readonly path: string;
readonly chainCode: string;
readonly index: number;
readonly depth: number;
// @TODO: Private constructor?
2018-06-15 11:18:17 +03:00
constructor(keyPair: KeyPair, chainCode: Uint8Array, index: number, depth: number, mnemonic: string, path: string) {
errors.checkNew(this, HDNode);
2018-06-13 22:39:39 +03:00
this.keyPair = keyPair;
2018-06-15 11:18:17 +03:00
this.privateKey = keyPair.privateKey;
this.publicKey = keyPair.compressedPublicKey;
2018-06-13 22:39:39 +03:00
this.chainCode = hexlify(chainCode);
this.index = index;
this.depth = depth;
this.mnemonic = mnemonic;
this.path = path;
}
private _derive(index: number): HDNode {
// Public parent key -> public child key
if (!this.privateKey) {
if (index >= HardenedBit) { throw new Error('cannot derive child of neutered node'); }
throw new Error('not implemented');
}
var data = new Uint8Array(37);
// Base path
var mnemonic = this.mnemonic;
var path = this.path;
if (path) { path += '/' + index; }
if (index & HardenedBit) {
// Data = 0x00 || ser_256(k_par)
data.set(arrayify(this.privateKey), 1);
// Hardened path
if (path) { path += "'"; }
} else {
// Data = ser_p(point(k_par))
2018-06-15 11:18:17 +03:00
data.set(this.keyPair.publicKeyBytes);
2018-06-13 22:39:39 +03:00
}
// Data += ser_32(i)
for (var i = 24; i >= 0; i -= 8) { data[33 + (i >> 3)] = ((index >> (24 - i)) & 0xff); }
var I = arrayify(createSha512Hmac(this.chainCode).update(data).digest());
var IL = bigNumberify(I.slice(0, 32));
var IR = I.slice(32);
2018-06-15 11:18:17 +03:00
var ki = IL.add(this.keyPair.privateKey).mod(N);
2018-06-13 22:39:39 +03:00
2018-06-15 11:18:17 +03:00
return new HDNode(new KeyPair(arrayify(ki)), IR, index, this.depth + 1, mnemonic, path);
2018-06-13 22:39:39 +03:00
}
derivePath(path: string): HDNode {
var components = path.split('/');
if (components.length === 0 || (components[0] === 'm' && this.depth !== 0)) {
throw new Error('invalid path');
}
if (components[0] === 'm') { components.shift(); }
var result: HDNode = this;
for (var i = 0; i < components.length; i++) {
var component = components[i];
if (component.match(/^[0-9]+'$/)) {
var index = parseInt(component.substring(0, component.length - 1));
if (index >= HardenedBit) { throw new Error('invalid path index - ' + component); }
result = result._derive(HardenedBit + index);
} else if (component.match(/^[0-9]+$/)) {
var index = parseInt(component);
if (index >= HardenedBit) { throw new Error('invalid path index - ' + component); }
result = result._derive(index);
} else {
throw new Error('invlaid path component - ' + component);
}
}
return result;
}
}
2018-06-15 11:18:17 +03:00
function _fromSeed(seed: Arrayish, mnemonic: string): HDNode {
let seedArray: Uint8Array = arrayify(seed);
if (seedArray.length < 16 || seedArray.length > 64) { throw new Error('invalid seed'); }
2018-06-13 22:39:39 +03:00
2018-06-15 11:18:17 +03:00
var I: Uint8Array = arrayify(createSha512Hmac(MasterSecret).update(seedArray).digest());
2018-06-13 22:39:39 +03:00
2018-06-15 11:18:17 +03:00
return new HDNode(new KeyPair(I.slice(0, 32)), I.slice(32), 0, 0, mnemonic, 'm');
2018-06-13 22:39:39 +03:00
}
export function fromMnemonic(mnemonic: string): HDNode {
// Check that the checksum s valid (will throw an error)
mnemonicToEntropy(mnemonic);
return _fromSeed(mnemonicToSeed(mnemonic), mnemonic);
}
2018-06-15 11:18:17 +03:00
export function fromSeed(seed: Arrayish): HDNode {
2018-06-13 22:39:39 +03:00
return _fromSeed(seed, null);
}
export function mnemonicToSeed(mnemonic: string, password?: string): string {
if (!password) {
password = '';
} else if (password.normalize) {
password = password.normalize('NFKD');
} else {
for (var i = 0; i < password.length; i++) {
var c = password.charCodeAt(i);
if (c < 32 || c > 127) { throw new Error('passwords with non-ASCII characters not supported in this environment'); }
}
}
var salt = toUtf8Bytes('mnemonic' + password, UnicodeNormalizationForm.NFKD);
return hexlify(pbkdf2(toUtf8Bytes(mnemonic, UnicodeNormalizationForm.NFKD), salt, 2048, 64, createSha512Hmac));
}
export function mnemonicToEntropy(mnemonic: string): string {
var words = mnemonic.toLowerCase().split(' ');
if ((words.length % 3) !== 0) { throw new Error('invalid mnemonic'); }
var entropy = arrayify(new Uint8Array(Math.ceil(11 * words.length / 8)));
var offset = 0;
for (var i = 0; i < words.length; i++) {
2018-06-15 11:18:17 +03:00
var index = getWordIndex(words[i]);
2018-06-13 22:39:39 +03:00
if (index === -1) { throw new Error('invalid mnemonic'); }
for (var bit = 0; bit < 11; bit++) {
if (index & (1 << (10 - bit))) {
entropy[offset >> 3] |= (1 << (7 - (offset % 8)));
}
offset++;
}
}
var entropyBits = 32 * words.length / 3;
var checksumBits = words.length / 3;
var checksumMask = getUpperMask(checksumBits);
var checksum = arrayify(sha256(entropy.slice(0, entropyBits / 8)))[0];
checksum &= checksumMask;
if (checksum !== (entropy[entropy.length - 1] & checksumMask)) {
throw new Error('invalid checksum');
}
return hexlify(entropy.slice(0, entropyBits / 8));
}
2018-06-15 11:18:17 +03:00
export function entropyToMnemonic(entropy: Arrayish): string {
2018-06-13 22:39:39 +03:00
entropy = arrayify(entropy);
if ((entropy.length % 4) !== 0 || entropy.length < 16 || entropy.length > 32) {
throw new Error('invalid entropy');
}
2018-06-15 11:18:17 +03:00
var indices: Array<number> = [ 0 ];
2018-06-13 22:39:39 +03:00
var remainingBits = 11;
for (var i = 0; i < entropy.length; i++) {
// Consume the whole byte (with still more to go)
if (remainingBits > 8) {
2018-06-15 11:18:17 +03:00
indices[indices.length - 1] <<= 8;
indices[indices.length - 1] |= entropy[i];
2018-06-13 22:39:39 +03:00
remainingBits -= 8;
// This byte will complete an 11-bit index
} else {
2018-06-15 11:18:17 +03:00
indices[indices.length - 1] <<= remainingBits;
indices[indices.length - 1] |= entropy[i] >> (8 - remainingBits);
2018-06-13 22:39:39 +03:00
// Start the next word
2018-06-15 11:18:17 +03:00
indices.push(entropy[i] & getLowerMask(8 - remainingBits));
2018-06-13 22:39:39 +03:00
remainingBits += 3;
}
}
// Compute the checksum bits
var checksum = arrayify(sha256(entropy))[0];
var checksumBits = entropy.length / 4;
checksum &= getUpperMask(checksumBits);
// Shift the checksum into the word indices
2018-06-15 11:18:17 +03:00
indices[indices.length - 1] <<= checksumBits;
indices[indices.length - 1] |= (checksum >> (8 - checksumBits));
2018-06-13 22:39:39 +03:00
2018-06-15 11:18:17 +03:00
return indices.map((index) => getWord(index)).join(' ');
2018-06-13 22:39:39 +03:00
}
export function isValidMnemonic(mnemonic: string): boolean {
try {
mnemonicToEntropy(mnemonic);
return true;
} catch (error) { }
return false;
}