2021-09-28 12:32:06 +02:00
|
|
|
import Zemu from '@zondax/zemu';
|
|
|
|
|
import Eth from '@ledgerhq/hw-app-eth';
|
2021-11-22 14:39:36 +01:00
|
|
|
import {RLP} from "ethers/lib/utils";
|
2021-09-28 12:32:06 +02:00
|
|
|
|
|
|
|
|
const transactionUploadDelay = 60000;
|
|
|
|
|
|
|
|
|
|
async function waitForAppScreen(sim) {
|
|
|
|
|
await sim.waitUntilScreenIsNot(sim.getMainMenuSnapshot(), transactionUploadDelay);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const sim_options_nanos = {
|
|
|
|
|
model: 'nanos',
|
|
|
|
|
logging: true,
|
|
|
|
|
X11: true,
|
|
|
|
|
startDelay: 5000,
|
|
|
|
|
custom: '',
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const sim_options_nanox = {
|
|
|
|
|
model: 'nanox',
|
|
|
|
|
logging: true,
|
|
|
|
|
X11: true,
|
|
|
|
|
startDelay: 5000,
|
|
|
|
|
custom: '',
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const Resolve = require('path').resolve;
|
|
|
|
|
|
|
|
|
|
const NANOS_ELF_PATH = Resolve('elfs/ethereum_nanos.elf');
|
|
|
|
|
const NANOX_ELF_PATH = Resolve('elfs/ethereum_nanox.elf');
|
|
|
|
|
|
|
|
|
|
const NANOS_ETH_LIB = { "Ethereum": NANOS_ELF_PATH };
|
|
|
|
|
const NANOX_ETH_LIB = { "Ethereum": NANOX_ELF_PATH };
|
|
|
|
|
|
|
|
|
|
const NANOS_CLONE_ELF_PATH = Resolve("elfs/ethereum_classic_nanos.elf");
|
|
|
|
|
const NANOX_CLONE_ELF_PATH = Resolve("elfs/ethereum_classic_nanox.elf");
|
|
|
|
|
|
|
|
|
|
const TIMEOUT = 1000000;
|
|
|
|
|
|
2021-11-22 14:39:36 +01:00
|
|
|
// Generates a serializedTransaction from a rawHexTransaction copy pasted from etherscan.
|
|
|
|
|
function txFromEtherscan(rawTx) {
|
|
|
|
|
// Remove 0x prefix
|
|
|
|
|
rawTx = rawTx.slice(2);
|
|
|
|
|
|
|
|
|
|
let txType = rawTx.slice(0, 2);
|
|
|
|
|
if (txType == "02" || txType == "01") {
|
|
|
|
|
// Remove "02" prefix
|
|
|
|
|
rawTx = rawTx.slice(2);
|
|
|
|
|
} else {
|
|
|
|
|
txType = "";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let decoded = RLP.decode("0x" + rawTx);
|
|
|
|
|
if (txType != "") {
|
|
|
|
|
decoded = decoded.slice(0, decoded.length - 3); // remove v, r, s
|
|
|
|
|
} else {
|
|
|
|
|
decoded[decoded.length - 1] = "0x"; // empty
|
|
|
|
|
decoded[decoded.length - 2] = "0x"; // empty
|
|
|
|
|
decoded[decoded.length - 3] = "0x01"; // chainID 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Encode back the data, drop the '0x' prefix
|
|
|
|
|
let encoded = RLP.encode(decoded).slice(2);
|
|
|
|
|
|
|
|
|
|
// Don't forget to prepend the txtype
|
|
|
|
|
return txType + encoded;
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-28 12:32:06 +02:00
|
|
|
function zemu(device, func) {
|
|
|
|
|
return async () => {
|
|
|
|
|
jest.setTimeout(TIMEOUT);
|
|
|
|
|
let zemu_args;
|
|
|
|
|
let sim_options;
|
|
|
|
|
if(device === "nanos") {
|
|
|
|
|
zemu_args = [NANOS_ELF_PATH];
|
|
|
|
|
sim_options = sim_options_nanos;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
zemu_args = [NANOX_ELF_PATH];
|
|
|
|
|
sim_options = sim_options_nanox;
|
|
|
|
|
}
|
|
|
|
|
const sim = new Zemu(...zemu_args);
|
|
|
|
|
try {
|
|
|
|
|
await sim.start(sim_options);
|
|
|
|
|
const transport = await sim.getTransport();
|
|
|
|
|
await func(sim, new Eth(transport));
|
|
|
|
|
} finally {
|
|
|
|
|
await sim.close();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
|
zemu,
|
|
|
|
|
waitForAppScreen,
|
|
|
|
|
NANOS_ELF_PATH,
|
|
|
|
|
NANOX_ELF_PATH,
|
|
|
|
|
NANOS_ETH_LIB,
|
|
|
|
|
NANOX_ETH_LIB,
|
|
|
|
|
NANOS_CLONE_ELF_PATH,
|
|
|
|
|
NANOX_CLONE_ELF_PATH,
|
|
|
|
|
sim_options_nanos,
|
|
|
|
|
sim_options_nanox,
|
2021-11-22 14:39:36 +01:00
|
|
|
TIMEOUT,
|
|
|
|
|
txFromEtherscan,
|
2021-09-28 12:32:06 +02:00
|
|
|
}
|