92 lines
2.6 KiB
TypeScript
92 lines
2.6 KiB
TypeScript
|
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<string> {
|
||
|
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<string> {
|
||
|
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 signedMsg = JSON.stringify(
|
||
|
{
|
||
|
address: await this.getAddress(),
|
||
|
msg,
|
||
|
sig: addHexPrefix(combinedHex),
|
||
|
version: '2'
|
||
|
},
|
||
|
null,
|
||
|
2
|
||
|
);
|
||
|
resolve(signedMsg);
|
||
|
} catch (err) {
|
||
|
reject(err);
|
||
|
}
|
||
|
}
|
||
|
);
|
||
|
});
|
||
|
}
|
||
|
}
|