PHP WebShell

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

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

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const assert = require("assert");
const sdk_test_1 = require("@bitgo/sdk-test");
const nock = require("nock");
const sdk_core_1 = require("@bitgo/sdk-core");
const abstract_lightning_1 = require("@bitgo/abstract-lightning");
const src_1 = require("../../../../src");
describe('Lightning wallets', function () {
    const coinName = 'tlnbtc';
    const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
    let basecoin;
    let wallets;
    let bgUrl;
    const userAuthKey = {
        id: 'def',
        pub: 'xpub661MyMwAqRbcGYjYsnsDj1SHdiXynWEXNnfNgMSpokN54FKyMqbu7rWEfVNDs6uAJmz86UVFtq4sefhQpXZhSAzQcL9zrEPtiLNNZoeSxCG',
        encryptedPrv: '{"iv":"zYhhaNdW0wPfJEoBjZ4pvg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"tgAMua9jjhw=","ct":"HcrbxQvNlWG5tLMndYzdNCYa1l+1h7o+vSsweA0+q1le3tWt6jLUJSEjZN+JI8lTZ2KPFQgLulQQhsUa+ytUCBi0vSgjF7x7CprT7l2Cfjkew00XsEd7wnmtJUsrQk8m69Co7tIRA3oEgzrnYwy4qOM81lbNNyQ="}',
        source: 'user',
        coinSpecific: {
            tlnbtc: {
                purpose: 'userAuth',
            },
        },
    };
    const nodeAuthKey = {
        id: 'ghi',
        pub: 'xpub661MyMwAqRbcG9xnTnAnRbJPo3MAHyRtH4zeehN8exYk4VFz5buepUzebhix33BKhS5Eb4V3LEfW5pYiSR8qmaEnyrpeghhKY8JfzAsUDpq',
        encryptedPrv: '{"iv":"bH6eGbnl9x8PZECPrgvcng==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"o8yknV6nTI8=","ct":"nGyzAToIzYkQeIdcVafoWHtMx7+Fgj0YldCme3WA1yxJAA0QulZVhblMZN/7efCRIumA0NNmpH7dxH6n8cVlz/Z+RUgC2q9lgvZKUoJcYNTjWUfkmkJutXX2tr8yVxm+eC/hnRiyfVLZ2qPxctvDlBVBfgLuPyc="}',
        source: 'user',
        coinSpecific: {
            tlnbtc: {
                purpose: 'nodeAuth',
            },
        },
    };
    before(function () {
        bitgo.initializeTestVars();
        basecoin = bitgo.coin(coinName);
        wallets = basecoin.wallets();
        bgUrl = src_1.common.Environments[bitgo.getEnv()].uri;
    });
    after(function () {
        nock.cleanAll();
        nock.pendingMocks().length.should.equal(0);
    });
    describe('Generate lightning wallet', function () {
        it('should validate parameters', async function () {
            await wallets
                .generateWallet({
                passphrase: 'pass123',
                enterprise: 'ent123',
                passcodeEncryptionCode: 'code123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.label, expected string.");
            await wallets
                .generateWallet({
                label: 'my ln wallet',
                enterprise: 'ent123',
                passcodeEncryptionCode: 'code123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.passphrase, expected string.");
            await wallets
                .generateWallet({
                label: 'my ln wallet',
                passphrase: 'pass123',
                passcodeEncryptionCode: 'code123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.enterprise, expected string.");
            await wallets
                .generateWallet({
                label: 'my ln wallet',
                passphrase: 'pass123',
                enterprise: 'ent123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.passcodeEncryptionCode, expected string.");
            await wallets
                .generateWallet({
                label: 123,
                passphrase: 'pass123',
                enterprise: 'ent123',
                passcodeEncryptionCode: 'code123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.label, expected string.");
            await wallets
                .generateWallet({
                label: 'my ln wallet',
                passphrase: 123,
                enterprise: 'ent123',
                passcodeEncryptionCode: 'code123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.passphrase, expected string.");
            await wallets
                .generateWallet({
                label: 'my ln wallet',
                passphrase: 'pass123',
                enterprise: 123,
                passcodeEncryptionCode: 'code123',
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.enterprise, expected string.");
            await wallets
                .generateWallet({
                label: 'my ln wallet',
                passphrase: 'pass123',
                enterprise: 'ent123',
                passcodeEncryptionCode: 123,
                subType: 'lightningCustody',
            })
                .should.be.rejectedWith("error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.passcodeEncryptionCode, expected string.");
        });
        for (const subType of ['lightningCustody', 'lightningSelfCustody']) {
            it(`should generate ${subType} lightning wallet`, async function () {
                const params = {
                    label: 'my ln wallet',
                    passphrase: 'pass123',
                    enterprise: 'ent123',
                    passcodeEncryptionCode: 'code123',
                    subType: subType,
                };
                const validateKeyRequest = (body) => {
                    const baseChecks = body.pub.startsWith('xpub') &&
                        !!body.encryptedPrv &&
                        body.keyType === 'independent' &&
                        body.source === 'user';
                    if (body.originalPasscodeEncryptionCode !== undefined) {
                        return baseChecks && body.originalPasscodeEncryptionCode === 'code123' && body.coinSpecific === undefined;
                    }
                    else {
                        const coinSpecific = body.coinSpecific && body.coinSpecific.tlnbtc;
                        return baseChecks && !!coinSpecific && ['userAuth', 'nodeAuth'].includes(coinSpecific.purpose);
                    }
                };
                const validateWalletRequest = (body) => {
                    return (body.label === 'my ln wallet' &&
                        body.m === 1 &&
                        body.n === 1 &&
                        body.type === 'hot' &&
                        body.subType === subType &&
                        body.enterprise === 'ent123' &&
                        Array.isArray(body.keys) &&
                        body.keys.length === 1 &&
                        body.keys[0] === 'keyId1' &&
                        body.coinSpecific &&
                        body.coinSpecific.tlnbtc &&
                        Array.isArray(body.coinSpecific.tlnbtc.keys) &&
                        body.coinSpecific.tlnbtc.keys.length === 2 &&
                        body.coinSpecific.tlnbtc.keys.includes('keyId2') &&
                        body.coinSpecific.tlnbtc.keys.includes('keyId3'));
                };
                nock(bgUrl)
                    .post('/api/v2/' + coinName + '/key', (body) => validateKeyRequest(body))
                    .reply(200, { id: 'keyId1' });
                nock(bgUrl)
                    .post('/api/v2/' + coinName + '/key', (body) => validateKeyRequest(body))
                    .reply(200, { id: 'keyId2' });
                nock(bgUrl)
                    .post('/api/v2/' + coinName + '/key', (body) => validateKeyRequest(body))
                    .reply(200, { id: 'keyId3' });
                nock(bgUrl)
                    .post('/api/v2/' + coinName + '/wallet/add', (body) => validateWalletRequest(body))
                    .reply(200, { id: 'walletId' });
                const response = await wallets.generateWallet(params);
                assert.ok(response.wallet);
                assert.ok(response.encryptedWalletPassphrase);
                assert.equal(bitgo.decrypt({ input: response.encryptedWalletPassphrase, password: params.passcodeEncryptionCode }), params.passphrase);
            });
        }
    });
    describe('invoices', function () {
        let wallet;
        beforeEach(function () {
            wallet = (0, abstract_lightning_1.getLightningWallet)(new src_1.Wallet(bitgo, basecoin, {
                id: 'walletId',
                coin: 'tlnbtc',
                subType: 'lightningCustody',
                coinSpecific: { keys: ['def', 'ghi'] },
            }));
        });
        it('should list invoices', async function () {
            const invoice = {
                valueMsat: 1000n,
                paymentHash: 'foo',
                invoice: 'tlnfoobar',
                walletId: wallet.wallet.id(),
                status: 'open',
                expiresAt: new Date(),
                createdAt: new Date(),
                updatedAt: new Date(),
            };
            const query = {
                status: 'open',
                startDate: new Date(),
                limit: 100n,
            };
            const listInvoicesNock = nock(bgUrl)
                .get(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`)
                .query(abstract_lightning_1.InvoiceQuery.encode(query))
                .reply(200, [abstract_lightning_1.InvoiceInfo.encode(invoice)]);
            const invoiceResponse = await wallet.listInvoices(query);
            assert.strictEqual(invoiceResponse.length, 1);
            assert.deepStrictEqual(invoiceResponse[0], invoice);
            listInvoicesNock.done();
        });
        it('listInvoices should throw error if wp response is invalid', async function () {
            const listInvoicesNock = nock(bgUrl)
                .get(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`)
                .reply(200, [{ valueMsat: '1000' }]);
            await assert.rejects(async () => await wallet.listInvoices({}), /Invalid list invoices response/);
            listInvoicesNock.done();
        });
        it('should create invoice', async function () {
            const createInvoice = {
                valueMsat: 1000n,
                memo: 'test invoice',
                expiry: 100,
            };
            const invoice = {
                invoice: 'tlnabc',
                paymentHash: '123',
                expiresAt: new Date(),
                status: 'open',
                walletId: wallet.wallet.id(),
                valueMsat: 1000n,
            };
            const createInvoiceNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`, abstract_lightning_1.CreateInvoiceBody.encode(createInvoice))
                .reply(200, abstract_lightning_1.Invoice.encode(invoice));
            const createInvoiceResponse = await wallet.createInvoice(createInvoice);
            assert.deepStrictEqual(createInvoiceResponse, invoice);
            createInvoiceNock.done();
        });
        it('createInvoice should throw error if wp response is invalid', async function () {
            const createInvoice = {
                valueMsat: 1000n,
                memo: 'test invoice',
                expiry: 100,
            };
            const createInvoiceNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`)
                .reply(200, { valueMsat: '1000' });
            await assert.rejects(async () => await wallet.createInvoice(createInvoice), /Invalid create invoice response/);
            createInvoiceNock.done();
        });
        it('should pay invoice', async function () {
            const params = {
                invoice: 'lnbc1...',
                amountMsat: 1000n,
                feeLimitMsat: 100n,
                feeLimitRatio: 0.1,
                sequenceId: '123',
                comment: 'test payment',
                passphrase: 'password123',
            };
            const txRequestResponse = {
                txRequestId: 'txReq123',
                state: 'delivered',
            };
            const lndResponse = {
                status: 'settled',
                paymentHash: 'paymentHash123',
                amountMsat: params.amountMsat !== undefined ? params.amountMsat.toString() : undefined,
                feeMsat: params.feeLimitMsat !== undefined ? params.feeLimitMsat.toString() : undefined,
                paymentPreimage: 'preimage123',
            };
            const finalPaymentResponse = {
                txRequestId: 'txReq123',
                state: 'delivered',
                transactions: [
                    {
                        unsignedTx: {
                            coinSpecific: {
                                ...lndResponse,
                            },
                        },
                    },
                ],
            };
            const transferData = {
                id: 'fake_id',
                coin: 'tlnbtc',
                state: 'confirmed',
                txid: lndResponse.paymentHash,
            };
            const getTransferNock = nock(bgUrl)
                .get(`/api/v2/${coinName}/wallet/${wallet.wallet.id()}/transfer/${lndResponse.paymentHash}`)
                .reply(200, transferData);
            const createTxRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)
                .reply(200, txRequestResponse);
            const sendTxRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests/${txRequestResponse.txRequestId}/transactions/0/send`)
                .reply(200, finalPaymentResponse);
            const userAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/def')
                .reply(200, userAuthKey);
            const nodeAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/ghi')
                .reply(200, nodeAuthKey);
            const response = await wallet.payInvoice(params);
            assert.strictEqual(response.txRequestId, 'txReq123');
            assert.strictEqual(response.txRequestState, 'delivered');
            assert(response.paymentStatus);
            assert.strictEqual(response.paymentStatus.status, finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.status);
            assert.strictEqual(response.paymentStatus.paymentHash, finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.paymentHash);
            assert.strictEqual(response.paymentStatus.amountMsat, finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.amountMsat);
            assert.strictEqual(response.paymentStatus.feeMsat, finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.feeMsat);
            assert.strictEqual(response.paymentStatus.paymentPreimage, finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.paymentPreimage);
            getTransferNock.done();
            createTxRequestNock.done();
            sendTxRequestNock.done();
            userAuthKeyNock.done();
            nodeAuthKeyNock.done();
        });
        it('should handle pending approval when paying invoice', async function () {
            const params = {
                invoice: 'lnbc1...',
                amountMsat: 1000n,
                feeLimitMsat: 100n,
                feeLimitRatio: 0.1,
                sequenceId: '123',
                comment: 'test payment',
                passphrase: 'password123',
            };
            const txRequestResponse = {
                txRequestId: 'txReq123',
                state: 'pendingApproval',
                pendingApprovalId: 'approval123',
            };
            const pendingApprovalData = {
                id: 'approval123',
                state: sdk_core_1.State.PENDING,
                creator: 'user123',
                info: {
                    type: sdk_core_1.Type.TRANSACTION_REQUEST,
                },
            };
            const createTxRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)
                .reply(200, txRequestResponse);
            const getPendingApprovalNock = nock(bgUrl)
                .get(`/api/v2/${coinName}/pendingapprovals/${txRequestResponse.pendingApprovalId}`)
                .reply(200, pendingApprovalData);
            const userAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/def')
                .reply(200, userAuthKey);
            const nodeAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/ghi')
                .reply(200, nodeAuthKey);
            const response = await wallet.payInvoice(params);
            assert.strictEqual(response.txRequestId, 'txReq123');
            assert.strictEqual(response.txRequestState, 'pendingApproval');
            assert(response.pendingApproval);
            assert.strictEqual(response.paymentStatus, undefined);
            createTxRequestNock.done();
            getPendingApprovalNock.done();
            userAuthKeyNock.done();
            nodeAuthKeyNock.done();
        });
    });
    describe('Get lightning key(s)', function () {
        const walletData = {
            id: 'fakeid',
            coin: coinName,
            keys: ['abc'],
            coinSpecific: { keys: ['def', 'ghi'] },
            subType: 'lightningCustody',
        };
        const userKeyData = {
            id: 'abc',
            pub: 'xpub1',
            encryptedPrv: 'encryptedPrv1',
            source: 'user',
        };
        const userAuthKeyData = {
            id: 'def',
            pub: 'xpub2',
            encryptedPrv: 'encryptedPrv2',
            source: 'user',
            coinSpecific: {
                tlnbtc: {
                    purpose: 'userAuth',
                },
            },
        };
        const nodeAuthKeyData = {
            id: 'ghi',
            pub: 'xpub3',
            encryptedPrv: 'encryptedPrv3',
            source: 'user',
            coinSpecific: {
                tlnbtc: {
                    purpose: 'nodeAuth',
                },
            },
        };
        it('should get lightning key', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, walletData);
            const keyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/abc')
                .reply(200, userKeyData);
            const key = await (0, abstract_lightning_1.getLightningKeychain)(wallet);
            assert.deepStrictEqual(key, userKeyData);
            keyNock.done();
        });
        it('should get lightning auth keys', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, walletData);
            const userAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/def')
                .reply(200, userAuthKeyData);
            const nodeAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/ghi')
                .reply(200, nodeAuthKeyData);
            const { userAuthKey, nodeAuthKey } = await (0, abstract_lightning_1.getLightningAuthKeychains)(wallet);
            assert.deepStrictEqual(userAuthKey, userAuthKeyData);
            assert.deepStrictEqual(nodeAuthKey, nodeAuthKeyData);
            userAuthKeyNock.done();
            nodeAuthKeyNock.done();
        });
        it('should fail to get lightning key for invalid number of keys', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, { ...walletData, keys: [] });
            await assert.rejects(async () => await (0, abstract_lightning_1.getLightningKeychain)(wallet), /Error: Invalid number of key in lightning wallet: 0/);
        });
        it('should fail to get lightning auth keys for invalid number of keys', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, { ...walletData, coinSpecific: { keys: ['def'] } });
            await assert.rejects(async () => await (0, abstract_lightning_1.getLightningAuthKeychains)(wallet), /Error: Invalid number of auth keys in lightning wallet: 1/);
        });
        it('should fail to get lightning key for invalid response', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, walletData);
            nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/abc')
                .reply(200, { ...userKeyData, source: 'backup' });
            await assert.rejects(async () => await (0, abstract_lightning_1.getLightningKeychain)(wallet), /Error: Invalid user key/);
        });
        it('should fail to get lightning auth keys for invalid response', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, walletData);
            nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/def')
                .reply(200, { ...userAuthKeyData, source: 'backup' });
            nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/ghi')
                .reply(200, nodeAuthKeyData);
            await assert.rejects(async () => await (0, abstract_lightning_1.getLightningAuthKeychains)(wallet), /Error: Invalid lightning auth key: def/);
        });
    });
    describe('Update lightning wallet coin specific', function () {
        const walletData = {
            id: 'fakeid',
            coin: coinName,
            keys: ['abc'],
            coinSpecific: { keys: ['def', 'ghi'] },
            subType: 'lightningSelfCustody',
        };
        const watchOnlyAccounts = {
            master_key_birthday_timestamp: 'dummy',
            master_key_fingerprint: 'dummy',
            accounts: [
                {
                    xpub: 'upub5Eep7H5q39PzQZLVEYLBytDyBNeV74E8mQsyeL6UozFq9Y3MsZ52G7YGuqrJPgoyAqF7TBeJdnkrHrVrB5pkWkPJ9cJGAePMU6F1Gyw6aFH',
                    purpose: 49,
                    coin_type: 0,
                    account: 0,
                },
                {
                    xpub: 'vpub5ZU1PHGpQoDSHckYico4nsvwsD3mTh6UjqL5zyGWXZXzBjTYMNKot7t9eRPQY71hJcnNN9r1ss25g3xA9rmoJ5nWPg8jEWavrttnsVa1qw1',
                    purpose: 84,
                    coin_type: 0,
                    account: 0,
                },
            ],
        };
        it('should update wallet', async function () {
            const wallet = new src_1.Wallet(bitgo, basecoin, walletData);
            const userAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/def')
                .reply(200, userAuthKey);
            const nodeAuthKeyNock = nock(bgUrl)
                .get('/api/v2/' + coinName + '/key/ghi')
                .reply(200, nodeAuthKey);
            let capturedBody;
            const wpWalletUpdateNock = nock(bgUrl)
                .put(`/api/v2/tlnbtc/wallet/${walletData.id}`)
                .reply(function (uri, requestBody) {
                capturedBody = requestBody;
                return [200];
            });
            const params = {
                signerMacaroon: 'signerMacaroon',
                signerAdminMacaroon: 'signerAdminMacaroon',
                signerTlsKey: 'signerTlsKey',
                signerTlsCert: 'signerTlsCert',
                watchOnlyAccounts,
                passphrase: 'password123',
            };
            await assert.doesNotReject(async () => await (0, abstract_lightning_1.updateWalletCoinSpecific)(wallet, params));
            assert(userAuthKeyNock.isDone());
            assert(nodeAuthKeyNock.isDone());
            assert(wpWalletUpdateNock.isDone());
            // Verify structure and required fields
            assert.ok(capturedBody.coinSpecific?.tlnbtc?.signedRequest, 'signedRequest should exist');
            const signedRequest = capturedBody.coinSpecific.tlnbtc.signedRequest;
            assert.ok(signedRequest.signerTlsCert, 'signerTlsCert should exist');
            assert.ok(signedRequest.watchOnlyAccounts, 'watchOnlyAccounts should exist');
            assert.ok(signedRequest.encryptedSignerTlsKey, 'encryptedSignerTlsKey should exist');
            assert.ok(signedRequest.encryptedSignerAdminMacaroon, 'encryptedSignerAdminMacaroon should exist');
            assert.ok(signedRequest.encryptedSignerMacaroon, 'encryptedSignerMacaroon should exist');
            // Verify signature exists
            assert.ok(capturedBody.coinSpecific.tlnbtc.signature, 'signature should exist');
            // we should not pass passphrase to the backend
            assert.strictEqual(signedRequest.passphrase, undefined, 'passphrase should not exist in request');
        });
    });
    describe('On chain withdrawal', function () {
        let wallet;
        beforeEach(function () {
            wallet = (0, abstract_lightning_1.getLightningWallet)(new src_1.Wallet(bitgo, basecoin, {
                id: 'walletId',
                coin: 'tlnbtc',
                subType: 'lightningCustody',
                coinSpecific: { keys: ['def', 'ghi'] },
            }));
        });
        it('should withdraw on chain', async function () {
            const params = {
                recipients: [
                    {
                        amountSat: 500000n,
                        address: 'bcrt1qjq48cqk2u80hewdcndf539m8nnnvt845nl68x7',
                    },
                ],
                satsPerVbyte: 15n,
            };
            const txRequestResponse = {
                txRequestId: 'txReq123',
                state: 'pendingDelivery',
            };
            const finalPaymentResponse = {
                txRequestId: 'txReq123',
                state: 'delivered',
            };
            const createTxRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)
                .reply(200, txRequestResponse);
            const sendTxRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests/${txRequestResponse.txRequestId}/transactions/0/send`)
                .reply(200, finalPaymentResponse);
            const response = await wallet.withdrawOnchain(params);
            assert.strictEqual(response.txRequestId, 'txReq123');
            assert.strictEqual(response.txRequestState, 'delivered');
            createTxRequestNock.done();
            sendTxRequestNock.done();
        });
        it('should handle pending approval when withdrawing onchain', async function () {
            const params = {
                recipients: [
                    {
                        amountSat: 500000n,
                        address: 'bcrt1qjq48cqk2u80hewdcndf539m8nnnvt845nl68x7',
                    },
                ],
                satsPerVbyte: 15n,
            };
            const txRequestResponse = {
                txRequestId: 'txReq123',
                state: 'pendingApproval',
                pendingApprovalId: 'approval123',
            };
            const pendingApprovalData = {
                id: 'approval123',
                state: sdk_core_1.State.PENDING,
                creator: 'user123',
                info: {
                    type: sdk_core_1.Type.TRANSACTION_REQUEST,
                },
            };
            const createTxRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)
                .reply(200, txRequestResponse);
            const getPendingApprovalNock = nock(bgUrl)
                .get(`/api/v2/${coinName}/pendingapprovals/${txRequestResponse.pendingApprovalId}`)
                .reply(200, pendingApprovalData);
            const response = await wallet.withdrawOnchain(params);
            assert.strictEqual(response.txRequestId, 'txReq123');
            assert.strictEqual(response.txRequestState, 'pendingApproval');
            assert(response.pendingApproval);
            createTxRequestNock.done();
            getPendingApprovalNock.done();
        });
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lightningWallets.js","sourceRoot":"","sources":["../../../../../test/v2/unit/lightning/lightningWallets.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport { TestBitGo } from '@bitgo/sdk-test';\nimport * as nock from 'nock';\nimport { BaseCoin, PendingApprovalData, State, Type } from '@bitgo/sdk-core';\nimport {\n  CreateInvoiceBody,\n  getLightningWallet,\n  Invoice,\n  InvoiceInfo,\n  InvoiceQuery,\n  LndCreatePaymentResponse,\n  LightningWallet,\n  SubmitPaymentParams,\n  UpdateLightningWalletClientRequest,\n  getLightningKeychain,\n  getLightningAuthKeychains,\n  updateWalletCoinSpecific,\n  LightningOnchainWithdrawParams,\n} from '@bitgo/abstract-lightning';\n\nimport { BitGo, common, GenerateLightningWalletOptions, Wallet, Wallets } from '../../../../src';\n\ndescribe('Lightning wallets', function () {\n  const coinName = 'tlnbtc';\n  const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n  let basecoin: BaseCoin;\n  let wallets: Wallets;\n  let bgUrl: string;\n\n  const userAuthKey = {\n    id: 'def',\n    pub: 'xpub661MyMwAqRbcGYjYsnsDj1SHdiXynWEXNnfNgMSpokN54FKyMqbu7rWEfVNDs6uAJmz86UVFtq4sefhQpXZhSAzQcL9zrEPtiLNNZoeSxCG',\n    encryptedPrv:\n      '{\"iv\":\"zYhhaNdW0wPfJEoBjZ4pvg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"tgAMua9jjhw=\",\"ct\":\"HcrbxQvNlWG5tLMndYzdNCYa1l+1h7o+vSsweA0+q1le3tWt6jLUJSEjZN+JI8lTZ2KPFQgLulQQhsUa+ytUCBi0vSgjF7x7CprT7l2Cfjkew00XsEd7wnmtJUsrQk8m69Co7tIRA3oEgzrnYwy4qOM81lbNNyQ=\"}',\n    source: 'user',\n    coinSpecific: {\n      tlnbtc: {\n        purpose: 'userAuth',\n      },\n    },\n  };\n\n  const nodeAuthKey = {\n    id: 'ghi',\n    pub: 'xpub661MyMwAqRbcG9xnTnAnRbJPo3MAHyRtH4zeehN8exYk4VFz5buepUzebhix33BKhS5Eb4V3LEfW5pYiSR8qmaEnyrpeghhKY8JfzAsUDpq',\n    encryptedPrv:\n      '{\"iv\":\"bH6eGbnl9x8PZECPrgvcng==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"o8yknV6nTI8=\",\"ct\":\"nGyzAToIzYkQeIdcVafoWHtMx7+Fgj0YldCme3WA1yxJAA0QulZVhblMZN/7efCRIumA0NNmpH7dxH6n8cVlz/Z+RUgC2q9lgvZKUoJcYNTjWUfkmkJutXX2tr8yVxm+eC/hnRiyfVLZ2qPxctvDlBVBfgLuPyc=\"}',\n    source: 'user',\n    coinSpecific: {\n      tlnbtc: {\n        purpose: 'nodeAuth',\n      },\n    },\n  };\n\n  before(function () {\n    bitgo.initializeTestVars();\n\n    basecoin = bitgo.coin(coinName);\n    wallets = basecoin.wallets();\n    bgUrl = common.Environments[bitgo.getEnv()].uri;\n  });\n\n  after(function () {\n    nock.cleanAll();\n    nock.pendingMocks().length.should.equal(0);\n  });\n\n  describe('Generate lightning wallet', function () {\n    it('should validate parameters', async function () {\n      await wallets\n        .generateWallet({\n          passphrase: 'pass123',\n          enterprise: 'ent123',\n          passcodeEncryptionCode: 'code123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.label, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 'my ln wallet',\n          enterprise: 'ent123',\n          passcodeEncryptionCode: 'code123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.passphrase, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 'my ln wallet',\n          passphrase: 'pass123',\n          passcodeEncryptionCode: 'code123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.enterprise, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 'my ln wallet',\n          passphrase: 'pass123',\n          enterprise: 'ent123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value 'undefined' supplied to GenerateLightningWalletOptions.passcodeEncryptionCode, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 123 as any,\n          passphrase: 'pass123',\n          enterprise: 'ent123',\n          passcodeEncryptionCode: 'code123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.label, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 'my ln wallet',\n          passphrase: 123 as any,\n          enterprise: 'ent123',\n          passcodeEncryptionCode: 'code123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.passphrase, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 'my ln wallet',\n          passphrase: 'pass123',\n          enterprise: 123 as any,\n          passcodeEncryptionCode: 'code123',\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.enterprise, expected string.\"\n        );\n\n      await wallets\n        .generateWallet({\n          label: 'my ln wallet',\n          passphrase: 'pass123',\n          enterprise: 'ent123',\n          passcodeEncryptionCode: 123 as any,\n          subType: 'lightningCustody',\n        })\n        .should.be.rejectedWith(\n          \"error(s) parsing generate lightning wallet request params: Invalid value '123' supplied to GenerateLightningWalletOptions.passcodeEncryptionCode, expected string.\"\n        );\n    });\n\n    for (const subType of ['lightningCustody', 'lightningSelfCustody'] as const) {\n      it(`should generate ${subType} lightning wallet`, async function () {\n        const params: GenerateLightningWalletOptions = {\n          label: 'my ln wallet',\n          passphrase: 'pass123',\n          enterprise: 'ent123',\n          passcodeEncryptionCode: 'code123',\n          subType: subType as 'lightningCustody' | 'lightningSelfCustody',\n        };\n\n        const validateKeyRequest = (body) => {\n          const baseChecks =\n            body.pub.startsWith('xpub') &&\n            !!body.encryptedPrv &&\n            body.keyType === 'independent' &&\n            body.source === 'user';\n\n          if (body.originalPasscodeEncryptionCode !== undefined) {\n            return baseChecks && body.originalPasscodeEncryptionCode === 'code123' && body.coinSpecific === undefined;\n          } else {\n            const coinSpecific = body.coinSpecific && body.coinSpecific.tlnbtc;\n            return baseChecks && !!coinSpecific && ['userAuth', 'nodeAuth'].includes(coinSpecific.purpose);\n          }\n        };\n\n        const validateWalletRequest = (body) => {\n          return (\n            body.label === 'my ln wallet' &&\n            body.m === 1 &&\n            body.n === 1 &&\n            body.type === 'hot' &&\n            body.subType === subType &&\n            body.enterprise === 'ent123' &&\n            Array.isArray(body.keys) &&\n            body.keys.length === 1 &&\n            body.keys[0] === 'keyId1' &&\n            body.coinSpecific &&\n            body.coinSpecific.tlnbtc &&\n            Array.isArray(body.coinSpecific.tlnbtc.keys) &&\n            body.coinSpecific.tlnbtc.keys.length === 2 &&\n            body.coinSpecific.tlnbtc.keys.includes('keyId2') &&\n            body.coinSpecific.tlnbtc.keys.includes('keyId3')\n          );\n        };\n\n        nock(bgUrl)\n          .post('/api/v2/' + coinName + '/key', (body) => validateKeyRequest(body))\n          .reply(200, { id: 'keyId1' });\n        nock(bgUrl)\n          .post('/api/v2/' + coinName + '/key', (body) => validateKeyRequest(body))\n          .reply(200, { id: 'keyId2' });\n        nock(bgUrl)\n          .post('/api/v2/' + coinName + '/key', (body) => validateKeyRequest(body))\n          .reply(200, { id: 'keyId3' });\n\n        nock(bgUrl)\n          .post('/api/v2/' + coinName + '/wallet/add', (body) => validateWalletRequest(body))\n          .reply(200, { id: 'walletId' });\n\n        const response = await wallets.generateWallet(params);\n\n        assert.ok(response.wallet);\n        assert.ok(response.encryptedWalletPassphrase);\n        assert.equal(\n          bitgo.decrypt({ input: response.encryptedWalletPassphrase, password: params.passcodeEncryptionCode }),\n          params.passphrase\n        );\n      });\n    }\n  });\n\n  describe('invoices', function () {\n    let wallet: LightningWallet;\n    beforeEach(function () {\n      wallet = getLightningWallet(\n        new Wallet(bitgo, basecoin, {\n          id: 'walletId',\n          coin: 'tlnbtc',\n          subType: 'lightningCustody',\n          coinSpecific: { keys: ['def', 'ghi'] },\n        })\n      ) as LightningWallet;\n    });\n\n    it('should list invoices', async function () {\n      const invoice: InvoiceInfo = {\n        valueMsat: 1000n,\n        paymentHash: 'foo',\n        invoice: 'tlnfoobar',\n        walletId: wallet.wallet.id(),\n        status: 'open',\n        expiresAt: new Date(),\n        createdAt: new Date(),\n        updatedAt: new Date(),\n      };\n      const query = {\n        status: 'open',\n        startDate: new Date(),\n        limit: 100n,\n      } as InvoiceQuery;\n      const listInvoicesNock = nock(bgUrl)\n        .get(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`)\n        .query(InvoiceQuery.encode(query))\n        .reply(200, [InvoiceInfo.encode(invoice)]);\n      const invoiceResponse = await wallet.listInvoices(query);\n      assert.strictEqual(invoiceResponse.length, 1);\n      assert.deepStrictEqual(invoiceResponse[0], invoice);\n      listInvoicesNock.done();\n    });\n\n    it('listInvoices should throw error if wp response is invalid', async function () {\n      const listInvoicesNock = nock(bgUrl)\n        .get(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`)\n        .reply(200, [{ valueMsat: '1000' }]);\n      await assert.rejects(async () => await wallet.listInvoices({}), /Invalid list invoices response/);\n      listInvoicesNock.done();\n    });\n\n    it('should create invoice', async function () {\n      const createInvoice: CreateInvoiceBody = {\n        valueMsat: 1000n,\n        memo: 'test invoice',\n        expiry: 100,\n      };\n      const invoice: Invoice = {\n        invoice: 'tlnabc',\n        paymentHash: '123',\n        expiresAt: new Date(),\n        status: 'open',\n        walletId: wallet.wallet.id(),\n        valueMsat: 1000n,\n      };\n      const createInvoiceNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`, CreateInvoiceBody.encode(createInvoice))\n        .reply(200, Invoice.encode(invoice));\n      const createInvoiceResponse = await wallet.createInvoice(createInvoice);\n      assert.deepStrictEqual(createInvoiceResponse, invoice);\n      createInvoiceNock.done();\n    });\n\n    it('createInvoice should throw error if wp response is invalid', async function () {\n      const createInvoice: CreateInvoiceBody = {\n        valueMsat: 1000n,\n        memo: 'test invoice',\n        expiry: 100,\n      };\n      const createInvoiceNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/lightning/invoice`)\n        .reply(200, { valueMsat: '1000' });\n      await assert.rejects(async () => await wallet.createInvoice(createInvoice), /Invalid create invoice response/);\n      createInvoiceNock.done();\n    });\n\n    it('should pay invoice', async function () {\n      const params: SubmitPaymentParams = {\n        invoice: 'lnbc1...',\n        amountMsat: 1000n,\n        feeLimitMsat: 100n,\n        feeLimitRatio: 0.1,\n        sequenceId: '123',\n        comment: 'test payment',\n        passphrase: 'password123',\n      };\n\n      const txRequestResponse = {\n        txRequestId: 'txReq123',\n        state: 'delivered',\n      };\n\n      const lndResponse: LndCreatePaymentResponse = {\n        status: 'settled',\n        paymentHash: 'paymentHash123',\n        amountMsat: params.amountMsat !== undefined ? params.amountMsat.toString() : undefined,\n        feeMsat: params.feeLimitMsat !== undefined ? params.feeLimitMsat.toString() : undefined,\n        paymentPreimage: 'preimage123',\n      };\n\n      const finalPaymentResponse = {\n        txRequestId: 'txReq123',\n        state: 'delivered',\n        transactions: [\n          {\n            unsignedTx: {\n              coinSpecific: {\n                ...lndResponse,\n              },\n            },\n          },\n        ],\n      };\n\n      const transferData = {\n        id: 'fake_id',\n        coin: 'tlnbtc',\n        state: 'confirmed',\n        txid: lndResponse.paymentHash,\n      };\n\n      const getTransferNock = nock(bgUrl)\n        .get(`/api/v2/${coinName}/wallet/${wallet.wallet.id()}/transfer/${lndResponse.paymentHash}`)\n        .reply(200, transferData);\n\n      const createTxRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)\n        .reply(200, txRequestResponse);\n\n      const sendTxRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests/${txRequestResponse.txRequestId}/transactions/0/send`)\n        .reply(200, finalPaymentResponse);\n\n      const userAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/def')\n        .reply(200, userAuthKey);\n      const nodeAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/ghi')\n        .reply(200, nodeAuthKey);\n\n      const response = await wallet.payInvoice(params);\n      assert.strictEqual(response.txRequestId, 'txReq123');\n      assert.strictEqual(response.txRequestState, 'delivered');\n      assert(response.paymentStatus);\n      assert.strictEqual(\n        response.paymentStatus.status,\n        finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.status\n      );\n      assert.strictEqual(\n        response.paymentStatus.paymentHash,\n        finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.paymentHash\n      );\n      assert.strictEqual(\n        response.paymentStatus.amountMsat,\n        finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.amountMsat\n      );\n      assert.strictEqual(\n        response.paymentStatus.feeMsat,\n        finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.feeMsat\n      );\n      assert.strictEqual(\n        response.paymentStatus.paymentPreimage,\n        finalPaymentResponse.transactions[0].unsignedTx.coinSpecific.paymentPreimage\n      );\n\n      getTransferNock.done();\n      createTxRequestNock.done();\n      sendTxRequestNock.done();\n      userAuthKeyNock.done();\n      nodeAuthKeyNock.done();\n    });\n\n    it('should handle pending approval when paying invoice', async function () {\n      const params: SubmitPaymentParams = {\n        invoice: 'lnbc1...',\n        amountMsat: 1000n,\n        feeLimitMsat: 100n,\n        feeLimitRatio: 0.1,\n        sequenceId: '123',\n        comment: 'test payment',\n        passphrase: 'password123',\n      };\n\n      const txRequestResponse = {\n        txRequestId: 'txReq123',\n        state: 'pendingApproval',\n        pendingApprovalId: 'approval123',\n      };\n\n      const pendingApprovalData: PendingApprovalData = {\n        id: 'approval123',\n        state: State.PENDING,\n        creator: 'user123',\n        info: {\n          type: Type.TRANSACTION_REQUEST,\n        },\n      };\n\n      const createTxRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)\n        .reply(200, txRequestResponse);\n\n      const getPendingApprovalNock = nock(bgUrl)\n        .get(`/api/v2/${coinName}/pendingapprovals/${txRequestResponse.pendingApprovalId}`)\n        .reply(200, pendingApprovalData);\n\n      const userAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/def')\n        .reply(200, userAuthKey);\n      const nodeAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/ghi')\n        .reply(200, nodeAuthKey);\n\n      const response = await wallet.payInvoice(params);\n      assert.strictEqual(response.txRequestId, 'txReq123');\n      assert.strictEqual(response.txRequestState, 'pendingApproval');\n      assert(response.pendingApproval);\n      assert.strictEqual(response.paymentStatus, undefined);\n\n      createTxRequestNock.done();\n      getPendingApprovalNock.done();\n      userAuthKeyNock.done();\n      nodeAuthKeyNock.done();\n    });\n  });\n\n  describe('Get lightning key(s)', function () {\n    const walletData = {\n      id: 'fakeid',\n      coin: coinName,\n      keys: ['abc'],\n      coinSpecific: { keys: ['def', 'ghi'] },\n      subType: 'lightningCustody',\n    };\n\n    const userKeyData = {\n      id: 'abc',\n      pub: 'xpub1',\n      encryptedPrv: 'encryptedPrv1',\n      source: 'user',\n    };\n\n    const userAuthKeyData = {\n      id: 'def',\n      pub: 'xpub2',\n      encryptedPrv: 'encryptedPrv2',\n      source: 'user',\n      coinSpecific: {\n        tlnbtc: {\n          purpose: 'userAuth',\n        },\n      },\n    };\n\n    const nodeAuthKeyData = {\n      id: 'ghi',\n      pub: 'xpub3',\n      encryptedPrv: 'encryptedPrv3',\n      source: 'user',\n      coinSpecific: {\n        tlnbtc: {\n          purpose: 'nodeAuth',\n        },\n      },\n    };\n\n    it('should get lightning key', async function () {\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      const keyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/abc')\n        .reply(200, userKeyData);\n\n      const key = await getLightningKeychain(wallet);\n      assert.deepStrictEqual(key, userKeyData);\n      keyNock.done();\n    });\n\n    it('should get lightning auth keys', async function () {\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      const userAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/def')\n        .reply(200, userAuthKeyData);\n      const nodeAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/ghi')\n        .reply(200, nodeAuthKeyData);\n\n      const { userAuthKey, nodeAuthKey } = await getLightningAuthKeychains(wallet);\n      assert.deepStrictEqual(userAuthKey, userAuthKeyData);\n      assert.deepStrictEqual(nodeAuthKey, nodeAuthKeyData);\n      userAuthKeyNock.done();\n      nodeAuthKeyNock.done();\n    });\n\n    it('should fail to get lightning key for invalid number of keys', async function () {\n      const wallet = new Wallet(bitgo, basecoin, { ...walletData, keys: [] });\n      await assert.rejects(\n        async () => await getLightningKeychain(wallet),\n        /Error: Invalid number of key in lightning wallet: 0/\n      );\n    });\n\n    it('should fail to get lightning auth keys for invalid number of keys', async function () {\n      const wallet = new Wallet(bitgo, basecoin, { ...walletData, coinSpecific: { keys: ['def'] } });\n      await assert.rejects(\n        async () => await getLightningAuthKeychains(wallet),\n        /Error: Invalid number of auth keys in lightning wallet: 1/\n      );\n    });\n\n    it('should fail to get lightning key for invalid response', async function () {\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/abc')\n        .reply(200, { ...userKeyData, source: 'backup' });\n\n      await assert.rejects(async () => await getLightningKeychain(wallet), /Error: Invalid user key/);\n    });\n\n    it('should fail to get lightning auth keys for invalid response', async function () {\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/def')\n        .reply(200, { ...userAuthKeyData, source: 'backup' });\n\n      nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/ghi')\n        .reply(200, nodeAuthKeyData);\n\n      await assert.rejects(\n        async () => await getLightningAuthKeychains(wallet),\n        /Error: Invalid lightning auth key: def/\n      );\n    });\n  });\n\n  describe('Update lightning wallet coin specific', function () {\n    const walletData = {\n      id: 'fakeid',\n      coin: coinName,\n      keys: ['abc'],\n      coinSpecific: { keys: ['def', 'ghi'] },\n      subType: 'lightningSelfCustody',\n    };\n\n    const watchOnlyAccounts = {\n      master_key_birthday_timestamp: 'dummy',\n      master_key_fingerprint: 'dummy',\n      accounts: [\n        {\n          xpub: 'upub5Eep7H5q39PzQZLVEYLBytDyBNeV74E8mQsyeL6UozFq9Y3MsZ52G7YGuqrJPgoyAqF7TBeJdnkrHrVrB5pkWkPJ9cJGAePMU6F1Gyw6aFH',\n          purpose: 49,\n          coin_type: 0,\n          account: 0,\n        },\n        {\n          xpub: 'vpub5ZU1PHGpQoDSHckYico4nsvwsD3mTh6UjqL5zyGWXZXzBjTYMNKot7t9eRPQY71hJcnNN9r1ss25g3xA9rmoJ5nWPg8jEWavrttnsVa1qw1',\n          purpose: 84,\n          coin_type: 0,\n          account: 0,\n        },\n      ],\n    };\n    it('should update wallet', async function () {\n      const wallet = new Wallet(bitgo, basecoin, walletData);\n\n      const userAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/def')\n        .reply(200, userAuthKey);\n      const nodeAuthKeyNock = nock(bgUrl)\n        .get('/api/v2/' + coinName + '/key/ghi')\n        .reply(200, nodeAuthKey);\n      let capturedBody;\n      const wpWalletUpdateNock = nock(bgUrl)\n        .put(`/api/v2/tlnbtc/wallet/${walletData.id}`)\n        .reply(function (uri, requestBody) {\n          capturedBody = requestBody;\n          return [200];\n        });\n\n      const params: UpdateLightningWalletClientRequest = {\n        signerMacaroon: 'signerMacaroon',\n        signerAdminMacaroon: 'signerAdminMacaroon',\n        signerTlsKey: 'signerTlsKey',\n        signerTlsCert: 'signerTlsCert',\n        watchOnlyAccounts,\n        passphrase: 'password123',\n      };\n\n      await assert.doesNotReject(async () => await updateWalletCoinSpecific(wallet, params));\n      assert(userAuthKeyNock.isDone());\n      assert(nodeAuthKeyNock.isDone());\n      assert(wpWalletUpdateNock.isDone());\n\n      // Verify structure and required fields\n      assert.ok(capturedBody.coinSpecific?.tlnbtc?.signedRequest, 'signedRequest should exist');\n      const signedRequest = capturedBody.coinSpecific.tlnbtc.signedRequest;\n\n      assert.ok(signedRequest.signerTlsCert, 'signerTlsCert should exist');\n      assert.ok(signedRequest.watchOnlyAccounts, 'watchOnlyAccounts should exist');\n      assert.ok(signedRequest.encryptedSignerTlsKey, 'encryptedSignerTlsKey should exist');\n      assert.ok(signedRequest.encryptedSignerAdminMacaroon, 'encryptedSignerAdminMacaroon should exist');\n      assert.ok(signedRequest.encryptedSignerMacaroon, 'encryptedSignerMacaroon should exist');\n\n      // Verify signature exists\n      assert.ok(capturedBody.coinSpecific.tlnbtc.signature, 'signature should exist');\n\n      // we should not pass passphrase to the backend\n      assert.strictEqual(signedRequest.passphrase, undefined, 'passphrase should not exist in request');\n    });\n  });\n  describe('On chain withdrawal', function () {\n    let wallet: LightningWallet;\n    beforeEach(function () {\n      wallet = getLightningWallet(\n        new Wallet(bitgo, basecoin, {\n          id: 'walletId',\n          coin: 'tlnbtc',\n          subType: 'lightningCustody',\n          coinSpecific: { keys: ['def', 'ghi'] },\n        })\n      ) as LightningWallet;\n    });\n    it('should withdraw on chain', async function () {\n      const params: LightningOnchainWithdrawParams = {\n        recipients: [\n          {\n            amountSat: 500000n,\n            address: 'bcrt1qjq48cqk2u80hewdcndf539m8nnnvt845nl68x7',\n          },\n        ],\n        satsPerVbyte: 15n,\n      };\n\n      const txRequestResponse = {\n        txRequestId: 'txReq123',\n        state: 'pendingDelivery',\n      };\n\n      const finalPaymentResponse = {\n        txRequestId: 'txReq123',\n        state: 'delivered',\n      };\n\n      const createTxRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)\n        .reply(200, txRequestResponse);\n\n      const sendTxRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests/${txRequestResponse.txRequestId}/transactions/0/send`)\n        .reply(200, finalPaymentResponse);\n\n      const response = await wallet.withdrawOnchain(params);\n      assert.strictEqual(response.txRequestId, 'txReq123');\n      assert.strictEqual(response.txRequestState, 'delivered');\n\n      createTxRequestNock.done();\n      sendTxRequestNock.done();\n    });\n\n    it('should handle pending approval when withdrawing onchain', async function () {\n      const params: LightningOnchainWithdrawParams = {\n        recipients: [\n          {\n            amountSat: 500000n,\n            address: 'bcrt1qjq48cqk2u80hewdcndf539m8nnnvt845nl68x7',\n          },\n        ],\n        satsPerVbyte: 15n,\n      };\n\n      const txRequestResponse = {\n        txRequestId: 'txReq123',\n        state: 'pendingApproval',\n        pendingApprovalId: 'approval123',\n      };\n\n      const pendingApprovalData: PendingApprovalData = {\n        id: 'approval123',\n        state: State.PENDING,\n        creator: 'user123',\n        info: {\n          type: Type.TRANSACTION_REQUEST,\n        },\n      };\n\n      const createTxRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${wallet.wallet.id()}/txrequests`)\n        .reply(200, txRequestResponse);\n\n      const getPendingApprovalNock = nock(bgUrl)\n        .get(`/api/v2/${coinName}/pendingapprovals/${txRequestResponse.pendingApprovalId}`)\n        .reply(200, pendingApprovalData);\n\n      const response = await wallet.withdrawOnchain(params);\n      assert.strictEqual(response.txRequestId, 'txReq123');\n      assert.strictEqual(response.txRequestState, 'pendingApproval');\n      assert(response.pendingApproval);\n\n      createTxRequestNock.done();\n      getPendingApprovalNock.done();\n    });\n  });\n});\n"]}

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


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