PHP WebShell

Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/coins

Просмотр файла: abstractEthCoin.js

"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 * as should from '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();\n      };\n\n      before(function () {\n        bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n        bitgo.initializeTestVars();\n        basecoin = bitgo.coin(coinName);\n        coin = coins.get(coinName);\n      });\n\n      describe('Is valid address', () => {\n        it('Should find valid addresses to be valid', () => {\n          basecoin.isValidAddress('0x2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(true);\n          basecoin.isValidAddress('0x2af9152FC4afd89A8124731BdFb8710c8751f3eD').should.equal(true);\n        });\n\n        it('Should find invalid addresses to be invalid', () => {\n          basecoin.isValidAddress('0x2af9152fc4afd89a8124731bdfb8710c8751f3edd').should.equal(false);\n          basecoin.isValidAddress('0x2af9152fc4afd89a8124731bdfb8710c8751f3e').should.equal(false);\n          basecoin.isValidAddress('2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(false);\n          basecoin.isValidAddress('notanaddress').should.equal(false);\n          basecoin.isValidAddress('not an address').should.equal(false);\n          basecoin.isValidAddress('3KgL6DTUb6gEoqSwMMJzyf96ekH8oZtWtZ').should.equal(false);\n        });\n\n        xit('Should not throw when verifying valid addresses', function () {\n          // FIXME(BG-43225): not implemented\n        });\n\n        xit('Should throw when verifying invalid addresses', function () {\n          // FIXME(BG-43225): not implemented\n        });\n      });\n\n      describe('Is valid pub', () => {\n        it('Should find valid pubs to be valid', () => {\n          basecoin\n            .isValidPub(\n              'xpub661MyMwAqRbcF9Nc7TbBo1rZAagiWEVPWKbDKThNG8zqjk76HAKLkaSbTn6dK2dQPfuD7xjicxCZVWvj67fP5nQ9W7QURmoMVAX8m6jZsGp'\n            )\n            .should.equal(true);\n          basecoin\n            .isValidPub(\n              '04614C070C6D1C18A6A2D6EE2BBBE1FF291A0ABA8ED6B55023C03BE42583AC23A743BCB5EF9DB59E14FD7025A9A5D93C6BA89EEFEB40215BF24933D4F2935D14CB'\n            )\n            .should.equal(true);\n          basecoin.isValidPub('034f355bdcb7cc0af728ef3cceb9615d90684bb5b2ca5f859ab0f0b704075871aa').should.equal(true);\n        });\n\n        it('Should find invalid pubs to be invalid', () => {\n          basecoin.isValidPub('0x2af9152fc4afd89a8124731bdfb8710c8751f3e').should.equal(false);\n          basecoin.isValidPub('0x2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(false);\n          basecoin.isValidPub('2af9152fc4afd89a8124731bdfb8710c8751f3ed').should.equal(false);\n          basecoin.isValidPub('notapub').should.equal(false);\n          basecoin.isValidPub('not a pub').should.equal(false);\n          basecoin.isValidPub('3KgL6DTUb6gEoqSwMMJzyf96ekH8oZtWtZ').should.equal(false);\n        });\n      });\n\n      describe('Generate keypair', () => {\n        it('Should generate valid keypair without seed', () => {\n          const { pub, prv } = basecoin.generateKeyPair();\n          basecoin.isValidPub(pub).should.equal(true);\n          const bitgoKey = bip32.fromBase58(prv);\n          basecoin.isValidPub(bitgoKey.neutered().toBase58()).should.equal(true);\n        });\n\n        it('Should generate valid keypair with seed', () => {\n          const seed = Buffer.from('c3b09c24731be2851b641d9d5b3f60fa129695c24071768d15654bea207b7bb6', 'hex');\n          const { pub, prv } = basecoin.generateKeyPair(seed);\n          basecoin.isValidPub(pub).should.equal(true);\n          const bitgoKey = bip32.fromBase58(prv);\n          basecoin.isValidPub(bitgoKey.neutered().toBase58()).should.equal(true);\n        });\n      });\n\n      describe('Sign transaction:', () => {\n        const xprv =\n          'xprv9s21ZrQH143K3D8TXfvAJgHVfTEeQNW5Ys9wZtnUZkqPzFzSjbEJrWC1vZ4GnXCvR7rQL2UFX3RSuYeU9MrERm1XBvACow7c36vnz5iYyj2';\n\n        it('should sign transaction internally', async function () {\n          const key = new Eth.KeyPair({ prv: xprv });\n          const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';\n          const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';\n          const amount = '100000';\n          const inputExpireTime = Math.floor(new Date().getTime() / 1000);\n          const inputSequenceId = 1;\n\n          const unsignedTransaction = await buildUnsignedTransaction({\n            destination,\n            contractAddress,\n            amount,\n            expireTime: inputExpireTime,\n            contractSequenceId: inputSequenceId,\n          });\n\n          const tx = await basecoin.signTransaction({\n            prv: key.getKeys().prv,\n            txPrebuild: {\n              txHex: unsignedTransaction.toBroadcastFormat(),\n            },\n          });\n\n          const txBuilder = basecoin.getTransactionBuilder();\n          txBuilder.from(tx.halfSigned.txHex);\n          const transaction = await txBuilder.build();\n          const txJson = transaction.toJson();\n          txJson.to.should.equal(contractAddress);\n\n          let decodedData;\n          let recipient;\n          let value;\n          let data;\n          let expireTime;\n          let sequenceId;\n          if (coin instanceof ContractAddressDefinedToken) {\n            decodedData = ethAbi.rawDecode(sendMultisigTokenTypes, Buffer.from(txJson.data.slice(10), 'hex'));\n            [recipient, value /* tokenContractAddress */, , expireTime, sequenceId] = decodedData;\n            data = Buffer.from('');\n          } else {\n            decodedData = ethAbi.rawDecode(sendMultisigTypes, Buffer.from(txJson.data.slice(10), 'hex'));\n            [recipient, value, data, expireTime, sequenceId] = decodedData;\n          }\n          ethUtil.addHexPrefix(recipient).should.equal(destination);\n          value.toString(10).should.equal(amount);\n          inputExpireTime.should.equal(parseInt(expireTime.toString('hex'), 16));\n          inputSequenceId.should.equal(parseInt(sequenceId.toString('hex'), 16));\n          data.length.should.equal(0);\n\n          const recoveredAddress = recoverSigner(transaction);\n          recoveredAddress.should.equal(key.getAddress());\n        });\n\n        it('should sign transaction internally with an xprv', async function () {\n          const key = new Eth.KeyPair({ prv: xprv });\n          const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';\n          const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';\n          const amount = '100000';\n          const inputExpireTime = Math.floor(new Date().getTime() / 1000);\n          const inputSequenceId = 1;\n\n          const unsignedTransaction = await buildUnsignedTransaction({\n            destination,\n            contractAddress,\n            amount,\n            expireTime: inputExpireTime,\n            contractSequenceId: inputSequenceId,\n          });\n\n          const tx = await basecoin.signTransaction({\n            prv: xprv,\n            txPrebuild: {\n              txHex: unsignedTransaction.toBroadcastFormat(),\n            },\n          });\n\n          const txBuilder = basecoin.getTransactionBuilder();\n          txBuilder.from(tx.halfSigned.txHex);\n          const transaction = await txBuilder.build();\n          const txJson = transaction.toJson();\n          txJson.to.should.equal(contractAddress);\n\n          let decodedData;\n          let recipient;\n          let value;\n          let data;\n          let expireTime;\n          let sequenceId;\n          if (coin instanceof ContractAddressDefinedToken) {\n            decodedData = ethAbi.rawDecode(sendMultisigTokenTypes, Buffer.from(txJson.data.slice(10), 'hex'));\n            [recipient, value /* tokenContractAddress */, , expireTime, sequenceId] = decodedData;\n            data = Buffer.from('');\n          } else {\n            decodedData = ethAbi.rawDecode(sendMultisigTypes, Buffer.from(txJson.data.slice(10), 'hex'));\n            [recipient, value, data, expireTime, sequenceId] = decodedData;\n          }\n\n          ethUtil.addHexPrefix(recipient).should.equal(destination);\n          value.toString(10).should.equal(amount);\n          inputExpireTime.should.equal(parseInt(expireTime.toString('hex'), 16));\n          inputSequenceId.should.equal(parseInt(sequenceId.toString('hex'), 16));\n          data.length.should.equal(0);\n\n          const recoveredAddress = recoverSigner(transaction);\n          recoveredAddress.should.equal(key.getAddress());\n        });\n\n        it('should sign a half signed transaction', async function () {\n          const key = new Eth.KeyPair({ prv: xprv });\n          const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';\n          const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';\n          const amount = '100000';\n          const inputExpireTime = Math.floor(new Date().getTime() / 1000);\n          const inputSequenceId = 1;\n\n          const unsignedTransaction = await buildUnsignedTransaction({\n            destination,\n            contractAddress,\n            amount,\n            expireTime: inputExpireTime,\n            contractSequenceId: inputSequenceId,\n          });\n\n          const tx = await basecoin.signTransaction({\n            prv: key.getKeys().prv,\n            txPrebuild: {\n              txHex: unsignedTransaction.toBroadcastFormat(),\n            },\n          });\n\n          const fullySignedTx = await basecoin.signTransaction({\n            prv: key.getKeys().prv,\n            txPrebuild: {\n              txHex: tx.halfSigned.txHex,\n            },\n          });\n\n          fullySignedTx.halfSigned.recipients.length.should.equal(1);\n          fullySignedTx.halfSigned.recipients[0].address.should.equal(destination);\n          fullySignedTx.halfSigned.recipients[0].amount.should.equal(amount);\n\n          const txBuilder = basecoin.getTransactionBuilder();\n          txBuilder.from(fullySignedTx.halfSigned.txHex);\n          const transaction = await txBuilder.build();\n          const txJson = transaction.toJson();\n          txJson.to.should.equal(contractAddress);\n\n          let decodedData;\n          let recipient;\n          let value;\n          let data;\n          let expireTime;\n          let sequenceId;\n          if (coin instanceof ContractAddressDefinedToken) {\n            decodedData = ethAbi.rawDecode(sendMultisigTokenTypes, Buffer.from(txJson.data.slice(10), 'hex'));\n            [recipient, value /* tokenContractAddress */, , expireTime, sequenceId] = decodedData;\n            data = Buffer.from('');\n          } else {\n            decodedData = ethAbi.rawDecode(sendMultisigTypes, Buffer.from(txJson.data.slice(10), 'hex'));\n            [recipient, value, data, expireTime, sequenceId] = decodedData;\n          }\n\n          ethUtil.addHexPrefix(recipient).should.equal(destination);\n          value.toString(10).should.equal(amount);\n          inputExpireTime.should.equal(parseInt(expireTime.toString('hex'), 16));\n          inputSequenceId.should.equal(parseInt(sequenceId.toString('hex'), 16));\n          data.length.should.equal(0);\n\n          const recoveredAddress = recoverSigner(transaction);\n          recoveredAddress.should.equal(key.getAddress());\n        });\n\n        it('should fail to sign transaction with invalid tx hex', async function () {\n          const key = new Eth.KeyPair({ prv: xprv });\n          await basecoin\n            .signTransaction({\n              prv: key.getKeys().prv,\n              txPrebuild: {\n                txHex: '0xinvalid',\n              },\n            })\n            .should.be.rejected();\n        });\n      });\n\n      describe('Explain transaction:', () => {\n        const xprv =\n          'xprv9s21ZrQH143K3D8TXfvAJgHVfTEeQNW5Ys9wZtnUZkqPzFzSjbEJrWC1vZ4GnXCvR7rQL2UFX3RSuYeU9MrERm1XBvACow7c36vnz5iYyj2';\n\n        it('should fail if the params object is missing parameters', async function () {\n          const explainParams = {\n            feeInfo: { fee: 1 },\n            txHex: null,\n          };\n          await basecoin.explainTransaction(explainParams).should.be.rejectedWith('missing explain tx parameters');\n        });\n\n        it('explain an unsigned transfer transaction', async function () {\n          const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';\n          const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';\n\n          const unsignedTransaction = await buildUnsignedTransaction({\n            destination,\n            contractAddress,\n          });\n\n          const explainParams = {\n            halfSigned: {\n              txHex: unsignedTransaction.toBroadcastFormat(),\n            },\n            feeInfo: { fee: 1 },\n          };\n          const explanation = await basecoin.explainTransaction(explainParams);\n          should.exist(explanation.id);\n          // TODO check other fields once account-lib properly explains transaction\n        });\n\n        it('explain a signed transfer transaction', async function () {\n          const key = new Eth.KeyPair({ prv: xprv });\n          const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';\n          const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';\n\n          const unsignedTransaction = await buildUnsignedTransaction({\n            destination,\n            contractAddress,\n          });\n\n          const signedTx = await basecoin.signTransaction({\n            prv: key.getKeys().prv,\n            txPrebuild: {\n              txHex: unsignedTransaction.toBroadcastFormat(),\n            },\n          });\n\n          const explainParams = {\n            txHex: signedTx.halfSigned.txHex,\n            feeInfo: { fee: 1 },\n          };\n          const explanation = await basecoin.explainTransaction(explainParams);\n          should.exist(explanation.id);\n          // TODO check other fields once account-lib properly explains transaction\n        });\n      });\n    });\n  });\n});\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!