import Ledger3 from 'vendor/ledger3'; import LedgerEth from 'vendor/ledger-eth'; import EthTx from 'ethereumjs-tx'; import { addHexPrefix, rlp } from 'ethereumjs-util'; import DeterministicWallet from './deterministic'; import { IWallet } from './IWallet'; import { RawTransaction } from 'libs/transaction'; export default class LedgerWallet extends DeterministicWallet implements IWallet { private ledger: any; private ethApp: any; constructor(address: string, dPath: string, index: number) { super(address, dPath, index); this.ledger = new Ledger3('w0w'); this.ethApp = new LedgerEth(this.ledger); } // modeled after // https://github.com/kvhnuke/etherwallet/blob/3f7ff809e5d02d7ea47db559adaca1c930025e24/app/scripts/uiFuncs.js#L58 public signRawTransaction(rawTx: RawTransaction): Promise { return new Promise((resolve, reject) => { const eTx = new EthTx({ ...rawTx, v: Buffer.from([rawTx.chainId]), r: 0, s: 0 }); this.ethApp.signTransaction( this.getPath(), rlp.encode(eTx.raw).toString('hex'), (result, error) => { if (error) { return reject(this.ethApp.getError(error)); } const txToSerialize = { ...rawTx, v: addHexPrefix(result.v), r: addHexPrefix(result.r), s: addHexPrefix(result.s) }; const serializedTx = new EthTx(txToSerialize) .serialize() .toString('hex'); resolve(addHexPrefix(serializedTx)); } ); }); } // modeled after // https://github.com/kvhnuke/etherwallet/blob/3f7ff809e5d02d7ea47db559adaca1c930025e24/app/scripts/controllers/signMsgCtrl.js#L53 public signMessage(msg: string): Promise { return new Promise((resolve, reject) => { const msgHex = Buffer.from(msg).toString('hex'); this.ethApp.signPersonalMessage_async( this.getPath(), msgHex, async (signed, error) => { if (error) { return reject(this.ethApp.getError(error)); } try { const combined = signed.r + signed.s + signed.v; const combinedHex = combined.toString('hex'); const signature = addHexPrefix(combinedHex); resolve(signature); } catch (err) { reject(err); } } ); }); } }