PHP WebShell

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

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

"use strict";
//
// Tests for a Pending Approval
//
// Copyright 2015, BitGo, Inc.  All Rights Reserved.
//
Object.defineProperty(exports, "__esModule", { value: true });
const _ = require("lodash");
const assert_1 = require("assert");
require("should");
const utxolib = require("@bitgo/utxo-lib");
const BitGoJS = require('../../src/index');
const TestBitGo = require('../lib/test_bitgo');
const TestUtil = require('./testutil');
const Q = require('q');
describe('PendingApproval', function () {
    let bitgo;
    let bitgoSharedKeyUser;
    let bitgoThirdUser;
    let sharedWallet;
    /**
     * There is a 0-limit policy on the shared wallet
     * Create a pending approval by attempting to send coins
     */
    const createTransactionPendingApproval = function () {
        return sharedWallet
            .sendCoins({
            address: TestBitGo.TEST_WALLET2_ADDRESS,
            amount: 0.0001 * 1e8,
            walletPassphrase: TestBitGo.TEST_PASSWORD,
            otp: bitgo.testUserOTP(),
            message: 'never gonna',
        })
            .then(function (result) {
            result.should.have.property('pendingApproval');
            return bitgo.pendingApprovals().get({ id: result.pendingApproval });
        });
    };
    /**
     * There is a 0-limit policy on the shared wallet
     * Create a pending approval by attempting to send to many
     */
    const createTransactionPendingApprovalToMultipleRecipients = function () {
        return sharedWallet
            .sendMany({
            recipients: [
                {
                    address: TestBitGo.TEST_WALLET3_ADDRESS,
                    amount: 0.0002 * 1e8,
                },
                {
                    address: TestBitGo.TEST_WALLET2_ADDRESS,
                    amount: 0.0001 * 1e8,
                },
                {
                    address: TestBitGo.TEST_SHARED_WALLET_CHANGE_ADDRESS,
                    amount: 0.0005 * 1e8,
                },
            ],
            walletPassphrase: TestBitGo.TEST_PASSWORD,
            otp: bitgo.testUserOTP(),
            message: 'never gonna',
        })
            .then(function (result) {
            result.should.have.property('pendingApproval');
            return bitgo.pendingApprovals().get({ id: result.pendingApproval });
        });
    };
    /**
     * Create a pending approval by attempting to add a user to the wallet
     */
    const createPolicyPendingApproval = function () {
        // it's ok to set up any tx limit since the daily limit is 0
        return sharedWallet
            .updatePolicyRule({
            action: {
                type: 'getApproval',
            },
            condition: {
                amount: 1e8 + Math.round(Math.random() * 1e8),
            },
            id: 'com.bitgo.limit.tx',
            type: 'transactionLimit',
        })
            .then(function (result) {
            result.should.have.property('pendingApproval');
            return bitgo.pendingApprovals().get({ id: result.pendingApproval.id });
        });
    };
    before(function () {
        bitgo = new TestBitGo();
        bitgo.initializeTestVars();
        bitgoSharedKeyUser = new TestBitGo();
        bitgoSharedKeyUser.initializeTestVars();
        bitgoThirdUser = new TestBitGo();
        bitgoThirdUser.initializeTestVars();
        return bitgo
            .authenticateTestUser(bitgo.testUserOTP())
            .then(function () {
            return bitgoSharedKeyUser.authenticate({
                username: TestBitGo.TEST_SHARED_KEY_USER,
                password: TestBitGo.TEST_SHARED_KEY_PASSWORD,
                otp: bitgo.testUserOTP(),
            });
        })
            .then(function () {
            return bitgo.unlock({ otp: bitgo.testUserOTP() });
        })
            .then(function () {
            return bitgoSharedKeyUser.unlock({ otp: bitgo.testUserOTP() });
        })
            .then(function () {
            return bitgo.wallets().get({ id: TestBitGo.TEST_SHARED_WALLET_ADDRESS });
        })
            .then(function (result) {
            sharedWallet = result;
        });
    });
    describe('Create and Get', function () {
        let pendingApproval;
        before(function () {
            return createTransactionPendingApproval().then(function (result) {
                pendingApproval = result;
            });
        });
        after(function () {
            return pendingApproval.reject();
        });
        it('arguments', function () {
            assert_1.strict.throws(function () {
                bitgo.pendingApprovals().get({}, 'invalid');
            });
            assert_1.strict.throws(function () {
                bitgo.pendingApprovals().get('invalid');
            });
            assert_1.strict.throws(function () {
                bitgo.pendingApprovals().get({ id: 54321 }, 'invalid');
            });
        });
        it('get property methods', function () {
            pendingApproval.id().should.eql(pendingApproval.pendingApproval.id);
            pendingApproval.ownerType().should.eql('wallet');
            pendingApproval.walletId().should.eql(sharedWallet.id());
            assert_1.strict.equal(pendingApproval.enterpriseId(), undefined);
            pendingApproval.state().should.eql('pending');
            pendingApproval.creator().should.eql(TestBitGo.TEST_USERID);
            pendingApproval.type().should.eql('transactionRequest');
            pendingApproval.info().transactionRequest.message.should.eql('never gonna');
            pendingApproval.info().transactionRequest.destinationAddress.should.eql(TestBitGo.TEST_WALLET2_ADDRESS);
        });
        it('get', function () {
            pendingApproval.get({}).then(function (result) {
                pendingApproval = result;
                pendingApproval.id().should.eql(pendingApproval.pendingApproval.id);
                pendingApproval.ownerType().should.eql('wallet');
                pendingApproval.walletId().should.eql(sharedWallet.id());
                assert_1.strict.equal(pendingApproval.enterpriseId(), undefined);
                pendingApproval.state().should.eql('pending');
                pendingApproval.creator().should.eql(TestBitGo.TEST_USERID);
                pendingApproval.type().should.eql('transactionRequest');
                pendingApproval.info().transactionRequest.destinationAddress.should.eql(TestBitGo.TEST_WALLET2_ADDRESS);
            });
        });
    });
    describe('Approve', function () {
        let pendingApproval;
        before(function () {
            return createTransactionPendingApproval().then(function (result) {
                pendingApproval = result;
            });
        });
        after(function () {
            return pendingApproval.reject();
        });
        it('arguments', function () {
            assert_1.strict.throws(function () {
                pendingApproval.approve({}, 'invalid');
            });
            assert_1.strict.throws(function () {
                pendingApproval.approve('invalid');
            });
        });
        it('error when self approving', function () {
            return createPolicyPendingApproval()
                .then(function (pendingApproval) {
                return pendingApproval.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });
            })
                .catch(function (err) {
                err.message.should.containEql('cannot approve by self');
                return pendingApproval.reject();
            });
        });
        it('can approve when it does not require tx signing', function () {
            return createPolicyPendingApproval().then(function (pendingApproval) {
                return bitgoSharedKeyUser
                    .pendingApprovals()
                    .get({ id: pendingApproval.id() })
                    .then(function (result) {
                    return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });
                })
                    .then(function (result) {
                    result.state.should.eql('approved');
                    result.info.policyRuleRequest.update.condition.amount.should.eql(pendingApproval.info().policyRuleRequest.update.condition.amount);
                });
            });
        });
        it('can approve when tx does not require reconstruction', function () {
            return createTransactionPendingApproval().then(function (pendingApproval) {
                return bitgoSharedKeyUser
                    .pendingApprovals()
                    .get({ id: pendingApproval.id() })
                    .then(function (result) {
                    return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });
                })
                    .then(function (result) {
                    result.state.should.eql('approved');
                });
            });
        });
        it('error when it does require tx signing but wrong passphrase', function () {
            return createTransactionPendingApproval()
                .then(function (pendingApproval) {
                return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });
            })
                .then(function (result) {
                return result.approve({ walletPassphrase: 'abcdef', otp: bitgo.testUserOTP() });
            })
                .catch(function (err) {
                err.message.should.containEql('Unable to decrypt user keychain');
                return pendingApproval.reject();
            });
        });
        it('can approve when it does require tx signing', function () {
            return createTransactionPendingApproval()
                .then(function (pendingApproval) {
                return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });
            })
                .then(function (result) {
                return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });
            })
                .then(function (result) {
                result.state.should.eql('approved');
            });
        });
        it('cannot approve when transaction needs reconstructing', function () {
            let approvals = [];
            return Q.all([createTransactionPendingApproval(), createTransactionPendingApproval()])
                .spread(function (approval1, approval2) {
                approvals = [approval1, approval2];
                return bitgoSharedKeyUser.pendingApprovals().get({ id: approval1.id() });
            })
                .then(function (result) {
                return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });
            })
                .then(function (result) {
                result.state.should.eql('approved');
                const approval2 = approvals[1];
                return bitgoSharedKeyUser.pendingApprovals().get({ id: approval2.id() });
            })
                .then(function (result) {
                return result.approve({ otp: bitgo.testUserOTP() });
            })
                .then(function () {
                throw new Error('approval success');
            })
                .catch(function (error) {
                error.message.should.equal('unspents expired, wallet passphrase or xprv required to recreate transaction');
            });
        });
        it('can approve when it does require tx signing (multiple recipients)', function () {
            return createTransactionPendingApprovalToMultipleRecipients()
                .then(function (pendingApproval) {
                return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });
            })
                .then(function (result) {
                return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });
            })
                .then(function (result) {
                result.state.should.eql('approved');
                // Parse the completed tx hex and make sure it was built with proper outputs
                const completedTxHex = result.info.transactionRequest.validTransaction;
                const transaction = utxolib.bitgo.createTransactionFromHex(completedTxHex, utxolib.networks.bitcoin);
                if (!transaction || !transaction.outs) {
                    throw new Error('transaction had no outputs or failed to parse successfully');
                }
                const outputAddresses = _.map(transaction.outs, function (out) {
                    return utxolib.address.fromOutputScript(out.script, BitGoJS.getNetworkObj());
                });
                // Output addresses should contain the 2 destinations, but not the change address
                outputAddresses.should.containEql(TestBitGo.TEST_WALLET3_ADDRESS);
                outputAddresses.should.containEql(TestBitGo.TEST_WALLET2_ADDRESS);
                outputAddresses.should.not.containEql(TestBitGo.TEST_SHARED_WALLET_CHANGE_ADDRESS);
            });
        });
        it('can manually pass in reconstructed tx', function () {
            let pendingApproval;
            return createTransactionPendingApprovalToMultipleRecipients()
                .then(function (pendingApproval) {
                return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });
            })
                .then(function (result) {
                pendingApproval = result;
                return pendingApproval.constructApprovalTx({
                    walletPassphrase: TestBitGo.TEST_PASSWORD,
                    otp: bitgo.testUserOTP(),
                });
            })
                .then(function (result) {
                return pendingApproval.approve({
                    walletPassphrase: TestBitGo.TEST_PASSWORD,
                    tx: result.tx,
                    otp: bitgo.testUserOTP(),
                });
            })
                .then(function (result) {
                result.state.should.eql('approved');
                // Parse the completed tx hex and make sure it was built with proper outputs
                const completedTxHex = result.info.transactionRequest.validTransaction;
                const transaction = utxolib.bitgo.createTransactionFromHex(completedTxHex, utxolib.networks.bitcoin);
                if (!transaction || !transaction.outs) {
                    throw new Error('transaction had no outputs or failed to parse successfully');
                }
                const outputAddresses = _.map(transaction.outs, function (out) {
                    return utxolib.address.fromOutputScript(out.script, BitGoJS.getNetworkObj());
                });
                // Output addresses should contain the 2 destinations, but not the change address
                outputAddresses.should.containEql(TestBitGo.TEST_WALLET3_ADDRESS);
                outputAddresses.should.containEql(TestBitGo.TEST_WALLET2_ADDRESS);
                outputAddresses.should.not.containEql(TestBitGo.TEST_SHARED_WALLET_CHANGE_ADDRESS);
            });
        });
    });
    describe('Reject', function () {
        let pendingApproval;
        before(function () {
            return Q.delay(500)
                .then(function () {
                return createTransactionPendingApproval();
            })
                .then(function (result) {
                pendingApproval = result;
            });
        });
        it('arguments', function () {
            assert_1.strict.throws(function () {
                pendingApproval.reject({}, 'invalid');
            });
            assert_1.strict.throws(function () {
                pendingApproval.reject('invalid');
            });
        });
        it('can cancel', function () {
            return pendingApproval.reject().then(function (result) {
                result.state.should.eql('rejected');
            });
        });
        it('can reject', function () {
            return bitgoSharedKeyUser
                .pendingApprovals()
                .get({ id: pendingApproval.id() })
                .then(function (result) {
                return result.reject();
            })
                .then(function (result) {
                result.state.should.eql('rejected');
            });
        });
    });
    describe('Create, Get, Approve and Reject with Multiple Approvers', function () {
        // setup third user
        let multipleApproversWallet;
        let pendingApproval;
        before(function () {
            return bitgoThirdUser
                .authenticate({
                username: TestBitGo.TEST_THIRD_USER,
                password: TestBitGo.TEST_THIRD_PASSWORD,
                otp: bitgo.testUserOTP(),
            })
                .then(function () {
                return bitgoThirdUser.unlock({ otp: bitgoThirdUser.testUserOTP() });
            })
                .then(function () {
                return bitgoThirdUser.wallets().get({ id: TestBitGo.TEST_WALLETMULTAPPROVERS_ADDRESS });
            })
                .then(function (result) {
                multipleApproversWallet = result;
                if (multipleApproversWallet.approvalsRequired() === 2) {
                    // we don't need to bother setting the number of approvals required
                    return;
                }
                return multipleApproversWallet
                    .updateApprovalsRequired({ approvalsRequired: 2 })
                    .then(function (result) {
                    return bitgoSharedKeyUser.pendingApprovals().get({ id: result.id });
                })
                    .then(function (result) {
                    pendingApproval = result;
                    pendingApproval.approvalsRequired().should.equal(1);
                    return result.approve({ otp: bitgo.testUserOTP() });
                })
                    .then(function (result) {
                    result.state.should.eql('approved');
                    // update wallet variable with new approvalsRequired
                    return bitgoThirdUser.wallets().get({ id: TestBitGo.TEST_WALLETMULTAPPROVERS_ADDRESS });
                })
                    .then(function (result) {
                    multipleApproversWallet = result;
                });
            });
        });
        after(function () {
            pendingApproval.reject();
        });
        it('should fail with too low approvalsRequired', function () {
            assert_1.strict.throws(function () {
                multipleApproversWallet.updateApprovalsRequired({ approvalsRequired: 0 });
            }, 'invalid approvalsRequired');
        });
        it('should fail with too high approvalsRequired', function () {
            const promise = multipleApproversWallet.updateApprovalsRequired({ approvalsRequired: 3 });
            return TestUtil.throws(promise, 'approvalsRequired must be less than the number of admins on the wallet');
        });
        it('should be a no-op with same approvalsRequired', function () {
            return multipleApproversWallet.updateApprovalsRequired({ approvalsRequired: 2 }).then(function (wallet) {
                wallet.should.equal(multipleApproversWallet.wallet);
            });
        });
        it('should set approvals required to 1 after 2 approvals', function () {
            return multipleApproversWallet
                .updateApprovalsRequired({ approvalsRequired: 1 })
                .then(function (result) {
                return bitgoSharedKeyUser.pendingApprovals().get({ id: result.id });
            })
                .then(function (result) {
                pendingApproval = result;
                pendingApproval.approvalsRequired().should.equal(2);
                return result.approve({ otp: bitgo.testUserOTP() });
            })
                .then(function (result) {
                result.state.should.eql('pending');
                return bitgo.pendingApprovals().get({ id: result.id });
            })
                .then(function (result) {
                return result.approve({ otp: bitgo.testUserOTP() });
            })
                .then(function (result) {
                result.state.should.eql('approved');
                return bitgoThirdUser.wallets().get({ id: TestBitGo.TEST_WALLETMULTAPPROVERS_ADDRESS });
            })
                .then(function (wallet) {
                multipleApproversWallet = wallet;
                multipleApproversWallet.approvalsRequired().should.equal(1);
            });
        });
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pendingapproval.js","sourceRoot":"","sources":["../../../test/integration/pendingapproval.ts"],"names":[],"mappings":"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","sourcesContent":["//\n// Tests for a Pending Approval\n//\n// Copyright 2015, BitGo, Inc.  All Rights Reserved.\n//\n\nimport * as _ from 'lodash';\nimport { strict as assert } from 'assert';\nimport 'should';\nimport * as utxolib from '@bitgo/utxo-lib';\n\nconst BitGoJS = require('../../src/index');\nconst TestBitGo = require('../lib/test_bitgo');\nconst TestUtil = require('./testutil');\n\nconst Q = require('q');\n\ndescribe('PendingApproval', function () {\n  let bitgo;\n  let bitgoSharedKeyUser;\n  let bitgoThirdUser;\n  let sharedWallet;\n\n  /**\n   * There is a 0-limit policy on the shared wallet\n   * Create a pending approval by attempting to send coins\n   */\n  const createTransactionPendingApproval = function () {\n    return sharedWallet\n      .sendCoins({\n        address: TestBitGo.TEST_WALLET2_ADDRESS,\n        amount: 0.0001 * 1e8,\n        walletPassphrase: TestBitGo.TEST_PASSWORD,\n        otp: bitgo.testUserOTP(),\n        message: 'never gonna',\n      })\n      .then(function (result) {\n        result.should.have.property('pendingApproval');\n        return bitgo.pendingApprovals().get({ id: result.pendingApproval });\n      });\n  };\n\n  /**\n   * There is a 0-limit policy on the shared wallet\n   * Create a pending approval by attempting to send to many\n   */\n  const createTransactionPendingApprovalToMultipleRecipients = function () {\n    return sharedWallet\n      .sendMany({\n        recipients: [\n          {\n            address: TestBitGo.TEST_WALLET3_ADDRESS,\n            amount: 0.0002 * 1e8,\n          },\n          {\n            address: TestBitGo.TEST_WALLET2_ADDRESS,\n            amount: 0.0001 * 1e8,\n          },\n          {\n            address: TestBitGo.TEST_SHARED_WALLET_CHANGE_ADDRESS,\n            amount: 0.0005 * 1e8,\n          },\n        ],\n        walletPassphrase: TestBitGo.TEST_PASSWORD,\n        otp: bitgo.testUserOTP(),\n        message: 'never gonna',\n      })\n      .then(function (result) {\n        result.should.have.property('pendingApproval');\n        return bitgo.pendingApprovals().get({ id: result.pendingApproval });\n      });\n  };\n\n  /**\n   * Create a pending approval by attempting to add a user to the wallet\n   */\n  const createPolicyPendingApproval = function () {\n    // it's ok to set up any tx limit since the daily limit is 0\n    return sharedWallet\n      .updatePolicyRule({\n        action: {\n          type: 'getApproval',\n        },\n        condition: {\n          amount: 1e8 + Math.round(Math.random() * 1e8),\n        },\n        id: 'com.bitgo.limit.tx',\n        type: 'transactionLimit',\n      })\n      .then(function (result) {\n        result.should.have.property('pendingApproval');\n        return bitgo.pendingApprovals().get({ id: result.pendingApproval.id });\n      });\n  };\n\n  before(function () {\n    bitgo = new TestBitGo();\n    bitgo.initializeTestVars();\n    bitgoSharedKeyUser = new TestBitGo();\n    bitgoSharedKeyUser.initializeTestVars();\n    bitgoThirdUser = new TestBitGo();\n    bitgoThirdUser.initializeTestVars();\n\n    return bitgo\n      .authenticateTestUser(bitgo.testUserOTP())\n      .then(function () {\n        return bitgoSharedKeyUser.authenticate({\n          username: TestBitGo.TEST_SHARED_KEY_USER,\n          password: TestBitGo.TEST_SHARED_KEY_PASSWORD,\n          otp: bitgo.testUserOTP(),\n        });\n      })\n      .then(function () {\n        return bitgo.unlock({ otp: bitgo.testUserOTP() });\n      })\n      .then(function () {\n        return bitgoSharedKeyUser.unlock({ otp: bitgo.testUserOTP() });\n      })\n      .then(function () {\n        return bitgo.wallets().get({ id: TestBitGo.TEST_SHARED_WALLET_ADDRESS });\n      })\n      .then(function (result) {\n        sharedWallet = result;\n      });\n  });\n\n  describe('Create and Get', function () {\n    let pendingApproval;\n\n    before(function () {\n      return createTransactionPendingApproval().then(function (result) {\n        pendingApproval = result;\n      });\n    });\n\n    after(function () {\n      return pendingApproval.reject();\n    });\n\n    it('arguments', function () {\n      assert.throws(function () {\n        bitgo.pendingApprovals().get({}, 'invalid');\n      });\n      assert.throws(function () {\n        bitgo.pendingApprovals().get('invalid');\n      });\n      assert.throws(function () {\n        bitgo.pendingApprovals().get({ id: 54321 }, 'invalid');\n      });\n    });\n\n    it('get property methods', function () {\n      pendingApproval.id().should.eql(pendingApproval.pendingApproval.id);\n      pendingApproval.ownerType().should.eql('wallet');\n      pendingApproval.walletId().should.eql(sharedWallet.id());\n      assert.equal(pendingApproval.enterpriseId(), undefined);\n      pendingApproval.state().should.eql('pending');\n      pendingApproval.creator().should.eql(TestBitGo.TEST_USERID);\n      pendingApproval.type().should.eql('transactionRequest');\n      pendingApproval.info().transactionRequest.message.should.eql('never gonna');\n      pendingApproval.info().transactionRequest.destinationAddress.should.eql(TestBitGo.TEST_WALLET2_ADDRESS);\n    });\n\n    it('get', function () {\n      pendingApproval.get({}).then(function (result) {\n        pendingApproval = result;\n        pendingApproval.id().should.eql(pendingApproval.pendingApproval.id);\n        pendingApproval.ownerType().should.eql('wallet');\n        pendingApproval.walletId().should.eql(sharedWallet.id());\n        assert.equal(pendingApproval.enterpriseId(), undefined);\n        pendingApproval.state().should.eql('pending');\n        pendingApproval.creator().should.eql(TestBitGo.TEST_USERID);\n        pendingApproval.type().should.eql('transactionRequest');\n        pendingApproval.info().transactionRequest.destinationAddress.should.eql(TestBitGo.TEST_WALLET2_ADDRESS);\n      });\n    });\n  });\n\n  describe('Approve', function () {\n    let pendingApproval;\n\n    before(function () {\n      return createTransactionPendingApproval().then(function (result) {\n        pendingApproval = result;\n      });\n    });\n\n    after(function () {\n      return pendingApproval.reject();\n    });\n\n    it('arguments', function () {\n      assert.throws(function () {\n        pendingApproval.approve({}, 'invalid');\n      });\n      assert.throws(function () {\n        pendingApproval.approve('invalid');\n      });\n    });\n\n    it('error when self approving', function () {\n      return createPolicyPendingApproval()\n        .then(function (pendingApproval) {\n          return pendingApproval.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });\n        })\n        .catch(function (err) {\n          err.message.should.containEql('cannot approve by self');\n          return pendingApproval.reject();\n        });\n    });\n\n    it('can approve when it does not require tx signing', function () {\n      return createPolicyPendingApproval().then(function (pendingApproval) {\n        return bitgoSharedKeyUser\n          .pendingApprovals()\n          .get({ id: pendingApproval.id() })\n          .then(function (result) {\n            return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });\n          })\n          .then(function (result) {\n            result.state.should.eql('approved');\n            result.info.policyRuleRequest.update.condition.amount.should.eql(\n              pendingApproval.info().policyRuleRequest.update.condition.amount\n            );\n          });\n      });\n    });\n\n    it('can approve when tx does not require reconstruction', function () {\n      return createTransactionPendingApproval().then(function (pendingApproval) {\n        return bitgoSharedKeyUser\n          .pendingApprovals()\n          .get({ id: pendingApproval.id() })\n          .then(function (result) {\n            return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });\n          })\n          .then(function (result) {\n            result.state.should.eql('approved');\n          });\n      });\n    });\n\n    it('error when it does require tx signing but wrong passphrase', function () {\n      return createTransactionPendingApproval()\n        .then(function (pendingApproval) {\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });\n        })\n        .then(function (result) {\n          return result.approve({ walletPassphrase: 'abcdef', otp: bitgo.testUserOTP() });\n        })\n        .catch(function (err) {\n          err.message.should.containEql('Unable to decrypt user keychain');\n          return pendingApproval.reject();\n        });\n    });\n\n    it('can approve when it does require tx signing', function () {\n      return createTransactionPendingApproval()\n        .then(function (pendingApproval) {\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });\n        })\n        .then(function (result) {\n          return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });\n        })\n        .then(function (result) {\n          result.state.should.eql('approved');\n        });\n    });\n\n    it('cannot approve when transaction needs reconstructing', function () {\n      let approvals: any[] = [];\n      return Q.all([createTransactionPendingApproval(), createTransactionPendingApproval()])\n        .spread(function (approval1, approval2) {\n          approvals = [approval1, approval2];\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: approval1.id() });\n        })\n        .then(function (result) {\n          return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });\n        })\n        .then(function (result) {\n          result.state.should.eql('approved');\n          const approval2 = approvals[1];\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: approval2.id() });\n        })\n        .then(function (result) {\n          return result.approve({ otp: bitgo.testUserOTP() });\n        })\n        .then(function () {\n          throw new Error('approval success');\n        })\n        .catch(function (error) {\n          error.message.should.equal('unspents expired, wallet passphrase or xprv required to recreate transaction');\n        });\n    });\n\n    it('can approve when it does require tx signing (multiple recipients)', function () {\n      return createTransactionPendingApprovalToMultipleRecipients()\n        .then(function (pendingApproval) {\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });\n        })\n        .then(function (result) {\n          return result.approve({ walletPassphrase: TestBitGo.TEST_PASSWORD, otp: bitgo.testUserOTP() });\n        })\n        .then(function (result) {\n          result.state.should.eql('approved');\n\n          // Parse the completed tx hex and make sure it was built with proper outputs\n          const completedTxHex = result.info.transactionRequest.validTransaction;\n          const transaction = utxolib.bitgo.createTransactionFromHex(completedTxHex, utxolib.networks.bitcoin);\n          if (!transaction || !transaction.outs) {\n            throw new Error('transaction had no outputs or failed to parse successfully');\n          }\n          const outputAddresses = _.map(transaction.outs, function (out) {\n            return utxolib.address.fromOutputScript(out.script, BitGoJS.getNetworkObj());\n          });\n\n          // Output addresses should contain the 2 destinations, but not the change address\n          outputAddresses.should.containEql(TestBitGo.TEST_WALLET3_ADDRESS);\n          outputAddresses.should.containEql(TestBitGo.TEST_WALLET2_ADDRESS);\n          outputAddresses.should.not.containEql(TestBitGo.TEST_SHARED_WALLET_CHANGE_ADDRESS);\n        });\n    });\n\n    it('can manually pass in reconstructed tx', function () {\n      let pendingApproval;\n      return createTransactionPendingApprovalToMultipleRecipients()\n        .then(function (pendingApproval) {\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: pendingApproval.id() });\n        })\n        .then(function (result) {\n          pendingApproval = result;\n          return pendingApproval.constructApprovalTx({\n            walletPassphrase: TestBitGo.TEST_PASSWORD,\n            otp: bitgo.testUserOTP(),\n          });\n        })\n        .then(function (result) {\n          return pendingApproval.approve({\n            walletPassphrase: TestBitGo.TEST_PASSWORD,\n            tx: result.tx,\n            otp: bitgo.testUserOTP(),\n          });\n        })\n        .then(function (result) {\n          result.state.should.eql('approved');\n\n          // Parse the completed tx hex and make sure it was built with proper outputs\n          const completedTxHex = result.info.transactionRequest.validTransaction;\n          const transaction = utxolib.bitgo.createTransactionFromHex(completedTxHex, utxolib.networks.bitcoin);\n          if (!transaction || !transaction.outs) {\n            throw new Error('transaction had no outputs or failed to parse successfully');\n          }\n          const outputAddresses = _.map(transaction.outs, function (out) {\n            return utxolib.address.fromOutputScript(out.script, BitGoJS.getNetworkObj());\n          });\n\n          // Output addresses should contain the 2 destinations, but not the change address\n          outputAddresses.should.containEql(TestBitGo.TEST_WALLET3_ADDRESS);\n          outputAddresses.should.containEql(TestBitGo.TEST_WALLET2_ADDRESS);\n          outputAddresses.should.not.containEql(TestBitGo.TEST_SHARED_WALLET_CHANGE_ADDRESS);\n        });\n    });\n  });\n\n  describe('Reject', function () {\n    let pendingApproval;\n\n    before(function () {\n      return Q.delay(500)\n        .then(function () {\n          return createTransactionPendingApproval();\n        })\n        .then(function (result) {\n          pendingApproval = result;\n        });\n    });\n\n    it('arguments', function () {\n      assert.throws(function () {\n        pendingApproval.reject({}, 'invalid');\n      });\n      assert.throws(function () {\n        pendingApproval.reject('invalid');\n      });\n    });\n\n    it('can cancel', function () {\n      return pendingApproval.reject().then(function (result) {\n        result.state.should.eql('rejected');\n      });\n    });\n\n    it('can reject', function () {\n      return bitgoSharedKeyUser\n        .pendingApprovals()\n        .get({ id: pendingApproval.id() })\n        .then(function (result) {\n          return result.reject();\n        })\n        .then(function (result) {\n          result.state.should.eql('rejected');\n        });\n    });\n  });\n\n  describe('Create, Get, Approve and Reject with Multiple Approvers', function () {\n    // setup third user\n    let multipleApproversWallet;\n    let pendingApproval;\n    before(function () {\n      return bitgoThirdUser\n        .authenticate({\n          username: TestBitGo.TEST_THIRD_USER,\n          password: TestBitGo.TEST_THIRD_PASSWORD,\n          otp: bitgo.testUserOTP(),\n        })\n        .then(function () {\n          return bitgoThirdUser.unlock({ otp: bitgoThirdUser.testUserOTP() });\n        })\n        .then(function () {\n          return bitgoThirdUser.wallets().get({ id: TestBitGo.TEST_WALLETMULTAPPROVERS_ADDRESS });\n        })\n        .then(function (result) {\n          multipleApproversWallet = result;\n\n          if (multipleApproversWallet.approvalsRequired() === 2) {\n            // we don't need to bother setting the number of approvals required\n            return;\n          }\n\n          return multipleApproversWallet\n            .updateApprovalsRequired({ approvalsRequired: 2 })\n            .then(function (result) {\n              return bitgoSharedKeyUser.pendingApprovals().get({ id: result.id });\n            })\n            .then(function (result) {\n              pendingApproval = result;\n              pendingApproval.approvalsRequired().should.equal(1);\n              return result.approve({ otp: bitgo.testUserOTP() });\n            })\n            .then(function (result) {\n              result.state.should.eql('approved');\n\n              // update wallet variable with new approvalsRequired\n              return bitgoThirdUser.wallets().get({ id: TestBitGo.TEST_WALLETMULTAPPROVERS_ADDRESS });\n            })\n            .then(function (result) {\n              multipleApproversWallet = result;\n            });\n        });\n    });\n\n    after(function () {\n      pendingApproval.reject();\n    });\n\n    it('should fail with too low approvalsRequired', function () {\n      assert.throws(function () {\n        multipleApproversWallet.updateApprovalsRequired({ approvalsRequired: 0 });\n      }, 'invalid approvalsRequired');\n    });\n\n    it('should fail with too high approvalsRequired', function () {\n      const promise = multipleApproversWallet.updateApprovalsRequired({ approvalsRequired: 3 });\n      return TestUtil.throws(promise, 'approvalsRequired must be less than the number of admins on the wallet');\n    });\n\n    it('should be a no-op with same approvalsRequired', function () {\n      return multipleApproversWallet.updateApprovalsRequired({ approvalsRequired: 2 }).then(function (wallet) {\n        wallet.should.equal(multipleApproversWallet.wallet);\n      });\n    });\n\n    it('should set approvals required to 1 after 2 approvals', function () {\n      return multipleApproversWallet\n        .updateApprovalsRequired({ approvalsRequired: 1 })\n        .then(function (result) {\n          return bitgoSharedKeyUser.pendingApprovals().get({ id: result.id });\n        })\n        .then(function (result) {\n          pendingApproval = result;\n          pendingApproval.approvalsRequired().should.equal(2);\n          return result.approve({ otp: bitgo.testUserOTP() });\n        })\n        .then(function (result) {\n          result.state.should.eql('pending');\n\n          return bitgo.pendingApprovals().get({ id: result.id });\n        })\n        .then(function (result) {\n          return result.approve({ otp: bitgo.testUserOTP() });\n        })\n        .then(function (result) {\n          result.state.should.eql('approved');\n\n          return bitgoThirdUser.wallets().get({ id: TestBitGo.TEST_WALLETMULTAPPROVERS_ADDRESS });\n        })\n        .then(function (wallet) {\n          multipleApproversWallet = wallet;\n          multipleApproversWallet.approvalsRequired().should.equal(1);\n        });\n    });\n  });\n});\n"]}

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


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