bitgo
Version:
BitGo JavaScript SDK
398 lines • 68.7 kB
JavaScript
"use strict";
/**
* @prettier
*/
Object.defineProperty(exports, "__esModule", { value: true });
const _ = require("lodash");
const should = require("should");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const sdk_test_1 = require("@bitgo/sdk-test");
const bitgo_1 = require("../../../../src/bitgo");
const account_lib_1 = require("@bitgo/account-lib");
const ethAbi = require("ethereumjs-abi");
const ethUtil = require("ethereumjs-util");
const statics_1 = require("@bitgo/statics");
const sdk_core_1 = require("@bitgo/sdk-core");
describe('ETH-like coins', () => {
_.forEach(['tetc', 'tcelo', 'trbtc'], (coinName) => {
describe(`${coinName}`, () => {
let bitgo;
let basecoin;
let coin;
const sendMultisigTypes = ['address', 'uint256', 'bytes', 'uint256', 'uint256', 'bytes'];
const sendMultisigTokenTypes = ['address', 'uint256', 'address', 'uint256', 'uint256', 'bytes'];
const signatureSaltMap = {
native: {
tetc: 'ETC',
tcelo: 'CELO',
trbtc: 'RSK',
},
token: {
tetc: 'ETC-ERC20',
tcelo: 'CELO-ERC20',
trbtc: 'RSK-ERC20',
},
};
/**
* Get the operation hash that the user key signed
* @param tx The transaction to calculate operatino hash from
* @return The operation hash
*/
const getOperationHash = (tx) => {
const { data } = tx.toJson();
const { tokenContractAddress, expireTime, sequenceId, amount, to } = account_lib_1.Eth.Utils.decodeTransferData(data);
if (coin instanceof statics_1.ContractAddressDefinedToken) {
return ethAbi.soliditySHA3(...[
['string', 'address', 'uint', 'address', 'uint', 'uint'],
[
signatureSaltMap.token[coinName],
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore BG-34579: known compatibility issue with @types/ethereumjs-util
new ethUtil.BN(ethUtil.stripHexPrefix(to), 16),
amount,
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore BG-34579: known compatibility issue with @types/ethereumjs-util
new ethUtil.BN(ethUtil.stripHexPrefix(tokenContractAddress), 16),
expireTime,
sequenceId,
],
]);
}
else {
return ethAbi.soliditySHA3(...[
['string', 'address', 'uint', 'uint', 'uint'],
[
signatureSaltMap.native[coinName],
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore BG-34579: known compatibility issue with @types/ethereumjs-util
new ethUtil.BN(ethUtil.stripHexPrefix(to), 16),
amount,
expireTime,
sequenceId,
],
]);
}
};
/**
* Recover the signing address of a signature
* @param tx The transaction to recover a signer from
* @return The eth address of the signer
*/
const recoverSigner = function (tx) {
const { signature } = account_lib_1.Eth.Utils.decodeTransferData(tx.toJson().data);
const { v, r, s } = ethUtil.fromRpcSig(signature);
const operationHash = getOperationHash(tx);
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore known compatibility issue with @types/ethereumjs-util
const pubKeyBuffer = ethUtil.ecrecover(operationHash, v, r, s);
return ethUtil.bufferToHex(ethUtil.pubToAddress(ethUtil.importPublic(pubKeyBuffer)));
};
/**
* Build an unsigned account-lib multi-signature send transactino
* @param destination The destination address of the transaction
* @param contractAddress The address of the smart contract processing the transaction
* @param contractSequenceId The sequence id of the contract
* @param nonce The nonce of the sending address
* @param expireTime The expire time of the transaction
* @param amount The amount to send to the recipient
* @param gasPrice The gas price of the transaction
* @param gasLimit The gas limit of the transaction
*/
const buildUnsignedTransaction = async function ({ destination, contractAddress, contractSequenceId = 1, nonce = 0, expireTime = Math.floor(new Date().getTime() / 1000), amount = '100000', gasPrice = '10000', gasLimit = '20000', }) {
const txBuilder = (0, account_lib_1.getBuilder)(coinName);
txBuilder.type(sdk_core_1.TransactionType.Send);
txBuilder.fee({
fee: gasPrice,
gasLimit: gasLimit,
});
txBuilder.counter(nonce);
txBuilder.contract(contractAddress);
const transferBuilder = txBuilder.transfer();
transferBuilder
.coin(coinName)
.expirationTime(expireTime)
.amount(amount)
.to(destination)
.contractSequenceId(contractSequenceId);
return await txBuilder.build();
};
before(function () {
bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
bitgo.initializeTestVars();
basecoin = bitgo.coin(coinName);
coin = statics_1.coins.get(coinName);
});
describe('Is valid address', () => {
it('Should find valid addresses to be valid', () => {
basecoin.isValidAddress('0x2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(true);
basecoin.isValidAddress('0x2af9152FC4afd89A8124731BdFb8710c8751f3eD').should.equal(true);
});
it('Should find invalid addresses to be invalid', () => {
basecoin.isValidAddress('0x2af9152fc4afd89a8124731bdfb8710c8751f3edd').should.equal(false);
basecoin.isValidAddress('0x2af9152fc4afd89a8124731bdfb8710c8751f3e').should.equal(false);
basecoin.isValidAddress('2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(false);
basecoin.isValidAddress('notanaddress').should.equal(false);
basecoin.isValidAddress('not an address').should.equal(false);
basecoin.isValidAddress('3KgL6DTUb6gEoqSwMMJzyf96ekH8oZtWtZ').should.equal(false);
});
xit('Should not throw when verifying valid addresses', function () {
// FIXME(BG-43225): not implemented
});
xit('Should throw when verifying invalid addresses', function () {
// FIXME(BG-43225): not implemented
});
});
describe('Is valid pub', () => {
it('Should find valid pubs to be valid', () => {
basecoin
.isValidPub('xpub661MyMwAqRbcF9Nc7TbBo1rZAagiWEVPWKbDKThNG8zqjk76HAKLkaSbTn6dK2dQPfuD7xjicxCZVWvj67fP5nQ9W7QURmoMVAX8m6jZsGp')
.should.equal(true);
basecoin
.isValidPub('04614C070C6D1C18A6A2D6EE2BBBE1FF291A0ABA8ED6B55023C03BE42583AC23A743BCB5EF9DB59E14FD7025A9A5D93C6BA89EEFEB40215BF24933D4F2935D14CB')
.should.equal(true);
basecoin.isValidPub('034f355bdcb7cc0af728ef3cceb9615d90684bb5b2ca5f859ab0f0b704075871aa').should.equal(true);
});
it('Should find invalid pubs to be invalid', () => {
basecoin.isValidPub('0x2af9152fc4afd89a8124731bdfb8710c8751f3e').should.equal(false);
basecoin.isValidPub('0x2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(false);
basecoin.isValidPub('2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(false);
basecoin.isValidPub('notapub').should.equal(false);
basecoin.isValidPub('not a pub').should.equal(false);
basecoin.isValidPub('3KgL6DTUb6gEoqSwMMJzyf96ekH8oZtWtZ').should.equal(false);
});
});
describe('Generate keypair', () => {
it('Should generate valid keypair without seed', () => {
const { pub, prv } = basecoin.generateKeyPair();
basecoin.isValidPub(pub).should.equal(true);
const bitgoKey = utxo_lib_1.bip32.fromBase58(prv);
basecoin.isValidPub(bitgoKey.neutered().toBase58()).should.equal(true);
});
it('Should generate valid keypair with seed', () => {
const seed = Buffer.from('c3b09c24731be2851b641d9d5b3f60fa129695c24071768d15654bea207b7bb6', 'hex');
const { pub, prv } = basecoin.generateKeyPair(seed);
basecoin.isValidPub(pub).should.equal(true);
const bitgoKey = utxo_lib_1.bip32.fromBase58(prv);
basecoin.isValidPub(bitgoKey.neutered().toBase58()).should.equal(true);
});
});
describe('Sign transaction:', () => {
const xprv = 'xprv9s21ZrQH143K3D8TXfvAJgHVfTEeQNW5Ys9wZtnUZkqPzFzSjbEJrWC1vZ4GnXCvR7rQL2UFX3RSuYeU9MrERm1XBvACow7c36vnz5iYyj2';
it('should sign transaction internally', async function () {
const key = new account_lib_1.Eth.KeyPair({ prv: xprv });
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
const amount = '100000';
const inputExpireTime = Math.floor(new Date().getTime() / 1000);
const inputSequenceId = 1;
const unsignedTransaction = await buildUnsignedTransaction({
destination,
contractAddress,
amount,
expireTime: inputExpireTime,
contractSequenceId: inputSequenceId,
});
const tx = await basecoin.signTransaction({
prv: key.getKeys().prv,
txPrebuild: {
txHex: unsignedTransaction.toBroadcastFormat(),
},
});
const txBuilder = basecoin.getTransactionBuilder();
txBuilder.from(tx.halfSigned.txHex);
const transaction = await txBuilder.build();
const txJson = transaction.toJson();
txJson.to.should.equal(contractAddress);
let decodedData;
let recipient;
let value;
let data;
let expireTime;
let sequenceId;
if (coin instanceof statics_1.ContractAddressDefinedToken) {
decodedData = ethAbi.rawDecode(sendMultisigTokenTypes, Buffer.from(txJson.data.slice(10), 'hex'));
[recipient, value /* tokenContractAddress */, , expireTime, sequenceId] = decodedData;
data = Buffer.from('');
}
else {
decodedData = ethAbi.rawDecode(sendMultisigTypes, Buffer.from(txJson.data.slice(10), 'hex'));
[recipient, value, data, expireTime, sequenceId] = decodedData;
}
ethUtil.addHexPrefix(recipient).should.equal(destination);
value.toString(10).should.equal(amount);
inputExpireTime.should.equal(parseInt(expireTime.toString('hex'), 16));
inputSequenceId.should.equal(parseInt(sequenceId.toString('hex'), 16));
data.length.should.equal(0);
const recoveredAddress = recoverSigner(transaction);
recoveredAddress.should.equal(key.getAddress());
});
it('should sign transaction internally with an xprv', async function () {
const key = new account_lib_1.Eth.KeyPair({ prv: xprv });
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
const amount = '100000';
const inputExpireTime = Math.floor(new Date().getTime() / 1000);
const inputSequenceId = 1;
const unsignedTransaction = await buildUnsignedTransaction({
destination,
contractAddress,
amount,
expireTime: inputExpireTime,
contractSequenceId: inputSequenceId,
});
const tx = await basecoin.signTransaction({
prv: xprv,
txPrebuild: {
txHex: unsignedTransaction.toBroadcastFormat(),
},
});
const txBuilder = basecoin.getTransactionBuilder();
txBuilder.from(tx.halfSigned.txHex);
const transaction = await txBuilder.build();
const txJson = transaction.toJson();
txJson.to.should.equal(contractAddress);
let decodedData;
let recipient;
let value;
let data;
let expireTime;
let sequenceId;
if (coin instanceof statics_1.ContractAddressDefinedToken) {
decodedData = ethAbi.rawDecode(sendMultisigTokenTypes, Buffer.from(txJson.data.slice(10), 'hex'));
[recipient, value /* tokenContractAddress */, , expireTime, sequenceId] = decodedData;
data = Buffer.from('');
}
else {
decodedData = ethAbi.rawDecode(sendMultisigTypes, Buffer.from(txJson.data.slice(10), 'hex'));
[recipient, value, data, expireTime, sequenceId] = decodedData;
}
ethUtil.addHexPrefix(recipient).should.equal(destination);
value.toString(10).should.equal(amount);
inputExpireTime.should.equal(parseInt(expireTime.toString('hex'), 16));
inputSequenceId.should.equal(parseInt(sequenceId.toString('hex'), 16));
data.length.should.equal(0);
const recoveredAddress = recoverSigner(transaction);
recoveredAddress.should.equal(key.getAddress());
});
it('should sign a half signed transaction', async function () {
const key = new account_lib_1.Eth.KeyPair({ prv: xprv });
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
const amount = '100000';
const inputExpireTime = Math.floor(new Date().getTime() / 1000);
const inputSequenceId = 1;
const unsignedTransaction = await buildUnsignedTransaction({
destination,
contractAddress,
amount,
expireTime: inputExpireTime,
contractSequenceId: inputSequenceId,
});
const tx = await basecoin.signTransaction({
prv: key.getKeys().prv,
txPrebuild: {
txHex: unsignedTransaction.toBroadcastFormat(),
},
});
const fullySignedTx = await basecoin.signTransaction({
prv: key.getKeys().prv,
txPrebuild: {
txHex: tx.halfSigned.txHex,
},
});
fullySignedTx.halfSigned.recipients.length.should.equal(1);
fullySignedTx.halfSigned.recipients[0].address.should.equal(destination);
fullySignedTx.halfSigned.recipients[0].amount.should.equal(amount);
const txBuilder = basecoin.getTransactionBuilder();
txBuilder.from(fullySignedTx.halfSigned.txHex);
const transaction = await txBuilder.build();
const txJson = transaction.toJson();
txJson.to.should.equal(contractAddress);
let decodedData;
let recipient;
let value;
let data;
let expireTime;
let sequenceId;
if (coin instanceof statics_1.ContractAddressDefinedToken) {
decodedData = ethAbi.rawDecode(sendMultisigTokenTypes, Buffer.from(txJson.data.slice(10), 'hex'));
[recipient, value /* tokenContractAddress */, , expireTime, sequenceId] = decodedData;
data = Buffer.from('');
}
else {
decodedData = ethAbi.rawDecode(sendMultisigTypes, Buffer.from(txJson.data.slice(10), 'hex'));
[recipient, value, data, expireTime, sequenceId] = decodedData;
}
ethUtil.addHexPrefix(recipient).should.equal(destination);
value.toString(10).should.equal(amount);
inputExpireTime.should.equal(parseInt(expireTime.toString('hex'), 16));
inputSequenceId.should.equal(parseInt(sequenceId.toString('hex'), 16));
data.length.should.equal(0);
const recoveredAddress = recoverSigner(transaction);
recoveredAddress.should.equal(key.getAddress());
});
it('should fail to sign transaction with invalid tx hex', async function () {
const key = new account_lib_1.Eth.KeyPair({ prv: xprv });
await basecoin
.signTransaction({
prv: key.getKeys().prv,
txPrebuild: {
txHex: '0xinvalid',
},
})
.should.be.rejected();
});
});
describe('Explain transaction:', () => {
const xprv = 'xprv9s21ZrQH143K3D8TXfvAJgHVfTEeQNW5Ys9wZtnUZkqPzFzSjbEJrWC1vZ4GnXCvR7rQL2UFX3RSuYeU9MrERm1XBvACow7c36vnz5iYyj2';
it('should fail if the params object is missing parameters', async function () {
const explainParams = {
feeInfo: { fee: 1 },
txHex: null,
};
await basecoin.explainTransaction(explainParams).should.be.rejectedWith('missing explain tx parameters');
});
it('explain an unsigned transfer transaction', async function () {
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
const unsignedTransaction = await buildUnsignedTransaction({
destination,
contractAddress,
});
const explainParams = {
halfSigned: {
txHex: unsignedTransaction.toBroadcastFormat(),
},
feeInfo: { fee: 1 },
};
const explanation = await basecoin.explainTransaction(explainParams);
should.exist(explanation.id);
// TODO check other fields once account-lib properly explains transaction
});
it('explain a signed transfer transaction', async function () {
const key = new account_lib_1.Eth.KeyPair({ prv: xprv });
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
const unsignedTransaction = await buildUnsignedTransaction({
destination,
contractAddress,
});
const signedTx = await basecoin.signTransaction({
prv: key.getKeys().prv,
txPrebuild: {
txHex: unsignedTransaction.toBroadcastFormat(),
},
});
const explainParams = {
txHex: signedTx.halfSigned.txHex,
feeInfo: { fee: 1 },
};
const explanation = await basecoin.explainTransaction(explainParams);
should.exist(explanation.id);
// TODO check other fields once account-lib properly explains transaction
});
});
});
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractEthCoin.js","sourceRoot":"","sources":["../../../../../test/v2/unit/coins/abstractEthCoin.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\n\nimport * as _ from 'lodash';\nimport should = require('should');\nimport { bip32 } from '@bitgo/utxo-lib';\nimport { TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../../src/bitgo';\nimport { getBuilder, Eth } from '@bitgo/account-lib';\nimport * as ethAbi from 'ethereumjs-abi';\nimport * as ethUtil from 'ethereumjs-util';\nimport { coins, ContractAddressDefinedToken } from '@bitgo/statics';\nimport { BaseTransaction, TransactionType } from '@bitgo/sdk-core';\n\ndescribe('ETH-like coins', () => {\n  _.forEach(['tetc', 'tcelo', 'trbtc'], (coinName) => {\n    describe(`${coinName}`, () => {\n      let bitgo;\n      let basecoin;\n      let coin;\n\n      const sendMultisigTypes = ['address', 'uint256', 'bytes', 'uint256', 'uint256', 'bytes'];\n      const sendMultisigTokenTypes = ['address', 'uint256', 'address', 'uint256', 'uint256', 'bytes'];\n      const signatureSaltMap = {\n        native: {\n          tetc: 'ETC',\n          tcelo: 'CELO',\n          trbtc: 'RSK',\n        },\n        token: {\n          tetc: 'ETC-ERC20',\n          tcelo: 'CELO-ERC20',\n          trbtc: 'RSK-ERC20',\n        },\n      };\n\n      /**\n       * Get the operation hash that the user key signed\n       * @param tx The transaction to calculate operatino hash from\n       * @return The operation hash\n       */\n      const getOperationHash = (tx: BaseTransaction): string => {\n        const { data } = tx.toJson();\n        const { tokenContractAddress, expireTime, sequenceId, amount, to } = Eth.Utils.decodeTransferData(data);\n\n        if (coin instanceof ContractAddressDefinedToken) {\n          return ethAbi.soliditySHA3(\n            ...[\n              ['string', 'address', 'uint', 'address', 'uint', 'uint'],\n              [\n                signatureSaltMap.token[coinName],\n                // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n                // @ts-ignore BG-34579: known compatibility issue with @types/ethereumjs-util\n                new ethUtil.BN(ethUtil.stripHexPrefix(to), 16),\n                amount,\n                // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n                // @ts-ignore BG-34579: known compatibility issue with @types/ethereumjs-util\n                new ethUtil.BN(ethUtil.stripHexPrefix(tokenContractAddress), 16),\n                expireTime,\n                sequenceId,\n              ],\n            ]\n          );\n        } else {\n          return ethAbi.soliditySHA3(\n            ...[\n              ['string', 'address', 'uint', 'uint', 'uint'],\n              [\n                signatureSaltMap.native[coinName],\n                // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n                // @ts-ignore BG-34579: known compatibility issue with @types/ethereumjs-util\n                new ethUtil.BN(ethUtil.stripHexPrefix(to), 16),\n                amount,\n                expireTime,\n                sequenceId,\n              ],\n            ]\n          );\n        }\n      };\n\n      /**\n       * Recover the signing address of a signature\n       * @param tx The transaction to recover a signer from\n       * @return The eth address of the signer\n       */\n      const recoverSigner = function (tx: BaseTransaction) {\n        const { signature } = Eth.Utils.decodeTransferData(tx.toJson().data);\n        const { v, r, s } = ethUtil.fromRpcSig(signature);\n        const operationHash = getOperationHash(tx);\n        // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n        // @ts-ignore known compatibility issue with @types/ethereumjs-util\n        const pubKeyBuffer = ethUtil.ecrecover(operationHash, v, r, s);\n        return ethUtil.bufferToHex(ethUtil.pubToAddress(ethUtil.importPublic(pubKeyBuffer)));\n      };\n\n      /**\n       * Build an unsigned account-lib multi-signature send transactino\n       * @param destination The destination address of the transaction\n       * @param contractAddress The address of the smart contract processing the transaction\n       * @param contractSequenceId The sequence id of the contract\n       * @param nonce The nonce of the sending address\n       * @param expireTime The expire time of the transaction\n       * @param amount The amount to send to the recipient\n       * @param gasPrice The gas price of the transaction\n       * @param gasLimit The gas limit of the transaction\n       */\n      const buildUnsignedTransaction = async function ({\n        destination,\n        contractAddress,\n        contractSequenceId = 1,\n        nonce = 0,\n        expireTime = Math.floor(new Date().getTime() / 1000),\n        amount = '100000',\n        gasPrice = '10000',\n        gasLimit = '20000',\n      }) {\n        const txBuilder: Eth.TransactionBuilder = getBuilder(coinName) as Eth.TransactionBuilder;\n        txBuilder.type(TransactionType.Send);\n        txBuilder.fee({\n          fee: gasPrice,\n          gasLimit: gasLimit,\n        });\n        txBuilder.counter(nonce);\n        txBuilder.contract(contractAddress);\n        const transferBuilder = txBuilder.transfer() as Eth.TransferBuilder;\n\n        transferBuilder\n          .coin(coinName)\n          .expirationTime(expireTime)\n          .amount(amount)\n          .to(destination)\n          .contractSequenceId(contractSequenceId);\n\n        return await txBuilder.build(