import { BIP32Interface } from 'bip32'; import * as bip39 from 'bip39'; import * as bip39custom from '../../blue_modules/bip39'; import * as BlueElectrum from '../../blue_modules/BlueElectrum'; import { LegacyWallet } from './legacy-wallet'; import { Transaction } from './types'; type AbstractHDWalletStatics = { derivationPath?: string; }; /** * @deprecated */ export class AbstractHDWallet extends LegacyWallet { static readonly type = 'abstract'; static readonly typeReadable = 'abstract'; // @ts-ignore: override public readonly type = AbstractHDWallet.type; // @ts-ignore: override public readonly typeReadable = AbstractHDWallet.typeReadable; next_free_address_index: number; next_free_change_address_index: number; internal_addresses_cache: Record; external_addresses_cache: Record; _xpub: string; usedAddresses: string[]; _address_to_wif_cache: Record; gap_limit: number; passphrase?: string; _node0?: BIP32Interface; _node1?: BIP32Interface; constructor() { super(); const Constructor = this.constructor as unknown as AbstractHDWalletStatics; this.next_free_address_index = 0; this.next_free_change_address_index = 0; this.internal_addresses_cache = {}; // index => address this.external_addresses_cache = {}; // index => address this._xpub = ''; // cache this.usedAddresses = []; this._address_to_wif_cache = {}; this.gap_limit = 20; this._derivationPath = Constructor.derivationPath; } getNextFreeAddressIndex(): number { return this.next_free_address_index; } getNextFreeChangeAddressIndex(): number { return this.next_free_change_address_index; } prepareForSerialization(): void { // deleting structures that cant be serialized delete this._node0; delete this._node1; } generate(): Promise { throw new Error('Not implemented'); } allowSend(): boolean { return false; } getTransactions(): Transaction[] { throw new Error('Not implemented'); } /** * @return {Buffer} wallet seed */ _getSeed(): Buffer { const mnemonic = this.secret; const passphrase = this.passphrase; return bip39.mnemonicToSeedSync(mnemonic, passphrase); } setSecret(newSecret: string): this { // first, checking if it's a SeedQR: try { // compact seedQR should be between 32 - 64 chars long in hex format if (newSecret.length === 64 || newSecret.length === 32) { // not supported as mobile scanners dont recognize such QRs at all. // nop } else if (newSecret.length === 96 || newSecret.length === 48) { // standard seedQR const wordlist = bip39.wordlists[bip39.getDefaultWordlist()]; const words = newSecret.match(/[\d]{4}/g); if (words) { newSecret = words.map(num => wordlist[parseInt(num, 10)]).join(' '); } } } catch (e) {} // end SeedQR this.secret = newSecret.trim().toLowerCase(); this.secret = this.secret.replace(/[^a-zA-Z0-9]/g, ' ').replace(/\s+/g, ' '); // Try to match words to the default bip39 wordlist and complete partial words const wordlist = bip39.wordlists[bip39.getDefaultWordlist()]; const lookupMap = wordlist.reduce((map, word) => { const prefix3 = word.substr(0, 3); const prefix4 = word.substr(0, 4); map.set(prefix3, !map.has(prefix3) ? word : false); map.set(prefix4, !map.has(prefix4) ? word : false); return map; }, new Map()); this.secret = this.secret .split(' ') .map(word => lookupMap.get(word) || word) .join(' '); return this; } setPassphrase(passphrase: string): void { this.passphrase = passphrase; } getPassphrase(): string | undefined { return this.passphrase; } /** * @return {Boolean} is mnemonic in `this.secret` valid */ validateMnemonic(): boolean { return bip39custom.validateMnemonic(this.secret); } /** * Derives from hierarchy, returns next free address * (the one that has no transactions). Looks for several, * gives up if none found, and returns the used one * * @return {Promise.} */ async getAddressAsync(): Promise { // looking for free external address let freeAddress = ''; let c; for (c = 0; c < this.gap_limit + 1; c++) { if (this.next_free_address_index + c < 0) continue; const address = this._getExternalAddressByIndex(this.next_free_address_index + c); this.external_addresses_cache[this.next_free_address_index + c] = address; // updating cache just for any case let txs = []; try { txs = await BlueElectrum.getTransactionsByAddress(address); } catch (Err: any) { console.warn('BlueElectrum.getTransactionsByAddress()', Err.message); } if (txs.length === 0) { // found free address freeAddress = address; this.next_free_address_index += c; // now points to _this one_ break; } } if (!freeAddress) { // could not find in cycle above, give up freeAddress = this._getExternalAddressByIndex(this.next_free_address_index + c); // we didnt check this one, maybe its free this.next_free_address_index += c; // now points to this one } this._address = freeAddress; return freeAddress; } /** * Derives from hierarchy, returns next free CHANGE address * (the one that has no transactions). Looks for several, * gives up if none found, and returns the used one * * @return {Promise.} */ async getChangeAddressAsync(): Promise { // looking for free internal address let freeAddress = ''; let c; for (c = 0; c < this.gap_limit + 1; c++) { if (this.next_free_change_address_index + c < 0) continue; const address = this._getInternalAddressByIndex(this.next_free_change_address_index + c); this.internal_addresses_cache[this.next_free_change_address_index + c] = address; // updating cache just for any case let txs = []; try { txs = await BlueElectrum.getTransactionsByAddress(address); } catch (Err: any) { console.warn('BlueElectrum.getTransactionsByAddress()', Err.message); } if (txs.length === 0) { // found free address freeAddress = address; this.next_free_change_address_index += c; // now points to _this one_ break; } } if (!freeAddress) { // could not find in cycle above, give up freeAddress = this._getInternalAddressByIndex(this.next_free_change_address_index + c); // we didnt check this one, maybe its free this.next_free_change_address_index += c; // now points to this one } this._address = freeAddress; return freeAddress; } /** * Should not be used in HD wallets * * @deprecated * @return {string} */ getAddress(): string | false { return this._address; } _getExternalWIFByIndex(index: number): string | false { throw new Error('Not implemented'); } _getInternalWIFByIndex(index: number): string | false { throw new Error('Not implemented'); } _getExternalAddressByIndex(index: number): string { throw new Error('Not implemented'); } _getInternalAddressByIndex(index: number): string { throw new Error('Not implemented'); } getXpub(): string { throw new Error('Not implemented'); } /** * Async function to fetch all transactions. Use getter to get actual txs. * Also, sets internals: * `this.internal_addresses_cache` * `this.external_addresses_cache` * * @returns {Promise} */ async fetchTransactions(): Promise { throw new Error('not implemented'); } /** * Given that `address` is in our HD hierarchy, try to find * corresponding WIF * * @param address {String} In our HD hierarchy * @return {String} WIF if found */ _getWifForAddress(address: string): string { if (this._address_to_wif_cache[address]) return this._address_to_wif_cache[address]; // cache hit // fast approach, first lets iterate over all addressess we have in cache for (const indexStr of Object.keys(this.internal_addresses_cache)) { const index = parseInt(indexStr, 10); if (this._getInternalAddressByIndex(index) === address) { return (this._address_to_wif_cache[address] = this._getInternalWIFByIndex(index)); } } for (const indexStr of Object.keys(this.external_addresses_cache)) { const index = parseInt(indexStr, 10); if (this._getExternalAddressByIndex(index) === address) { return (this._address_to_wif_cache[address] = this._getExternalWIFByIndex(index)); } } // no luck - lets iterate over all addresses we have up to first unused address index for (let c = 0; c <= this.next_free_change_address_index + this.gap_limit; c++) { const possibleAddress = this._getInternalAddressByIndex(c); if (possibleAddress === address) { return (this._address_to_wif_cache[address] = this._getInternalWIFByIndex(c)); } } for (let c = 0; c <= this.next_free_address_index + this.gap_limit; c++) { const possibleAddress = this._getExternalAddressByIndex(c); if (possibleAddress === address) { return (this._address_to_wif_cache[address] = this._getExternalWIFByIndex(c)); } } throw new Error('Could not find WIF for ' + address); } async fetchBalance(): Promise { throw new Error('Not implemented'); } /** * @inheritDoc */ async fetchUtxo(): Promise { throw new Error('Not implemented'); } _getDerivationPathByAddress(address: string): string | false { throw new Error('Not implemented'); } _getNodePubkeyByIndex(node: number, index: number): Buffer | undefined { throw new Error('Not implemented'); } /** * @returns {string} Root derivation path for wallet if any */ getDerivationPath() { return this._derivationPath; } /* * Set derivation path for the wallet * * @param {String} path - path */ setDerivationPath(path: string) { this._derivationPath = path; } }