PHP WebShell

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

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

"use strict";
//
// Tests for BitGo Object
//
Object.defineProperty(exports, "__esModule", { value: true });
const crypto = require("crypto");
const nock = require("nock");
const should = require("should");
const assert = require("assert");
const sdk_core_1 = require("@bitgo/sdk-core");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const _ = require("lodash");
const BitGoJS = require("../../src/index");
const rp = require('request-promise');
const sdk_test_1 = require("@bitgo/sdk-test");
const bitgo_1 = require("../../src/bitgo");
nock.disableNetConnect();
describe('BitGo Prototype Methods', function () {
    describe('Version', () => {
        it('version', function () {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.initializeTestVars();
            const version = bitgo.version();
            version.should.be.a.String();
        });
    });
    describe('validate', () => {
        it('should get', () => {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.getValidate().should.equal(true);
        });
        it('should set', () => {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.setValidate(false);
            bitgo.getValidate().should.equal(false);
            bitgo['_validate'].should.equal(false);
        });
    });
    describe('Environments', () => {
        it('production', () => {
            BitGoJS.setNetwork('testnet');
            sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'prod' });
            BitGoJS.getNetwork().should.equal('bitcoin');
        });
        it('staging', () => {
            BitGoJS.setNetwork('testnet');
            sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'staging' });
            BitGoJS.getNetwork().should.equal('testnet');
        });
        it('test', () => {
            BitGoJS.setNetwork('bitcoin');
            sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'test' });
            BitGoJS.getNetwork().should.equal('testnet');
        });
        it('dev', () => {
            sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'dev' });
            BitGoJS.getNetwork().should.equal('testnet');
        });
        it('custom network (prod)', () => {
            sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { customBitcoinNetwork: 'bitcoin', customRootURI: 'http://rooturi.example' });
            BitGoJS.getNetwork().should.equal('bitcoin');
        });
        it('custom network (testnet)', () => {
            sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { customBitcoinNetwork: 'testnet', customRootURI: 'http://rooturi.example' });
            BitGoJS.getNetwork().should.equal('testnet');
        });
    });
    describe('HMAC request verification', () => {
        it('throws if HMAC request verification is disabled for non-prod environments', function () {
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'prod', hmacVerification: false })).should.throw(/Cannot disable request HMAC verification in environment/);
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'test', hmacVerification: false })).should.not.throw(/Cannot disable request HMAC verification in environment/);
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'adminProd', hmacVerification: false })).should.throw(/Cannot disable request HMAC verification in environment/);
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'adminTest', hmacVerification: false })).should.not.throw(/Cannot disable request HMAC verification in environment/);
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, {
                env: 'dev',
                customRootURI: 'http://rooturi.example',
                hmacVerification: false,
            })).should.not.throw(/Cannot disable request HMAC verification in environment/);
        });
        it('allows disabling of HMAC request verification only for dev environments', function () {
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'dev', hmacVerification: false })).should.not.throw();
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'latest', hmacVerification: false })).should.not.throw();
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'adminDev', hmacVerification: false })).should.not.throw();
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'adminLatest', hmacVerification: false })).should.not.throw();
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'local', hmacVerification: false })).should.not.throw();
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'localNonSecure', hmacVerification: false })).should.not.throw();
            (() => sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, {
                env: 'branch',
                customRootURI: 'http://rooturi.example',
                hmacVerification: false,
            })).should.not.throw();
        });
    });
    describe('Authenticate in Microservices', () => {
        let bitgo;
        const authenticateRequest = {
            username: 'test@bitgo.com',
            password: 'password',
            otp: '000000',
            extensible: false,
            extensionAddress: 'address',
            forceSMS: false,
        };
        it('goes to microservices', async function () {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock', microservicesUri: 'https://microservices.uri' });
            const scope = nock(BitGoJS.Environments[bitgo.getEnv()].uri)
                .post('/api/auth/v1/session')
                .reply(200, {
                user: {
                    username: 'test@bitgo.com',
                },
                access_token: 'token12356',
            });
            await bitgo.authenticate(authenticateRequest);
            scope.isDone().should.be.true();
        });
        it('goes to microservices even when microservicesUri is not specified', async function () {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            const scope = nock(BitGoJS.Environments[bitgo.getEnv()].uri)
                .post('/api/auth/v1/session')
                .reply(200, {
                user: {
                    username: 'test@bitgo.com',
                },
                access_token: 'token12356',
            });
            await bitgo.authenticate(authenticateRequest);
            scope.isDone().should.be.true();
        });
    });
    describe('Verify Address', () => {
        let bitgo;
        before(() => {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
        });
        it('errors', () => {
            (() => bitgo.verifyAddress()).should.throw();
            (() => bitgo.verifyAddress({})).should.throw();
            bitgo.verifyAddress({ address: 'xyzzy' }).should.be.false();
        });
        it('standard', () => {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'prod' });
            bitgo.verifyAddress({ address: '1Bu3bhwRmevHLAy1JrRB6AfcxfgDG2vXRd' }).should.be.true();
            // wrong version byte:
            bitgo.verifyAddress({ address: '9Ef7HsuByGBogqkjoF5Yng7MYkq5UCdmZz' }).should.be.false();
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.verifyAddress({ address: 'n4DNhSiEaodqaiF9tLYXTCh4kFbdUzxBHs' }).should.be.true();
        });
        it('p2sh', () => {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'prod' });
            bitgo.verifyAddress({ address: '3QJmV3qfvL9SuYo34YihAf3sRCW3qSinyC' }).should.be.true();
            // wrong version byte:
            bitgo.verifyAddress({ address: 'HV8swrGkmeN7Xig4vENr93aQSrX4iHjg7D' }).should.be.false();
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.verifyAddress({ address: '2NEeFWbfu4EA1rcKx48e82Mj8d6FKcWawZw' }).should.be.true();
        });
    });
    describe('Encrypt/Decrypt', () => {
        const password = 'mickey mouse';
        const secret = 'this is a secret';
        it('invalid password', () => {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.initializeTestVars();
            const opaque = bitgo.encrypt({ password: password, input: secret });
            (() => bitgo.decrypt({ password: 'hack hack', input: opaque })).should.throw();
        });
        it('valid password', () => {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.initializeTestVars();
            const opaque = bitgo.encrypt({ password: password, input: secret });
            bitgo.decrypt({ password: password, input: opaque }).should.equal(secret);
        });
    });
    describe('Password Generation', () => {
        it('generates a random password', () => {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.initializeTestVars();
            const password = bitgo.generateRandomPassword();
            should.exist(password);
        });
        it('generates a random password with a numWords argument', () => {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
            bitgo.initializeTestVars();
            for (let i = 0; i < 1000; i++) {
                const password = bitgo.generateRandomPassword(10);
                should.exist(password);
                // randomly generated password should be 55 characters roughly 92.5% of the time,
                // 54 characters roughly 7.5% of the time, 53 characters 0.001% of the time,
                // and fewer than 53 characters very, very rarely
                password.length.should.be.within(53, 55);
            }
        });
    });
    describe('Shamir Secret Sharing', () => {
        const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
        const seed = '8cc57dac9cdae42bf7848a2d12f2874d31eca1f9de8fe3f8fa13e7857b545d59';
        const xpub = 'xpub661MyMwAqRbcEusRjkJ64BXgR8ddYsXbuDJfbRc3eZcZVEa2ygswDiFZQpHFsA5N211YDvi2N898h4KrcXcfsR8PLhjJaPUwCUqg1ptBBHN';
        const passwords = ['mickey', 'mouse', 'donald', 'duck'];
        it('should fail to split secret with wrong m', () => {
            (() => bitgo.splitSecret({
                seed,
                passwords: ['abc'],
                m: 0,
            })).should.throw('m must be a positive integer greater than or equal to 2');
        });
        it('should fail to split secret with bad password count', () => {
            (() => bitgo.splitSecret({
                seed,
                passwords: ['abc'],
                m: 2,
            })).should.throw('passwords array length cannot be less than m');
        });
        it('should split and fail to reconstitute secret with bad passwords', () => {
            const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 3 });
            const shards = _.at(splitSecret.seedShares, [0, 2]);
            const subsetPasswords = _.at(passwords, [0, 3]);
            (() => bitgo.reconstituteSecret({
                shards,
                passwords: subsetPasswords,
                xpub,
            })).should.throw(/ccm: tag doesn't match/);
        });
        it('should split and reconstitute secret', () => {
            const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 2 });
            const shards = _.at(splitSecret.seedShares, [0, 2]);
            const subsetPasswords = _.at(passwords, [0, 2]);
            const reconstitutedSeed = bitgo.reconstituteSecret({ shards, passwords: subsetPasswords });
            reconstitutedSeed.seed.should.equal(seed);
            reconstitutedSeed.xpub.should.equal('xpub661MyMwAqRbcEusRjkJ64BXgR8ddYsXbuDJfbRc3eZcZVEa2ygswDiFZQpHFsA5N211YDvi2N898h4KrcXcfsR8PLhjJaPUwCUqg1ptBBHN');
            reconstitutedSeed.xprv.should.equal('xprv9s21ZrQH143K2Rnxdim5h3aws6o99QokXzP4o3CS6E5acSEtS9Zgfuw5ZWujhTHTWEAZDfmP3yxA1Ccn6myVkGEpRrT4xWgaEpoW7YiBAtC');
        });
        it('should split and incorrectly verify secret', () => {
            const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 3 });
            const isValid = bitgo.verifyShards({ shards: splitSecret.seedShares, passwords, m: 2 });
            isValid.should.equal(false);
        });
        it('should split and verify secret', () => {
            const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 2 });
            const isValid = bitgo.verifyShards({ shards: splitSecret.seedShares, passwords, m: 2, xpub });
            isValid.should.equal(true);
        });
        it('should split and verify secret with many parts', () => {
            const allPws = ['0', '1', '2', '3', '4', '5', '6', '7'];
            const splitSecret = bitgo.splitSecret({ seed, passwords: allPws, m: 3 });
            const isValid = bitgo.verifyShards({ shards: splitSecret.seedShares, passwords: allPws, m: 3, xpub });
            isValid.should.equal(true);
        });
    });
    describe('ECDH sharing secret', () => {
        function getKey(seed) {
            return utxo_lib_1.ECPair.fromPrivateKey(utxo_lib_1.bip32.fromSeed(crypto.createHash('sha256').update(seed).digest()).privateKey);
        }
        it('should calculate a new ECDH sharing secret correctly', () => {
            for (let i = 0; i < 256; i++) {
                const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo);
                const eckey1 = getKey(`${i}.a`);
                const eckey2 = getKey(`${i}.b`);
                const sharingKey1 = bitgo.getECDHSecret({ eckey: eckey1, otherPubKeyHex: eckey2.publicKey.toString('hex') });
                const sharingKey2 = bitgo.getECDHSecret({ eckey: eckey2, otherPubKeyHex: eckey1.publicKey.toString('hex') });
                sharingKey1.should.equal(sharingKey2);
                switch (i) {
                    case 0:
                        sharingKey1.should.eql('465ffe5745325998b83fb39631347148e24d4f21b3f3b54739c264d5c42db4b8');
                        break;
                    case 1:
                        sharingKey1.should.eql('61ff44fc1af8061a433a314b7b8be8ae352c10f62aac5887047dbaa5643b818d');
                        break;
                }
            }
        });
    });
    describe('change password', function () {
        let bitgo;
        let bgUrl;
        before(async function () {
            nock('https://bitgo.fakeurl')
                .post('/api/auth/v1/session')
                .reply(200, {
                access_token: 'access_token',
                user: { username: 'update_pw_tester@bitgo.com' },
            });
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            bitgo.initializeTestVars();
            bitgo.setValidate(false);
            await bitgo.authenticateChangePWTestUser(bitgo.testUserOTP());
            bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
        });
        const oldPassword = 'oldPassword';
        const newPassword = 'newPassword';
        const otherPassword = 'otherPassword';
        describe('should fail to change the password', function changePWFail() {
            it('wrong arguments', async function () {
                await bitgo.changePassword({ newPassword: '5678' }).should.be.rejectedWith('expected string oldPassword');
                await bitgo
                    .changePassword({ oldPassword: 1234, newPassword: '5678' })
                    .should.be.rejectedWith('expected string oldPassword');
                await bitgo.changePassword({ oldPassword: '1234' }).should.be.rejectedWith('expected string newPassword');
                await bitgo
                    .changePassword({ oldPassword: '1234', newPassword: 5678 })
                    .should.be.rejectedWith('expected string newPassword');
            });
            it('incorrect old password', async function () {
                nock(bgUrl).post('/api/v1/user/verifypassword').reply(200, { valid: false });
                await bitgo
                    .changePassword({ oldPassword, newPassword })
                    .should.be.rejectedWith('the provided oldPassword is incorrect');
            });
        });
        it('successful password change', async function () {
            nock(bgUrl).post('/api/v1/user/verifypassword').reply(200, { valid: true });
            nock(bgUrl)
                .post('/api/v1/user/encrypted')
                .reply(200, {
                version: 1,
                keychains: {
                    xpub11: bitgo.encrypt({ input: 'xprv11', password: oldPassword }),
                    xpub12: bitgo.encrypt({ input: 'xprv12', password: oldPassword }),
                    xpub13: bitgo.encrypt({ input: 'xprv13', password: otherPassword }),
                    xpub14: bitgo.encrypt({ input: 'xprv14', password: oldPassword }),
                },
            });
            nock(bgUrl)
                .get('/api/v2/tbtc/key')
                .query(true)
                .reply(200, {
                keys: [
                    {
                        pub: 'xpub21',
                        encryptedPrv: bitgo.encrypt({ input: 'xprv21', password: oldPassword }),
                    },
                    {
                        pub: 'xpub22',
                        encryptedPrv: bitgo.encrypt({ input: 'xprv22', password: otherPassword }),
                    },
                ],
            });
            nock(bgUrl).post('/api/v1/user/changepassword').reply(200, {});
            await bitgo.changePassword({ oldPassword, newPassword });
        });
        afterEach(function afterChangePassword() {
            nock.pendingMocks().should.be.empty();
        });
    });
    describe('HMAC Handling', () => {
        let bitgo;
        const token = 'v2x5b735fed2486593f8fea19113e5c717308f90a5fb00e740e46c7bfdcc078cfd0';
        before(() => {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock', accessToken: token });
        });
        it('should correctly calculate request headers', () => {
            const originalDateNow = Date.now;
            Date.now = () => 1521589882510;
            const fetchMeUrl = bitgo.url('/user/me');
            const requestHeaders = bitgo.calculateRequestHeaders({ url: fetchMeUrl, token });
            Date.now = originalDateNow;
            requestHeaders.timestamp.should.equal(1521589882510);
            requestHeaders.tokenHash.should.equal('a85af08e6723e41acd6a3fb9ef58422082e673df33c58e1db175bb740a2c934d');
            requestHeaders.hmac.should.equal('6de77d5a5446a3e5649456c11480706a71071b15639c3c787af65bdb02ecf1ec');
        });
        it('should correctly handle authentication response', () => {
            const responseJson = {
                encryptedToken: '{"iv":"EqxVaGTLY4naAYkuBaTz0w==","v":1,"iter":1000,"ks":128,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"4S4dBYcgL4s=","ct":"FgBRJljb8iSYxnAjMi4Qotr7sTKbSmWnlfHZShMSi8YeeE3kiS8bpHNUwAPhY8tgouh3UsEwrJnY+54MvqFD7yd19pG1V4CVssr8"}',
                derivationPath: 'm/999999/104490948/173846667',
                encryptedECDHXprv: '{"iv":"QKHEF2GNcwOJwy6+pwANRA==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"W2sVFvXDlOw=","ct":"8BTCqS25X37kLzmzQdGenhXH6znn9qEmkszAeS8kLnRdqKSiUiC7bTAVgg/Np5yrV7F7Jyiq+MTpVT76EoUT+PMJzArv0gUQKC2JPB3JuVKeAAVWBQmhWfkEwRfyv4hq4WMxwZtocwBqThvd2pJm9HE51GX4/Wo="}',
            };
            const parsedAuthenticationData = bitgo.handleTokenIssuance(responseJson, 'test@bitgo.com');
            parsedAuthenticationData.token.should.equal(token);
            parsedAuthenticationData.ecdhXprv.should.equal('xprv9s21ZrQH143K3si1bKGp7KqgCQv39ttQ7aUwWzVdytgHd8HtDCHyEp14mxfhiT3qHTq4BaSrA7uUkG6AJTfPJBsRu63drvBqYuMZyTxepH7');
        });
        it('should correctly verify a response hmac', async function () {
            const url = bitgo
                .coin('tltc')
                .url('/wallet/5941b202b42fcbc707170d5b597491d9/address/QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE?segwit=1');
            const requestHeaderData = bitgo.calculateRequestHeaders({ url, token });
            const requestHeaders = {
                'BitGo-Auth-Version': '2.0',
                'Content-Type': 'application/json',
                'Auth-Timestamp': requestHeaderData.timestamp,
                Authorization: 'Bearer ' + requestHeaderData.tokenHash,
                HMAC: requestHeaderData.hmac,
            };
            const responseBody = '{"id":"5a7ca8bcaf52c8e807c575fb692609ec","address":"QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE","chain":0,"index":2,"coin":"tltc","wallet":"5941b202b42fcbc707170d5b597491d9","coinSpecific":{"redeemScript":"522102835bcfd130f7a56f72c905b782d90b66e22f88ad3309cf72af5138a7d44be8b3210322c7f42a1eb212868eab78db7ba64846075d98c7f4c7aa25a02e57871039e0cd210265825be0d5bf957fb72abd7c23bf0836a78a15f951a073467cd5c99e03ce7ab753ae"},"balance":{"updated":"2018-02-28T23:48:07.341Z","numTx":1,"numUnspents":1,"totalReceived":20000000}}';
            nock('https://bitgo.fakeurl', { reqheaders: requestHeaders })
                .get('/api/v2/tltc/wallet/5941b202b42fcbc707170d5b597491d9/address/QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE?segwit=1')
                .reply(200, responseBody, {
                hmac: '30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec',
                timestamp: '1521590532925',
            });
            const responseData = (await rp({
                uri: url,
                method: 'GET',
                headers: requestHeaders,
                transform: (body, response) => {
                    // verify the response headers
                    const url = response.request.href;
                    const hmac = response.headers.hmac;
                    const timestamp = response.headers.timestamp;
                    const statusCode = response.statusCode;
                    const verificationParams = {
                        url,
                        hmac,
                        timestamp,
                        token,
                        statusCode,
                        text: body,
                    };
                    return bitgo.verifyResponse(verificationParams);
                },
            }));
            responseData.signatureSubject.should.equal('1521590532925|/api/v2/tltc/wallet/5941b202b42fcbc707170d5b597491d9/address/QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE?segwit=1|200|{"id":"5a7ca8bcaf52c8e807c575fb692609ec","address":"QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE","chain":0,"index":2,"coin":"tltc","wallet":"5941b202b42fcbc707170d5b597491d9","coinSpecific":{"redeemScript":"522102835bcfd130f7a56f72c905b782d90b66e22f88ad3309cf72af5138a7d44be8b3210322c7f42a1eb212868eab78db7ba64846075d98c7f4c7aa25a02e57871039e0cd210265825be0d5bf957fb72abd7c23bf0836a78a15f951a073467cd5c99e03ce7ab753ae"},"balance":{"updated":"2018-02-28T23:48:07.341Z","numTx":1,"numUnspents":1,"totalReceived":20000000}}');
            responseData.expectedHmac.should.equal('30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec');
            responseData.isValid.should.equal(true);
        });
        it('should include request body as part of the hmac', async function () {
            const url = 'https://bitgo.fakeurl';
            const body = { test: 'test' };
            const fixedUnixTime = 1627374646;
            const originalDateNow = Date.now;
            Date.now = () => fixedUnixTime;
            try {
                nock(url)
                    .post('/', body)
                    .reply(201, undefined, {
                    hmac: '677e0c9a65ca384415945cb19b40ae38eaadfbce3ccce8c5d7bf37c1973b2553',
                    timestamp: String(fixedUnixTime),
                });
                const resp = (await bitgo.post(url).send(body));
                resp.req.headers['hmac'].should.equal('4425a4004ef2724add25b4dd019d21c66394653a049d82e37df3a2c356b5706d');
            }
            finally {
                Date.now = originalDateNow;
            }
        });
        it('should recognize trailing slash inconsistency', () => {
            const verificationParams = {
                url: 'https://google.com/api',
                hmac: '30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec',
                timestamp: '1521590532925',
                token: token,
                statusCode: 200,
                text: 'fakedata',
            };
            const verificationDetails = bitgo.verifyResponse(verificationParams);
            verificationDetails.signatureSubject.should.equal('1521590532925|/api|200|fakedata');
            verificationDetails.signatureSubject.should.not.equal('1521590532925|/api/|200|fakedata');
            verificationDetails.expectedHmac.should.equal('2064f2adb168ef8808f6a42f588d7d6bc14e98e8b41239c6bbb7349e52f2249a');
            verificationDetails.isValid.should.equal(false);
        });
        it('should auto-amend trailing slash', () => {
            const verificationParams = {
                url: 'https://google.com',
                hmac: '30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec',
                timestamp: '1521590532925',
                token: token,
                statusCode: 200,
                text: 'fakedata',
            };
            const verificationDetails = bitgo.verifyResponse(verificationParams);
            verificationDetails.signatureSubject.should.equal('1521590532925|/|200|fakedata');
            verificationDetails.expectedHmac.should.equal('51c6d024f261e166e8a323f8fa36a9bb8d4d02b076334c2a9ae0a49efc5724d4');
            verificationDetails.isValid.should.equal(false);
        });
        it('should throw if hmac validation is enabled, and no valid hmac headers are returned', async function () {
            const url = 'https://fakeurl.invalid';
            const scope = nock(url).get('/').reply(200);
            // test suite bitgo object has hmac verification enabled, so it should throw when the nock responds
            await bitgo.get(url).should.be.rejectedWith(/invalid response HMAC, possible man-in-the-middle-attack/);
            scope.done();
        });
        it('should not enforce hmac verification if hmac verification is disabled', async function () {
            const bg = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock', hmacVerification: false, accessToken: token });
            const url = 'https://fakeurl.invalid';
            const scope = nock(url).get('/').reply(200, { ok: 1 });
            const res = (await bg.get(url));
            res.body.should.have.property('ok', 1);
            scope.done();
        });
    });
    describe('Token Definitions at Startup', function () {
        it('Should return a non-empty list of tokens before the server responds', async function () {
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            bitgo.initializeTestVars();
            const constants = bitgo.getConstants();
            constants.should.have.propertyByPath('eth', 'tokens', 'length').greaterThan(0);
        });
        after(function tokenDefinitionsAfter() {
            nock.pendingMocks().should.be.empty();
        });
    });
    describe('superagent wrappers', function () {
        let bitgo;
        let bgUrl;
        before(function () {
            bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            bitgo.initializeTestVars();
            bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
            nock(bgUrl).patch('/').reply(200);
        });
        it('PATCH requests', async function () {
            const res = await bitgo.patch(bgUrl);
            res.status.should.equal(200);
        });
        after(function () {
            nock.pendingMocks().should.be.empty();
        });
    });
    describe('preprocessAuthenticationParams', () => {
        const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
        it('should fail if passed non-string username or password', function () {
            (() => bitgo.preprocessAuthenticationParams({ username: 123 })).should.throw(/expected string username/);
            (() => bitgo.preprocessAuthenticationParams({ username: 'abc', password: {} })).should.throw(/expected string password/);
        });
    });
    describe('authenticate', function () {
        afterEach(function ensureNoPendingMocks() {
            nock.pendingMocks().should.be.empty();
        });
        it('should get the ecdhKeychain if ensureEcdhKeychain is set and user already has ecdhKeychain', async function () {
            nock('https://bitgo.fakeurl')
                .post('/api/auth/v1/session')
                .reply(200, {
                access_token: 'access_token',
                user: { username: 'auth-test@bitgo.com' },
            });
            nock('https://bitgo.fakeurl')
                .get('/api/v1/user/settings')
                .reply(200, {
                settings: {
                    ecdhKeychain: 'some-existing-xpub',
                },
            });
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            const response = await bitgo.authenticate({
                username: 'auth-test@bitgo.com',
                password: 'password123',
                otp: '000000',
                ensureEcdhKeychain: true,
            });
            should.exist(response.user.ecdhKeychain);
            response.user.ecdhKeychain.should.equal('some-existing-xpub');
        });
        it('should create the ecdhKeychain if ensureEcdhKeychain is set and the user does not already have ecdhKeychain', async function () {
            nock('https://bitgo.fakeurl')
                .post('/api/auth/v1/session')
                .reply(200, {
                access_token: 'access_token',
                user: { username: 'auth-test@bitgo.com' },
            });
            /**
             * This is {} because want to make sure the user has no ecdhXpub set before we set it
             */
            nock('https://bitgo.fakeurl').get('/api/v1/user/settings').reply(200, {
                settings: {},
            });
            nock('https://bitgo.fakeurl').post('/api/v1/keychain').reply(200, {
                xpub: 'some-xpub',
            });
            nock('https://bitgo.fakeurl')
                .put('/api/v2/user/settings')
                .reply(200, {
                settings: {
                    ecdhKeychain: 'some-xpub',
                },
            });
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            const response = await bitgo.authenticate({
                username: 'auth-test@bitgo.com',
                password: 'password123',
                otp: '000000',
                ensureEcdhKeychain: true,
            });
            should.exist(response.user.ecdhKeychain);
            response.user.ecdhKeychain.should.equal('some-xpub');
        });
    });
    describe('passkey authentication', () => {
        afterEach(function ensureNoPendingMocks() {
            nock.cleanAll();
            nock.pendingMocks().should.be.empty();
        });
        it('should authenticate with a passkey', async () => {
            const userId = '123';
            const passkey = `{"id": "id", "response": {"authenticatorData": "123", "clientDataJSON": "123", "signature": "123", "userHandle": "${userId}"}}`;
            const keyPair = await (0, sdk_core_1.generateGPGKeyPair)('secp256k1');
            nock('https://bitgo.fakeurl')
                .persist()
                .get('/api/v1/client/constants')
                .reply(200, { ttl: 3600, constants: { passkeyBitGoGpgKey: keyPair.publicKey } });
            nock('https://bitgo.fakeurl')
                .post('/api/auth/v1/session')
                .reply(200, async (uri, requestBody) => {
                assert(typeof requestBody === 'object');
                should.exist(requestBody.userId);
                should.exist(requestBody.passkey);
                requestBody.userId.should.equal(userId);
                requestBody.passkey.should.equal(passkey);
                return {
                    access_token: 'access_token',
                    user: { username: 'auth-test@bitgo.com' },
                };
            });
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            const response = await bitgo.authenticateWithPasskey(passkey);
            should.exist(response.access_token);
            response.access_token.should.equal('access_token');
        });
        it('should throw - invalid userHandle', async () => {
            const passkey = `{"id": "id", "response": {"authenticatorData": "123", "clientDataJSON": "123", "signature": "123", "userHandle": 123}}`;
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            try {
                await bitgo.validatePasskeyResponse(passkey);
                assert.fail('Expected error not thrown');
            }
            catch (e) {
                assert.equal(e.message, 'userHandle is missing');
            }
        });
        it('should throw - invalid authenticatorData', async () => {
            const passkey = `{"id": "id", "response": { "clientDataJSON": "123", "signature": "123", "userHandle": "123"}}`;
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            try {
                await bitgo.validatePasskeyResponse(passkey);
                assert.fail('Expected error not thrown');
            }
            catch (e) {
                assert.equal(e.message, 'authenticatorData is missing');
            }
        });
        it('should throw - invalid passkey json', async () => {
            const passkey = `{{"id": "id", "response": { "clientDataJSON": "123", "signature": "123", "userHandle": "123"}}`;
            const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
            try {
                await bitgo.validatePasskeyResponse(passkey);
                assert.fail('Expected error not thrown');
            }
            catch (e) {
                console.log(e);
                assert(e.message.includes('JSON'));
            }
        });
        it('should throw - missing access token', async () => {
            const passkey = `{"id": "id", "response": { "authenticatorData": "123", "clientDataJSON": "123", "signature": "123", "userHandle": "123"}}`;
            nock('https://bitgo.fakeurl')
                .post('/api/auth/v1/session')
                .reply(200, async () => {
                return {
                    user: { username: 'auth-test@bitgo.com' },
                };
            });
            try {
                const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
                await bitgo.authenticateWithPasskey(passkey);
                assert.fail('Expected error not thrown');
            }
            catch (e) {
                assert.equal(e.message, 'Failed to login. Please contact support@bitgo.com');
            }
        });
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bitgo.js","sourceRoot":"","sources":["../../../test/unit/bitgo.ts"],"names":[],"mappings":";AAAA,EAAE;AACF,yBAAyB;AACzB,EAAE;;AAEF,iCAAiC;AACjC,6BAA6B;AAC7B,iCAAiC;AACjC,iCAAkC;AAElC,8CAA6D;AAC7D,8CAAgD;AAChD,4BAA4B;AAC5B,2CAA2C;AAC3C,MAAM,EAAE,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAAC;AAEtC,8CAA4C;AAC5C,2CAAwC;AAExC,IAAI,CAAC,iBAAiB,EAAE,CAAC;AAEzB,QAAQ,CAAC,yBAAyB,EAAE;IAClC,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;QACvB,EAAE,CAAC,SAAS,EAAE;YACZ,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC;YAChC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;QAC/B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,UAAU,EAAE,GAAG,EAAE;QACxB,EAAE,CAAC,YAAY,EAAE,GAAG,EAAE;YACpB,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACzC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,YAAY,EAAE,GAAG,EAAE;YACpB,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YACzB,KAAK,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACxC,KAAK,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACzC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,cAAc,EAAE,GAAG,EAAE;QAC5B,EAAE,CAAC,YAAY,EAAE,GAAG,EAAE;YACpB,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YAC9B,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YAC3C,OAAO,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;YACjB,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YAC9B,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC;YAC9C,OAAO,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE;YACd,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YAC9B,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YAC3C,OAAO,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;YACb,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;YAC1C,OAAO,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uBAAuB,EAAE,GAAG,EAAE;YAC/B,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,oBAAoB,EAAE,SAAS,EAAE,aAAa,EAAE,wBAAwB,EAAE,CAAC,CAAC;YACxG,OAAO,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;YAClC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,oBAAoB,EAAE,SAAS,EAAE,aAAa,EAAE,wBAAwB,EAAE,CAAC,CAAC;YACxG,OAAO,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,2BAA2B,EAAE,GAAG,EAAE;QACzC,EAAE,CAAC,2EAA2E,EAAE;YAC9E,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CACtF,yDAAyD,CAC1D,CAAC;YACF,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAC1F,yDAAyD,CAC1D,CAAC;YACF,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,WAAW,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAC3F,yDAAyD,CAC1D,CAAC;YACF,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,WAAW,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAC/F,yDAAyD,CAC1D,CAAC;YACF,CAAC,GAAG,EAAE,CACJ,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE;gBACxB,GAAG,EAAE,KAAK;gBACV,aAAa,EAAE,wBAAwB;gBACvC,gBAAgB,EAAE,KAAK;aACxB,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAC;QACpF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yEAAyE,EAAE;YAC5E,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;YAC9F,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,QAAQ,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;YACjG,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,UAAU,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;YACnG,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,aAAa,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;YACtG,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,OAAO,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;YAChG,CAAC,GAAG,EAAE,CAAC,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;YACzG,CAAC,GAAG,EAAE,CACJ,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE;gBACxB,GAAG,EAAE,QAAQ;gBACb,aAAa,EAAE,wBAAwB;gBACvC,gBAAgB,EAAE,KAAK;aACxB,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;QAC3B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,+BAA+B,EAAE,GAAG,EAAE;QAC7C,IAAI,KAAK,CAAC;QACV,MAAM,mBAAmB,GAAG;YAC1B,QAAQ,EAAE,gBAAgB;YAC1B,QAAQ,EAAE,UAAU;YACpB,GAAG,EAAE,QAAQ;YACb,UAAU,EAAE,KAAK;YACjB,gBAAgB,EAAE,SAAS;YAC3B,QAAQ,EAAE,KAAK;SAChB,CAAC;QAEF,EAAE,CAAC,uBAAuB,EAAE,KAAK;YAC/B,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,gBAAgB,EAAE,2BAA2B,EAAS,CAAC,CAAC;YACzG,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,GAAG,CAAC;iBACzD,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,IAAI,EAAE;oBACJ,QAAQ,EAAE,gBAAgB;iBAC3B;gBACD,YAAY,EAAE,YAAY;aAC3B,CAAC,CAAC;YAEL,MAAM,KAAK,CAAC,YAAY,CAAC,mBAAmB,CAAC,CAAC;YAC9C,KAAK,CAAC,MAAM,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mEAAmE,EAAE,KAAK;YAC3E,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACnD,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,GAAG,CAAC;iBACzD,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,IAAI,EAAE;oBACJ,QAAQ,EAAE,gBAAgB;iBAC3B;gBACD,YAAY,EAAE,YAAY;aAC3B,CAAC,CAAC;YAEL,MAAM,KAAK,CAAC,YAAY,CAAC,mBAAmB,CAAC,CAAC;YAC9C,KAAK,CAAC,MAAM,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QAClC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;QAC9B,IAAI,KAAK,CAAC;QACV,MAAM,CAAC,GAAG,EAAE;YACV,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;QACpC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;YAChB,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;YAC7C,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;YAE/C,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,UAAU,EAAE,GAAG,EAAE;YAClB,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACnD,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,oCAAoC,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;YACxF,sBAAsB;YACtB,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,oCAAoC,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YAEzF,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YAClC,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,oCAAoC,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QAC1F,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE;YACd,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACnD,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,oCAAoC,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;YACxF,sBAAsB;YACtB,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,oCAAoC,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;YACzF,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YAClC,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAE,qCAAqC,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QAC3F,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;QAC/B,MAAM,QAAQ,GAAG,cAAc,CAAC;QAChC,MAAM,MAAM,GAAG,kBAAkB,CAAC;QAElC,EAAE,CAAC,kBAAkB,EAAE,GAAG,EAAE;YAC1B,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;YACpE,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,QAAQ,EAAE,WAAW,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACjF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gBAAgB,EAAE,GAAG,EAAE;YACxB,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;YACpE,KAAK,CAAC,OAAO,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAC5E,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,EAAE,CAAC,6BAA6B,EAAE,GAAG,EAAE;YACrC,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,MAAM,QAAQ,GAAG,KAAK,CAAC,sBAAsB,EAAE,CAAC;YAChD,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QACzB,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sDAAsD,EAAE,GAAG,EAAE;YAC9D,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;YACxC,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,EAAE,CAAC,EAAE,EAAE,CAAC;gBAC9B,MAAM,QAAQ,GAAG,KAAK,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC;gBAClD,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;gBACvB,iFAAiF;gBACjF,4EAA4E;gBAC5E,iDAAiD;gBACjD,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;YAC3C,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;QACrC,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;QACxC,MAAM,IAAI,GAAG,kEAAkE,CAAC;QAChF,MAAM,IAAI,GACR,iHAAiH,CAAC;QACpH,MAAM,SAAS,GAAG,CAAC,QAAQ,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QAExD,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,CAAC,GAAG,EAAE,CACJ,KAAK,CAAC,WAAW,CAAC;gBAChB,IAAI;gBACJ,SAAS,EAAE,CAAC,KAAK,CAAC;gBAClB,CAAC,EAAE,CAAC;aACL,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAC;QAChF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qDAAqD,EAAE,GAAG,EAAE;YAC7D,CAAC,GAAG,EAAE,CACJ,KAAK,CAAC,WAAW,CAAC;gBAChB,IAAI;gBACJ,SAAS,EAAE,CAAC,KAAK,CAAC;gBAClB,CAAC,EAAE,CAAC;aACL,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,8CAA8C,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iEAAiE,EAAE,GAAG,EAAE;YACzE,MAAM,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAC5E,MAAM,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACpD,MAAM,eAAe,GAAG,CAAC,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAChD,CAAC,GAAG,EAAE,CACJ,KAAK,CAAC,kBAAkB,CAAC;gBACvB,MAAM;gBACN,SAAS,EAAE,eAAe;gBAC1B,IAAI;aACE,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,wBAAwB,CAAC,CAAC;QACtD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;YAC9C,MAAM,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAC5E,MAAM,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACpD,MAAM,eAAe,GAAG,CAAC,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAChD,MAAM,iBAAiB,GAAG,KAAK,CAAC,kBAAkB,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;YAC3F,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YAC1C,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CACjC,iHAAiH,CAClH,CAAC;YACF,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CACjC,iHAAiH,CAClH,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;YACpD,MAAM,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAC5E,MAAM,OAAO,GAAG,KAAK,CAAC,YAAY,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,UAAU,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAS,CAAC,CAAC;YAC/F,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC9B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;YACxC,MAAM,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAC5E,MAAM,OAAO,GAAG,KAAK,CAAC,YAAY,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,UAAU,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;YAC9F,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;YACxD,MAAM,MAAM,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;YACxD,MAAM,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YACzE,MAAM,OAAO,GAAG,KAAK,CAAC,YAAY,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,UAAU,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;YACtG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,SAAS,MAAM,CAAC,IAAY;YAC1B,OAAO,iBAAM,CAAC,cAAc,CAC1B,gBAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,UAAoB,CACvF,CAAC;QACJ,CAAC;QAED,EAAE,CAAC,sDAAsD,EAAE,GAAG,EAAE;YAC9D,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;gBAC7B,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,CAAC,CAAC;gBACxC,MAAM,MAAM,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;gBAChC,MAAM,MAAM,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;gBAChC,MAAM,WAAW,GAAG,KAAK,CAAC,aAAa,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBAC7G,MAAM,WAAW,GAAG,KAAK,CAAC,aAAa,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBAC7G,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;gBAEtC,QAAQ,CAAC,EAAE,CAAC;oBACV,KAAK,CAAC;wBACJ,WAAW,CAAC,MAAM,CAAC,GAAG,CAAC,kEAAkE,CAAC,CAAC;wBAC3F,MAAM;oBACR,KAAK,CAAC;wBACJ,WAAW,CAAC,MAAM,CAAC,GAAG,CAAC,kEAAkE,CAAC,CAAC;wBAC3F,MAAM;gBACV,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iBAAiB,EAAE;QAC1B,IAAI,KAAK,CAAC;QACV,IAAI,KAAK,CAAC;QAEV,MAAM,CAAC,KAAK;YACV,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,YAAY,EAAE,cAAc;gBAC5B,IAAI,EAAE,EAAE,QAAQ,EAAE,4BAA4B,EAAE;aACjD,CAAC,CAAC;YAEL,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACnD,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,KAAK,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAEzB,MAAM,KAAK,CAAC,4BAA4B,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;YAE9D,KAAK,GAAG,iBAAM,CAAC,YAAY,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,GAAG,CAAC;QAClD,CAAC,CAAC,CAAC;QAEH,MAAM,WAAW,GAAG,aAAa,CAAC;QAClC,MAAM,WAAW,GAAG,aAAa,CAAC;QAClC,MAAM,aAAa,GAAG,eAAe,CAAC;QAEtC,QAAQ,CAAC,oCAAoC,EAAE,SAAS,YAAY;YAClE,EAAE,CAAC,iBAAiB,EAAE,KAAK;gBACzB,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,WAAW,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6BAA6B,CAAC,CAAC;gBAC1G,MAAM,KAAK;qBACR,cAAc,CAAC,EAAE,WAAW,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,EAAE,CAAC;qBAC1D,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6BAA6B,CAAC,CAAC;gBACzD,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,WAAW,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6BAA6B,CAAC,CAAC;gBAC1G,MAAM,KAAK;qBACR,cAAc,CAAC,EAAE,WAAW,EAAE,MAAM,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC;qBAC1D,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,6BAA6B,CAAC,CAAC;YAC3D,CAAC,CAAC,CAAC;YAEH,EAAE,CAAC,wBAAwB,EAAE,KAAK;gBAChC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC7E,MAAM,KAAK;qBACR,cAAc,CAAC,EAAE,WAAW,EAAE,WAAW,EAAE,CAAC;qBAC5C,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,uCAAuC,CAAC,CAAC;YACrE,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4BAA4B,EAAE,KAAK;YACpC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;YAE5E,IAAI,CAAC,KAAK,CAAC;iBACR,IAAI,CAAC,wBAAwB,CAAC;iBAC9B,KAAK,CAAC,GAAG,EAAE;gBACV,OAAO,EAAE,CAAC;gBACV,SAAS,EAAE;oBACT,MAAM,EAAE,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;oBACjE,MAAM,EAAE,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;oBACjE,MAAM,EAAE,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC;oBACnE,MAAM,EAAE,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;iBAClE;aACF,CAAC,CAAC;YAEL,IAAI,CAAC,KAAK,CAAC;iBACR,GAAG,CAAC,kBAAkB,CAAC;iBACvB,KAAK,CAAC,IAAI,CAAC;iBACX,KAAK,CAAC,GAAG,EAAE;gBACV,IAAI,EAAE;oBACJ;wBACE,GAAG,EAAE,QAAQ;wBACb,YAAY,EAAE,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;qBACxE;oBACD;wBACE,GAAG,EAAE,QAAQ;wBACb,YAAY,EAAE,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC;qBAC1E;iBACF;aACF,CAAC,CAAC;YAEL,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;YAE/D,MAAM,KAAK,CAAC,cAAc,CAAC,EAAE,WAAW,EAAE,WAAW,EAAE,CAAC,CAAC;QAC3D,CAAC,CAAC,CAAC;QAEH,SAAS,CAAC,SAAS,mBAAmB;YACpC,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;QAC7B,IAAI,KAAK,CAAC;QACV,MAAM,KAAK,GAAG,qEAAqE,CAAC;QAEpF,MAAM,CAAC,GAAG,EAAE;YACV,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,WAAW,EAAE,KAAK,EAAE,CAAC,CAAC;QACzE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;YACpD,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC;YACjC,IAAI,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,aAAa,CAAC;YAE/B,MAAM,UAAU,GAAG,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YACzC,MAAM,cAAc,GAAG,KAAK,CAAC,uBAAuB,CAAC,EAAE,GAAG,EAAE,UAAU,EAAE,KAAK,EAAE,CAAC,CAAC;YACjF,IAAI,CAAC,GAAG,GAAG,eAAe,CAAC;YAE3B,cAAc,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACrD,cAAc,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,kEAAkE,CAAC,CAAC;YAC1G,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,kEAAkE,CAAC,CAAC;QACvG,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,GAAG,EAAE;YACzD,MAAM,YAAY,GAAG;gBACnB,cAAc,EACZ,+OAA+O;gBACjP,cAAc,EAAE,8BAA8B;gBAC9C,iBAAiB,EACf,4SAA4S;aAC/S,CAAC;YACF,MAAM,wBAAwB,GAAG,KAAK,CAAC,mBAAmB,CAAC,YAAY,EAAE,gBAAgB,CAAC,CAAC;YAC3F,wBAAwB,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACnD,wBAAwB,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAC5C,iHAAiH,CAClH,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,yCAAyC,EAAE,KAAK;YACjD,MAAM,GAAG,GAAG,KAAK;iBACd,IAAI,CAAC,MAAM,CAAC;iBACZ,GAAG,CAAC,8FAA8F,CAAC,CAAC;YACvG,MAAM,iBAAiB,GAAG,KAAK,CAAC,uBAAuB,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;YACxE,MAAM,cAAc,GAAG;gBACrB,oBAAoB,EAAE,KAAK;gBAC3B,cAAc,EAAE,kBAAkB;gBAClC,gBAAgB,EAAE,iBAAiB,CAAC,SAAS;gBAC7C,aAAa,EAAE,SAAS,GAAG,iBAAiB,CAAC,SAAS;gBACtD,IAAI,EAAE,iBAAiB,CAAC,IAAI;aAC7B,CAAC;YACF,MAAM,YAAY,GAChB,igBAAigB,CAAC;YAEpgB,IAAI,CAAC,uBAAuB,EAAE,EAAE,UAAU,EAAE,cAAc,EAAE,CAAC;iBAC1D,GAAG,CAAC,0GAA0G,CAAC;iBAC/G,KAAK,CAAC,GAAG,EAAE,YAAY,EAAE;gBACxB,IAAI,EAAE,kEAAkE;gBACxE,SAAS,EAAE,eAAe;aAC3B,CAAC,CAAC;YAEL,MAAM,YAAY,GAAG,CAAC,MAAM,EAAE,CAAC;gBAC7B,GAAG,EAAE,GAAG;gBACR,MAAM,EAAE,KAAK;gBACb,OAAO,EAAE,cAAc;gBACvB,SAAS,EAAE,CAAC,IAAI,EAAE,QAAQ,EAAE,EAAE;oBAC5B,8BAA8B;oBAC9B,MAAM,GAAG,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC;oBAClC,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC;oBACnC,MAAM,SAAS,GAAG,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC;oBAC7C,MAAM,UAAU,GAAG,QAAQ,CAAC,UAAU,CAAC;oBACvC,MAAM,kBAAkB,GAAG;wBACzB,GAAG;wBACH,IAAI;wBACJ,SAAS;wBACT,KAAK;wBACL,UAAU;wBACV,IAAI,EAAE,IAAI;qBACX,CAAC;oBACF,OAAO,KAAK,CAAC,cAAc,CAAC,kBAAkB,CAAC,CAAC;gBAClD,CAAC;aACF,CAAC,CAAQ,CAAC;YACX,YAAY,CAAC,gBAAgB,CAAC,MAAM,CAAC,KAAK,CACxC,4nBAA4nB,CAC7nB,CAAC;YACF,YAAY,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,kEAAkE,CAAC,CAAC;YAC3G,YAAY,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC1C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK;YACzD,MAAM,GAAG,GAAG,uBAAuB,CAAC;YACpC,MAAM,IAAI,GAAG,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC;YAE9B,MAAM,aAAa,GAAG,UAAU,CAAC;YACjC,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC;YACjC,IAAI,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,aAAa,CAAC;YAE/B,IAAI,CAAC;gBACH,IAAI,CAAC,GAAG,CAAC;qBACN,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC;qBACf,KAAK,CAAC,GAAG,EAAE,SAAS,EAAE;oBACrB,IAAI,EAAE,kEAAkE;oBACxE,SAAS,EAAE,MAAM,CAAC,aAAa,CAAC;iBACjC,CAAC,CAAC;gBAEL,MAAM,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAQ,CAAC;gBACvD,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,kEAAkE,CAAC,CAAC;YAC5G,CAAC;oBAAS,CAAC;gBACT,IAAI,CAAC,GAAG,GAAG,eAAe,CAAC;YAC7B,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;YACvD,MAAM,kBAAkB,GAAG;gBACzB,GAAG,EAAE,wBAAwB;gBAC7B,IAAI,EAAE,kEAAkE;gBACxE,SAAS,EAAE,eAAe;gBAC1B,KAAK,EAAE,KAAK;gBACZ,UAAU,EAAE,GAAG;gBACf,IAAI,EAAE,UAAU;aACjB,CAAC;YACF,MAAM,mBAAmB,GAAG,KAAK,CAAC,cAAc,CAAC,kBAAkB,CAAC,CAAC;YACrE,mBAAmB,CAAC,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAAC;YACrF,mBAAmB,CAAC,gBAAgB,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;YAC1F,mBAAmB,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,kEAAkE,CAAC,CAAC;YAClH,mBAAmB,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kCAAkC,EAAE,GAAG,EAAE;YAC1C,MAAM,kBAAkB,GAAG;gBACzB,GAAG,EAAE,oBAAoB;gBACzB,IAAI,EAAE,kEAAkE;gBACxE,SAAS,EAAE,eAAe;gBAC1B,KAAK,EAAE,KAAK;gBACZ,UAAU,EAAE,GAAG;gBACf,IAAI,EAAE,UAAU;aACjB,CAAC;YACF,MAAM,mBAAmB,GAAG,KAAK,CAAC,cAAc,CAAC,kBAAkB,CAAC,CAAC;YACrE,mBAAmB,CAAC,gBAAgB,CAAC,MAAM,CAAC,KAAK,CAAC,8BAA8B,CAAC,CAAC;YAClF,mBAAmB,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,kEAAkE,CAAC,CAAC;YAClH,mBAAmB,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oFAAoF,EAAE,KAAK;YAC5F,MAAM,GAAG,GAAG,yBAAyB,CAAC;YACtC,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAE5C,mGAAmG;YACnG,MAAM,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,YAAY,CAAC,0DAA0D,CAAC,CAAC;YACxG,KAAK,CAAC,IAAI,EAAE,CAAC;QACf,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uEAAuE,EAAE,KAAK;YAC/E,MAAM,EAAE,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,gBAAgB,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,CAAC,CAAC;YACnG,MAAM,GAAG,GAAG,yBAAyB,CAAC;YACtC,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC;YAEvD,MAAM,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAQ,CAAC;YACvC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YACvC,KAAK,CAAC,IAAI,EAAE,CAAC;QACf,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,8BAA8B,EAAE;QACvC,EAAE,CAAC,qEAAqE,EAAE,KAAK;YAC7E,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAC3B,MAAM,SAAS,GAAG,KAAK,CAAC,YAAY,EAAE,CAAC;YACvC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;QACjF,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC,SAAS,qBAAqB;YAClC,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE;QAC9B,IAAI,KAAK,CAAC;QACV,IAAI,KAAK,CAAC;QACV,MAAM,CAAC;YACL,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACnD,KAAK,CAAC,kBAAkB,EAAE,CAAC;YAE3B,KAAK,GAAG,iBAAM,CAAC,YAAY,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,GAAG,CAAC;YAEhD,IAAI,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACpC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gBAAgB,EAAE,KAAK;YACxB,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACrC,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC;YACJ,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,gCAAgC,EAAE,GAAG,EAAE;QAC9C,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;QACzD,EAAE,CAAC,uDAAuD,EAAE;YAC1D,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,8BAA8B,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAS,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;YAChH,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,8BAA8B,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,EAAS,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CACjG,0BAA0B,CAC3B,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,cAAc,EAAE;QACvB,SAAS,CAAC,SAAS,oBAAoB;YACrC,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4FAA4F,EAAE,KAAK;YACpG,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,YAAY,EAAE,cAAc;gBAC5B,IAAI,EAAE,EAAE,QAAQ,EAAE,qBAAqB,EAAE;aAC1C,CAAC,CAAC;YACL,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,GAAG,CAAC,uBAAuB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,QAAQ,EAAE;oBACR,YAAY,EAAE,oBAAoB;iBACnC;aACF,CAAC,CAAC;YAEL,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,YAAY,CAAC;gBACxC,QAAQ,EAAE,qBAAqB;gBAC/B,QAAQ,EAAE,aAAa;gBACvB,GAAG,EAAE,QAAQ;gBACb,kBAAkB,EAAE,IAAI;aACzB,CAAC,CAAC;YAEH,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACzC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;QAChE,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,6GAA6G,EAAE,KAAK;YACrH,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,YAAY,EAAE,cAAc;gBAC5B,IAAI,EAAE,EAAE,QAAQ,EAAE,qBAAqB,EAAE;aAC1C,CAAC,CAAC;YACL;;eAEG;YACH,IAAI,CAAC,uBAAuB,CAAC,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE;gBACpE,QAAQ,EAAE,EAAE;aACb,CAAC,CAAC;YACH,IAAI,CAAC,uBAAuB,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE;gBAChE,IAAI,EAAE,WAAW;aAClB,CAAC,CAAC;YACH,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,GAAG,CAAC,uBAAuB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE;gBACV,QAAQ,EAAE;oBACR,YAAY,EAAE,WAAW;iBAC1B;aACF,CAAC,CAAC;YAEL,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,YAAY,CAAC;gBACxC,QAAQ,EAAE,qBAAqB;gBAC/B,QAAQ,EAAE,aAAa;gBACvB,GAAG,EAAE,QAAQ;gBACb,kBAAkB,EAAE,IAAI;aACzB,CAAC,CAAC;YAEH,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACzC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,wBAAwB,EAAE,GAAG,EAAE;QACtC,SAAS,CAAC,SAAS,oBAAoB;YACrC,IAAI,CAAC,QAAQ,EAAE,CAAC;YAChB,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;YAClD,MAAM,MAAM,GAAG,KAAK,CAAC;YACrB,MAAM,OAAO,GAAG,qHAAqH,MAAM,KAAK,CAAC;YACjJ,MAAM,OAAO,GAAG,MAAM,IAAA,6BAAkB,EAAC,WAAW,CAAC,CAAC;YAEtD,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,OAAO,EAAE;iBACT,GAAG,CAAC,0BAA0B,CAAC;iBAC/B,KAAK,CAAC,GAAG,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,EAAE,kBAAkB,EAAE,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;YAEnF,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE,KAAK,EAAE,GAAG,EAAE,WAAW,EAAE,EAAE;gBACrC,MAAM,CAAC,OAAO,WAAW,KAAK,QAAQ,CAAC,CAAC;gBACxC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;gBACjC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAClC,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;gBACxC,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;gBAC1C,OAAO;oBACL,YAAY,EAAE,cAAc;oBAC5B,IAAI,EAAE,EAAE,QAAQ,EAAE,qBAAqB,EAAE;iBAC1C,CAAC;YACJ,CAAC,CAAC,CAAC;YAEL,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC;YAC9D,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;YACpC,QAAQ,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mCAAmC,EAAE,KAAK,IAAI,EAAE;YACjD,MAAM,OAAO,GAAG,wHAAwH,CAAC;YACzI,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,IAAI,CAAC;gBACH,MAAM,KAAK,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC;gBAC7C,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;YAC3C,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,EAAE,uBAAuB,CAAC,CAAC;YACnD,CAAC;QACH,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,0CAA0C,EAAE,KAAK,IAAI,EAAE;YACxD,MAAM,OAAO,GAAG,+FAA+F,CAAC;YAChH,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,IAAI,CAAC;gBACH,MAAM,KAAK,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC;gBAC7C,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;YAC3C,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,EAAE,8BAA8B,CAAC,CAAC;YAC1D,CAAC;QACH,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;YACnD,MAAM,OAAO,GAAG,gGAAgG,CAAC;YACjH,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;YACzD,IAAI,CAAC;gBACH,MAAM,KAAK,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC;gBAC7C,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;YAC3C,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;gBACf,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;YACrC,CAAC;QACH,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;YACnD,MAAM,OAAO,GAAG,2HAA2H,CAAC;YAC5I,IAAI,CAAC,uBAAuB,CAAC;iBAC1B,IAAI,CAAC,sBAAsB,CAAC;iBAC5B,KAAK,CAAC,GAAG,EAAE,KAAK,IAAI,EAAE;gBACrB,OAAO;oBACL,IAAI,EAAE,EAAE,QAAQ,EAAE,qBAAqB,EAAE;iBAC1C,CAAC;YACJ,CAAC,CAAC,CAAC;YAEL,IAAI,CAAC;gBACH,MAAM,KAAK,GAAG,oBAAS,CAAC,QAAQ,CAAC,aAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,CAAC;gBACzD,MAAM,KAAK,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC;gBAC7C,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;YAC3C,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,EAAE,mDAAmD,CAAC,CAAC;YAC/E,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["//\n// Tests for BitGo Object\n//\n\nimport * as crypto from 'crypto';\nimport * as nock from 'nock';\nimport * as should from 'should';\nimport assert = require('assert');\n\nimport { common, generateGPGKeyPair } from '@bitgo/sdk-core';\nimport { bip32, ECPair } from '@bitgo/utxo-lib';\nimport * as _ from 'lodash';\nimport * as BitGoJS from '../../src/index';\nconst rp = require('request-promise');\n\nimport { TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../src/bitgo';\n\nnock.disableNetConnect();\n\ndescribe('BitGo Prototype Methods', function () {\n  describe('Version', () => {\n    it('version', function () {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.initializeTestVars();\n      const version = bitgo.version();\n      version.should.be.a.String();\n    });\n  });\n\n  describe('validate', () => {\n    it('should get', () => {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.getValidate().should.equal(true);\n    });\n\n    it('should set', () => {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.setValidate(false);\n      bitgo.getValidate().should.equal(false);\n      bitgo['_validate'].should.equal(false);\n    });\n  });\n\n  describe('Environments', () => {\n    it('production', () => {\n      BitGoJS.setNetwork('testnet');\n      TestBitGo.decorate(BitGo, { env: 'prod' });\n      BitGoJS.getNetwork().should.equal('bitcoin');\n    });\n\n    it('staging', () => {\n      BitGoJS.setNetwork('testnet');\n      TestBitGo.decorate(BitGo, { env: 'staging' });\n      BitGoJS.getNetwork().should.equal('testnet');\n    });\n\n    it('test', () => {\n      BitGoJS.setNetwork('bitcoin');\n      TestBitGo.decorate(BitGo, { env: 'test' });\n      BitGoJS.getNetwork().should.equal('testnet');\n    });\n\n    it('dev', () => {\n      TestBitGo.decorate(BitGo, { env: 'dev' });\n      BitGoJS.getNetwork().should.equal('testnet');\n    });\n\n    it('custom network (prod)', () => {\n      TestBitGo.decorate(BitGo, { customBitcoinNetwork: 'bitcoin', customRootURI: 'http://rooturi.example' });\n      BitGoJS.getNetwork().should.equal('bitcoin');\n    });\n\n    it('custom network (testnet)', () => {\n      TestBitGo.decorate(BitGo, { customBitcoinNetwork: 'testnet', customRootURI: 'http://rooturi.example' });\n      BitGoJS.getNetwork().should.equal('testnet');\n    });\n  });\n\n  describe('HMAC request verification', () => {\n    it('throws if HMAC request verification is disabled for non-prod environments', function () {\n      (() => TestBitGo.decorate(BitGo, { env: 'prod', hmacVerification: false })).should.throw(\n        /Cannot disable request HMAC verification in environment/\n      );\n      (() => TestBitGo.decorate(BitGo, { env: 'test', hmacVerification: false })).should.not.throw(\n        /Cannot disable request HMAC verification in environment/\n      );\n      (() => TestBitGo.decorate(BitGo, { env: 'adminProd', hmacVerification: false })).should.throw(\n        /Cannot disable request HMAC verification in environment/\n      );\n      (() => TestBitGo.decorate(BitGo, { env: 'adminTest', hmacVerification: false })).should.not.throw(\n        /Cannot disable request HMAC verification in environment/\n      );\n      (() =>\n        TestBitGo.decorate(BitGo, {\n          env: 'dev',\n          customRootURI: 'http://rooturi.example',\n          hmacVerification: false,\n        })).should.not.throw(/Cannot disable request HMAC verification in environment/);\n    });\n\n    it('allows disabling of HMAC request verification only for dev environments', function () {\n      (() => TestBitGo.decorate(BitGo, { env: 'dev', hmacVerification: false })).should.not.throw();\n      (() => TestBitGo.decorate(BitGo, { env: 'latest', hmacVerification: false })).should.not.throw();\n      (() => TestBitGo.decorate(BitGo, { env: 'adminDev', hmacVerification: false })).should.not.throw();\n      (() => TestBitGo.decorate(BitGo, { env: 'adminLatest', hmacVerification: false })).should.not.throw();\n      (() => TestBitGo.decorate(BitGo, { env: 'local', hmacVerification: false })).should.not.throw();\n      (() => TestBitGo.decorate(BitGo, { env: 'localNonSecure', hmacVerification: false })).should.not.throw();\n      (() =>\n        TestBitGo.decorate(BitGo, {\n          env: 'branch',\n          customRootURI: 'http://rooturi.example',\n          hmacVerification: false,\n        })).should.not.throw();\n    });\n  });\n\n  describe('Authenticate in Microservices', () => {\n    let bitgo;\n    const authenticateRequest = {\n      username: 'test@bitgo.com',\n      password: 'password',\n      otp: '000000',\n      extensible: false,\n      extensionAddress: 'address',\n      forceSMS: false,\n    };\n\n    it('goes to microservices', async function () {\n      bitgo = TestBitGo.decorate(BitGo, { env: 'mock', microservicesUri: 'https://microservices.uri' } as any);\n      const scope = nock(BitGoJS.Environments[bitgo.getEnv()].uri)\n        .post('/api/auth/v1/session')\n        .reply(200, {\n          user: {\n            username: 'test@bitgo.com',\n          },\n          access_token: 'token12356',\n        });\n\n      await bitgo.authenticate(authenticateRequest);\n      scope.isDone().should.be.true();\n    });\n\n    it('goes to microservices even when microservicesUri is not specified', async function () {\n      bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      const scope = nock(BitGoJS.Environments[bitgo.getEnv()].uri)\n        .post('/api/auth/v1/session')\n        .reply(200, {\n          user: {\n            username: 'test@bitgo.com',\n          },\n          access_token: 'token12356',\n        });\n\n      await bitgo.authenticate(authenticateRequest);\n      scope.isDone().should.be.true();\n    });\n  });\n\n  describe('Verify Address', () => {\n    let bitgo;\n    before(() => {\n      bitgo = TestBitGo.decorate(BitGo);\n    });\n\n    it('errors', () => {\n      (() => bitgo.verifyAddress()).should.throw();\n      (() => bitgo.verifyAddress({})).should.throw();\n\n      bitgo.verifyAddress({ address: 'xyzzy' }).should.be.false();\n    });\n\n    it('standard', () => {\n      bitgo = TestBitGo.decorate(BitGo, { env: 'prod' });\n      bitgo.verifyAddress({ address: '1Bu3bhwRmevHLAy1JrRB6AfcxfgDG2vXRd' }).should.be.true();\n      // wrong version byte:\n      bitgo.verifyAddress({ address: '9Ef7HsuByGBogqkjoF5Yng7MYkq5UCdmZz' }).should.be.false();\n\n      bitgo = TestBitGo.decorate(BitGo);\n      bitgo.verifyAddress({ address: 'n4DNhSiEaodqaiF9tLYXTCh4kFbdUzxBHs' }).should.be.true();\n    });\n\n    it('p2sh', () => {\n      bitgo = TestBitGo.decorate(BitGo, { env: 'prod' });\n      bitgo.verifyAddress({ address: '3QJmV3qfvL9SuYo34YihAf3sRCW3qSinyC' }).should.be.true();\n      // wrong version byte:\n      bitgo.verifyAddress({ address: 'HV8swrGkmeN7Xig4vENr93aQSrX4iHjg7D' }).should.be.false();\n      bitgo = TestBitGo.decorate(BitGo);\n      bitgo.verifyAddress({ address: '2NEeFWbfu4EA1rcKx48e82Mj8d6FKcWawZw' }).should.be.true();\n    });\n  });\n\n  describe('Encrypt/Decrypt', () => {\n    const password = 'mickey mouse';\n    const secret = 'this is a secret';\n\n    it('invalid password', () => {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.initializeTestVars();\n      const opaque = bitgo.encrypt({ password: password, input: secret });\n      (() => bitgo.decrypt({ password: 'hack hack', input: opaque })).should.throw();\n    });\n\n    it('valid password', () => {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.initializeTestVars();\n      const opaque = bitgo.encrypt({ password: password, input: secret });\n      bitgo.decrypt({ password: password, input: opaque }).should.equal(secret);\n    });\n  });\n\n  describe('Password Generation', () => {\n    it('generates a random password', () => {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.initializeTestVars();\n      const password = bitgo.generateRandomPassword();\n      should.exist(password);\n    });\n\n    it('generates a random password with a numWords argument', () => {\n      const bitgo = TestBitGo.decorate(BitGo);\n      bitgo.initializeTestVars();\n      for (let i = 0; i < 1000; i++) {\n        const password = bitgo.generateRandomPassword(10);\n        should.exist(password);\n        // randomly generated password should be 55 characters roughly 92.5% of the time,\n        // 54 characters roughly 7.5% of the time, 53 characters 0.001% of the time,\n        // and fewer than 53 characters very, very rarely\n        password.length.should.be.within(53, 55);\n      }\n    });\n  });\n\n  describe('Shamir Secret Sharing', () => {\n    const bitgo = TestBitGo.decorate(BitGo);\n    const seed = '8cc57dac9cdae42bf7848a2d12f2874d31eca1f9de8fe3f8fa13e7857b545d59';\n    const xpub =\n      'xpub661MyMwAqRbcEusRjkJ64BXgR8ddYsXbuDJfbRc3eZcZVEa2ygswDiFZQpHFsA5N211YDvi2N898h4KrcXcfsR8PLhjJaPUwCUqg1ptBBHN';\n    const passwords = ['mickey', 'mouse', 'donald', 'duck'];\n\n    it('should fail to split secret with wrong m', () => {\n      (() =>\n        bitgo.splitSecret({\n          seed,\n          passwords: ['abc'],\n          m: 0,\n        })).should.throw('m must be a positive integer greater than or equal to 2');\n    });\n\n    it('should fail to split secret with bad password count', () => {\n      (() =>\n        bitgo.splitSecret({\n          seed,\n          passwords: ['abc'],\n          m: 2,\n        })).should.throw('passwords array length cannot be less than m');\n    });\n\n    it('should split and fail to reconstitute secret with bad passwords', () => {\n      const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 3 });\n      const shards = _.at(splitSecret.seedShares, [0, 2]);\n      const subsetPasswords = _.at(passwords, [0, 3]);\n      (() =>\n        bitgo.reconstituteSecret({\n          shards,\n          passwords: subsetPasswords,\n          xpub,\n        } as any)).should.throw(/ccm: tag doesn't match/);\n    });\n\n    it('should split and reconstitute secret', () => {\n      const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 2 });\n      const shards = _.at(splitSecret.seedShares, [0, 2]);\n      const subsetPasswords = _.at(passwords, [0, 2]);\n      const reconstitutedSeed = bitgo.reconstituteSecret({ shards, passwords: subsetPasswords });\n      reconstitutedSeed.seed.should.equal(seed);\n      reconstitutedSeed.xpub.should.equal(\n        'xpub661MyMwAqRbcEusRjkJ64BXgR8ddYsXbuDJfbRc3eZcZVEa2ygswDiFZQpHFsA5N211YDvi2N898h4KrcXcfsR8PLhjJaPUwCUqg1ptBBHN'\n      );\n      reconstitutedSeed.xprv.should.equal(\n        'xprv9s21ZrQH143K2Rnxdim5h3aws6o99QokXzP4o3CS6E5acSEtS9Zgfuw5ZWujhTHTWEAZDfmP3yxA1Ccn6myVkGEpRrT4xWgaEpoW7YiBAtC'\n      );\n    });\n\n    it('should split and incorrectly verify secret', () => {\n      const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 3 });\n      const isValid = bitgo.verifyShards({ shards: splitSecret.seedShares, passwords, m: 2 } as any);\n      isValid.should.equal(false);\n    });\n\n    it('should split and verify secret', () => {\n      const splitSecret = bitgo.splitSecret({ seed, passwords: passwords, m: 2 });\n      const isValid = bitgo.verifyShards({ shards: splitSecret.seedShares, passwords, m: 2, xpub });\n      isValid.should.equal(true);\n    });\n\n    it('should split and verify secret with many parts', () => {\n      const allPws = ['0', '1', '2', '3', '4', '5', '6', '7'];\n      const splitSecret = bitgo.splitSecret({ seed, passwords: allPws, m: 3 });\n      const isValid = bitgo.verifyShards({ shards: splitSecret.seedShares, passwords: allPws, m: 3, xpub });\n      isValid.should.equal(true);\n    });\n  });\n\n  describe('ECDH sharing secret', () => {\n    function getKey(seed: string) {\n      return ECPair.fromPrivateKey(\n        bip32.fromSeed(crypto.createHash('sha256').update(seed).digest()).privateKey as Buffer\n      );\n    }\n\n    it('should calculate a new ECDH sharing secret correctly', () => {\n      for (let i = 0; i < 256; i++) {\n        const bitgo = TestBitGo.decorate(BitGo);\n        const eckey1 = getKey(`${i}.a`);\n        const eckey2 = getKey(`${i}.b`);\n        const sharingKey1 = bitgo.getECDHSecret({ eckey: eckey1, otherPubKeyHex: eckey2.publicKey.toString('hex') });\n        const sharingKey2 = bitgo.getECDHSecret({ eckey: eckey2, otherPubKeyHex: eckey1.publicKey.toString('hex') });\n        sharingKey1.should.equal(sharingKey2);\n\n        switch (i) {\n          case 0:\n            sharingKey1.should.eql('465ffe5745325998b83fb39631347148e24d4f21b3f3b54739c264d5c42db4b8');\n            break;\n          case 1:\n            sharingKey1.should.eql('61ff44fc1af8061a433a314b7b8be8ae352c10f62aac5887047dbaa5643b818d');\n            break;\n        }\n      }\n    });\n  });\n\n  describe('change password', function () {\n    let bitgo;\n    let bgUrl;\n\n    before(async function () {\n      nock('https://bitgo.fakeurl')\n        .post('/api/auth/v1/session')\n        .reply(200, {\n          access_token: 'access_token',\n          user: { username: 'update_pw_tester@bitgo.com' },\n        });\n\n      bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      bitgo.initializeTestVars();\n      bitgo.setValidate(false);\n\n      await bitgo.authenticateChangePWTestUser(bitgo.testUserOTP());\n\n      bgUrl = common.Environments[bitgo.getEnv()].uri;\n    });\n\n    const oldPassword = 'oldPassword';\n    const newPassword = 'newPassword';\n    const otherPassword = 'otherPassword';\n\n    describe('should fail to change the password', function changePWFail() {\n      it('wrong arguments', async function () {\n        await bitgo.changePassword({ newPassword: '5678' }).should.be.rejectedWith('expected string oldPassword');\n        await bitgo\n          .changePassword({ oldPassword: 1234, newPassword: '5678' })\n          .should.be.rejectedWith('expected string oldPassword');\n        await bitgo.changePassword({ oldPassword: '1234' }).should.be.rejectedWith('expected string newPassword');\n        await bitgo\n          .changePassword({ oldPassword: '1234', newPassword: 5678 })\n          .should.be.rejectedWith('expected string newPassword');\n      });\n\n      it('incorrect old password', async function () {\n        nock(bgUrl).post('/api/v1/user/verifypassword').reply(200, { valid: false });\n        await bitgo\n          .changePassword({ oldPassword, newPassword })\n          .should.be.rejectedWith('the provided oldPassword is incorrect');\n      });\n    });\n\n    it('successful password change', async function () {\n      nock(bgUrl).post('/api/v1/user/verifypassword').reply(200, { valid: true });\n\n      nock(bgUrl)\n        .post('/api/v1/user/encrypted')\n        .reply(200, {\n          version: 1,\n          keychains: {\n            xpub11: bitgo.encrypt({ input: 'xprv11', password: oldPassword }),\n            xpub12: bitgo.encrypt({ input: 'xprv12', password: oldPassword }),\n            xpub13: bitgo.encrypt({ input: 'xprv13', password: otherPassword }),\n            xpub14: bitgo.encrypt({ input: 'xprv14', password: oldPassword }),\n          },\n        });\n\n      nock(bgUrl)\n        .get('/api/v2/tbtc/key')\n        .query(true)\n        .reply(200, {\n          keys: [\n            {\n              pub: 'xpub21',\n              encryptedPrv: bitgo.encrypt({ input: 'xprv21', password: oldPassword }),\n            },\n            {\n              pub: 'xpub22',\n              encryptedPrv: bitgo.encrypt({ input: 'xprv22', password: otherPassword }),\n            },\n          ],\n        });\n\n      nock(bgUrl).post('/api/v1/user/changepassword').reply(200, {});\n\n      await bitgo.changePassword({ oldPassword, newPassword });\n    });\n\n    afterEach(function afterChangePassword() {\n      nock.pendingMocks().should.be.empty();\n    });\n  });\n\n  describe('HMAC Handling', () => {\n    let bitgo;\n    const token = 'v2x5b735fed2486593f8fea19113e5c717308f90a5fb00e740e46c7bfdcc078cfd0';\n\n    before(() => {\n      bitgo = TestBitGo.decorate(BitGo, { env: 'mock', accessToken: token });\n    });\n\n    it('should correctly calculate request headers', () => {\n      const originalDateNow = Date.now;\n      Date.now = () => 1521589882510;\n\n      const fetchMeUrl = bitgo.url('/user/me');\n      const requestHeaders = bitgo.calculateRequestHeaders({ url: fetchMeUrl, token });\n      Date.now = originalDateNow;\n\n      requestHeaders.timestamp.should.equal(1521589882510);\n      requestHeaders.tokenHash.should.equal('a85af08e6723e41acd6a3fb9ef58422082e673df33c58e1db175bb740a2c934d');\n      requestHeaders.hmac.should.equal('6de77d5a5446a3e5649456c11480706a71071b15639c3c787af65bdb02ecf1ec');\n    });\n\n    it('should correctly handle authentication response', () => {\n      const responseJson = {\n        encryptedToken:\n          '{\"iv\":\"EqxVaGTLY4naAYkuBaTz0w==\",\"v\":1,\"iter\":1000,\"ks\":128,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"4S4dBYcgL4s=\",\"ct\":\"FgBRJljb8iSYxnAjMi4Qotr7sTKbSmWnlfHZShMSi8YeeE3kiS8bpHNUwAPhY8tgouh3UsEwrJnY+54MvqFD7yd19pG1V4CVssr8\"}',\n        derivationPath: 'm/999999/104490948/173846667',\n        encryptedECDHXprv:\n          '{\"iv\":\"QKHEF2GNcwOJwy6+pwANRA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"W2sVFvXDlOw=\",\"ct\":\"8BTCqS25X37kLzmzQdGenhXH6znn9qEmkszAeS8kLnRdqKSiUiC7bTAVgg/Np5yrV7F7Jyiq+MTpVT76EoUT+PMJzArv0gUQKC2JPB3JuVKeAAVWBQmhWfkEwRfyv4hq4WMxwZtocwBqThvd2pJm9HE51GX4/Wo=\"}',\n      };\n      const parsedAuthenticationData = bitgo.handleTokenIssuance(responseJson, 'test@bitgo.com');\n      parsedAuthenticationData.token.should.equal(token);\n      parsedAuthenticationData.ecdhXprv.should.equal(\n        'xprv9s21ZrQH143K3si1bKGp7KqgCQv39ttQ7aUwWzVdytgHd8HtDCHyEp14mxfhiT3qHTq4BaSrA7uUkG6AJTfPJBsRu63drvBqYuMZyTxepH7'\n      );\n    });\n\n    it('should correctly verify a response hmac', async function () {\n      const url = bitgo\n        .coin('tltc')\n        .url('/wallet/5941b202b42fcbc707170d5b597491d9/address/QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE?segwit=1');\n      const requestHeaderData = bitgo.calculateRequestHeaders({ url, token });\n      const requestHeaders = {\n        'BitGo-Auth-Version': '2.0',\n        'Content-Type': 'application/json',\n        'Auth-Timestamp': requestHeaderData.timestamp,\n        Authorization: 'Bearer ' + requestHeaderData.tokenHash,\n        HMAC: requestHeaderData.hmac,\n      };\n      const responseBody =\n        '{\"id\":\"5a7ca8bcaf52c8e807c575fb692609ec\",\"address\":\"QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE\",\"chain\":0,\"index\":2,\"coin\":\"tltc\",\"wallet\":\"5941b202b42fcbc707170d5b597491d9\",\"coinSpecific\":{\"redeemScript\":\"522102835bcfd130f7a56f72c905b782d90b66e22f88ad3309cf72af5138a7d44be8b3210322c7f42a1eb212868eab78db7ba64846075d98c7f4c7aa25a02e57871039e0cd210265825be0d5bf957fb72abd7c23bf0836a78a15f951a073467cd5c99e03ce7ab753ae\"},\"balance\":{\"updated\":\"2018-02-28T23:48:07.341Z\",\"numTx\":1,\"numUnspents\":1,\"totalReceived\":20000000}}';\n\n      nock('https://bitgo.fakeurl', { reqheaders: requestHeaders })\n        .get('/api/v2/tltc/wallet/5941b202b42fcbc707170d5b597491d9/address/QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE?segwit=1')\n        .reply(200, responseBody, {\n          hmac: '30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec',\n          timestamp: '1521590532925',\n        });\n\n      const responseData = (await rp({\n        uri: url,\n        method: 'GET',\n        headers: requestHeaders,\n        transform: (body, response) => {\n          // verify the response headers\n          const url = response.request.href;\n          const hmac = response.headers.hmac;\n          const timestamp = response.headers.timestamp;\n          const statusCode = response.statusCode;\n          const verificationParams = {\n            url,\n            hmac,\n            timestamp,\n            token,\n            statusCode,\n            text: body,\n          };\n          return bitgo.verifyResponse(verificationParams);\n        },\n      })) as any;\n      responseData.signatureSubject.should.equal(\n        '1521590532925|/api/v2/tltc/wallet/5941b202b42fcbc707170d5b597491d9/address/QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE?segwit=1|200|{\"id\":\"5a7ca8bcaf52c8e807c575fb692609ec\",\"address\":\"QNc4RFAcbvqmtrR1kR2wbGLCx6tEvojFYE\",\"chain\":0,\"index\":2,\"coin\":\"tltc\",\"wallet\":\"5941b202b42fcbc707170d5b597491d9\",\"coinSpecific\":{\"redeemScript\":\"522102835bcfd130f7a56f72c905b782d90b66e22f88ad3309cf72af5138a7d44be8b3210322c7f42a1eb212868eab78db7ba64846075d98c7f4c7aa25a02e57871039e0cd210265825be0d5bf957fb72abd7c23bf0836a78a15f951a073467cd5c99e03ce7ab753ae\"},\"balance\":{\"updated\":\"2018-02-28T23:48:07.341Z\",\"numTx\":1,\"numUnspents\":1,\"totalReceived\":20000000}}'\n      );\n      responseData.expectedHmac.should.equal('30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec');\n      responseData.isValid.should.equal(true);\n    });\n\n    it('should include request body as part of the hmac', async function () {\n      const url = 'https://bitgo.fakeurl';\n      const body = { test: 'test' };\n\n      const fixedUnixTime = 1627374646;\n      const originalDateNow = Date.now;\n      Date.now = () => fixedUnixTime;\n\n      try {\n        nock(url)\n          .post('/', body)\n          .reply(201, undefined, {\n            hmac: '677e0c9a65ca384415945cb19b40ae38eaadfbce3ccce8c5d7bf37c1973b2553',\n            timestamp: String(fixedUnixTime),\n          });\n\n        const resp = (await bitgo.post(url).send(body)) as any;\n        resp.req.headers['hmac'].should.equal('4425a4004ef2724add25b4dd019d21c66394653a049d82e37df3a2c356b5706d');\n      } finally {\n        Date.now = originalDateNow;\n      }\n    });\n\n    it('should recognize trailing slash inconsistency', () => {\n      const verificationParams = {\n        url: 'https://google.com/api',\n        hmac: '30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec',\n        timestamp: '1521590532925',\n        token: token,\n        statusCode: 200,\n        text: 'fakedata',\n      };\n      const verificationDetails = bitgo.verifyResponse(verificationParams);\n      verificationDetails.signatureSubject.should.equal('1521590532925|/api|200|fakedata');\n      verificationDetails.signatureSubject.should.not.equal('1521590532925|/api/|200|fakedata');\n      verificationDetails.expectedHmac.should.equal('2064f2adb168ef8808f6a42f588d7d6bc14e98e8b41239c6bbb7349e52f2249a');\n      verificationDetails.isValid.should.equal(false);\n    });\n\n    it('should auto-amend trailing slash', () => {\n      const verificationParams = {\n        url: 'https://google.com',\n        hmac: '30a5943043ab4b0503d807f0cca7dac3a670e8785331322567db5189432b87ec',\n        timestamp: '1521590532925',\n        token: token,\n        statusCode: 200,\n        text: 'fakedata',\n      };\n      const verificationDetails = bitgo.verifyResponse(verificationParams);\n      verificationDetails.signatureSubject.should.equal('1521590532925|/|200|fakedata');\n      verificationDetails.expectedHmac.should.equal('51c6d024f261e166e8a323f8fa36a9bb8d4d02b076334c2a9ae0a49efc5724d4');\n      verificationDetails.isValid.should.equal(false);\n    });\n\n    it('should throw if hmac validation is enabled, and no valid hmac headers are returned', async function () {\n      const url = 'https://fakeurl.invalid';\n      const scope = nock(url).get('/').reply(200);\n\n      // test suite bitgo object has hmac verification enabled, so it should throw when the nock responds\n      await bitgo.get(url).should.be.rejectedWith(/invalid response HMAC, possible man-in-the-middle-attack/);\n      scope.done();\n    });\n\n    it('should not enforce hmac verification if hmac verification is disabled', async function () {\n      const bg = TestBitGo.decorate(BitGo, { env: 'mock', hmacVerification: false, accessToken: token });\n      const url = 'https://fakeurl.invalid';\n      const scope = nock(url).get('/').reply(200, { ok: 1 });\n\n      const res = (await bg.get(url)) as any;\n      res.body.should.have.property('ok', 1);\n      scope.done();\n    });\n  });\n\n  describe('Token Definitions at Startup', function () {\n    it('Should return a non-empty list of tokens before the server responds', async function () {\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      bitgo.initializeTestVars();\n      const constants = bitgo.getConstants();\n      constants.should.have.propertyByPath('eth', 'tokens', 'length').greaterThan(0);\n    });\n\n    after(function tokenDefinitionsAfter() {\n      nock.pendingMocks().should.be.empty();\n    });\n  });\n\n  describe('superagent wrappers', function () {\n    let bitgo;\n    let bgUrl;\n    before(function () {\n      bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      bitgo.initializeTestVars();\n\n      bgUrl = common.Environments[bitgo.getEnv()].uri;\n\n      nock(bgUrl).patch('/').reply(200);\n    });\n\n    it('PATCH requests', async function () {\n      const res = await bitgo.patch(bgUrl);\n      res.status.should.equal(200);\n    });\n\n    after(function () {\n      nock.pendingMocks().should.be.empty();\n    });\n  });\n\n  describe('preprocessAuthenticationParams', () => {\n    const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    it('should fail if passed non-string username or password', function () {\n      (() => bitgo.preprocessAuthenticationParams({ username: 123 } as any)).should.throw(/expected string username/);\n      (() => bitgo.preprocessAuthenticationParams({ username: 'abc', password: {} } as any)).should.throw(\n        /expected string password/\n      );\n    });\n  });\n\n  describe('authenticate', function () {\n    afterEach(function ensureNoPendingMocks() {\n      nock.pendingMocks().should.be.empty();\n    });\n\n    it('should get the ecdhKeychain if ensureEcdhKeychain is set and user already has ecdhKeychain', async function () {\n      nock('https://bitgo.fakeurl')\n        .post('/api/auth/v1/session')\n        .reply(200, {\n          access_token: 'access_token',\n          user: { username: 'auth-test@bitgo.com' },\n        });\n      nock('https://bitgo.fakeurl')\n        .get('/api/v1/user/settings')\n        .reply(200, {\n          settings: {\n            ecdhKeychain: 'some-existing-xpub',\n          },\n        });\n\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      const response = await bitgo.authenticate({\n        username: 'auth-test@bitgo.com',\n        password: 'password123',\n        otp: '000000',\n        ensureEcdhKeychain: true,\n      });\n\n      should.exist(response.user.ecdhKeychain);\n      response.user.ecdhKeychain.should.equal('some-existing-xpub');\n    });\n    it('should create the ecdhKeychain if ensureEcdhKeychain is set and the user does not already have ecdhKeychain', async function () {\n      nock('https://bitgo.fakeurl')\n        .post('/api/auth/v1/session')\n        .reply(200, {\n          access_token: 'access_token',\n          user: { username: 'auth-test@bitgo.com' },\n        });\n      /**\n       * This is {} because want to make sure the user has no ecdhXpub set before we set it\n       */\n      nock('https://bitgo.fakeurl').get('/api/v1/user/settings').reply(200, {\n        settings: {},\n      });\n      nock('https://bitgo.fakeurl').post('/api/v1/keychain').reply(200, {\n        xpub: 'some-xpub',\n      });\n      nock('https://bitgo.fakeurl')\n        .put('/api/v2/user/settings')\n        .reply(200, {\n          settings: {\n            ecdhKeychain: 'some-xpub',\n          },\n        });\n\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      const response = await bitgo.authenticate({\n        username: 'auth-test@bitgo.com',\n        password: 'password123',\n        otp: '000000',\n        ensureEcdhKeychain: true,\n      });\n\n      should.exist(response.user.ecdhKeychain);\n      response.user.ecdhKeychain.should.equal('some-xpub');\n    });\n  });\n\n  describe('passkey authentication', () => {\n    afterEach(function ensureNoPendingMocks() {\n      nock.cleanAll();\n      nock.pendingMocks().should.be.empty();\n    });\n\n    it('should authenticate with a passkey', async () => {\n      const userId = '123';\n      const passkey = `{\"id\": \"id\", \"response\": {\"authenticatorData\": \"123\", \"clientDataJSON\": \"123\", \"signature\": \"123\", \"userHandle\": \"${userId}\"}}`;\n      const keyPair = await generateGPGKeyPair('secp256k1');\n\n      nock('https://bitgo.fakeurl')\n        .persist()\n        .get('/api/v1/client/constants')\n        .reply(200, { ttl: 3600, constants: { passkeyBitGoGpgKey: keyPair.publicKey } });\n\n      nock('https://bitgo.fakeurl')\n        .post('/api/auth/v1/session')\n        .reply(200, async (uri, requestBody) => {\n          assert(typeof requestBody === 'object');\n          should.exist(requestBody.userId);\n          should.exist(requestBody.passkey);\n          requestBody.userId.should.equal(userId);\n          requestBody.passkey.should.equal(passkey);\n          return {\n            access_token: 'access_token',\n            user: { username: 'auth-test@bitgo.com' },\n          };\n        });\n\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      const response = await bitgo.authenticateWithPasskey(passkey);\n      should.exist(response.access_token);\n      response.access_token.should.equal('access_token');\n    });\n\n    it('should throw - invalid userHandle', async () => {\n      const passkey = `{\"id\": \"id\", \"response\": {\"authenticatorData\": \"123\", \"clientDataJSON\": \"123\", \"signature\": \"123\", \"userHandle\": 123}}`;\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      try {\n        await bitgo.validatePasskeyResponse(passkey);\n        assert.fail('Expected error not thrown');\n      } catch (e) {\n        assert.equal(e.message, 'userHandle is missing');\n      }\n    });\n    it('should throw - invalid authenticatorData', async () => {\n      const passkey = `{\"id\": \"id\", \"response\": { \"clientDataJSON\": \"123\", \"signature\": \"123\", \"userHandle\": \"123\"}}`;\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      try {\n        await bitgo.validatePasskeyResponse(passkey);\n        assert.fail('Expected error not thrown');\n      } catch (e) {\n        assert.equal(e.message, 'authenticatorData is missing');\n      }\n    });\n    it('should throw - invalid passkey json', async () => {\n      const passkey = `{{\"id\": \"id\", \"response\": { \"clientDataJSON\": \"123\", \"signature\": \"123\", \"userHandle\": \"123\"}}`;\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      try {\n        await bitgo.validatePasskeyResponse(passkey);\n        assert.fail('Expected error not thrown');\n      } catch (e) {\n        console.log(e);\n        assert(e.message.includes('JSON'));\n      }\n    });\n    it('should throw - missing access token', async () => {\n      const passkey = `{\"id\": \"id\", \"response\": { \"authenticatorData\": \"123\", \"clientDataJSON\": \"123\", \"signature\": \"123\", \"userHandle\": \"123\"}}`;\n      nock('https://bitgo.fakeurl')\n        .post('/api/auth/v1/session')\n        .reply(200, async () => {\n          return {\n            user: { username: 'auth-test@bitgo.com' },\n          };\n        });\n\n      try {\n        const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n        await bitgo.authenticateWithPasskey(passkey);\n        assert.fail('Expected error not thrown');\n      } catch (e) {\n        assert.equal(e.message, 'Failed to login. Please contact support@bitgo.com');\n      }\n    });\n  });\n});\n"]}

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


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