UNPKG

bitgo

Version:
398 lines • 68.7 kB
"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(