diff --git a/.github/workflows/mocha-anrdoid.sh b/.github/workflows/mocha-anrdoid.sh index 479ff2a6..33539132 100755 --- a/.github/workflows/mocha-anrdoid.sh +++ b/.github/workflows/mocha-anrdoid.sh @@ -1,5 +1,6 @@ #!/bin/bash +adb reverse tcp:3003 tcp:3003 adb reverse tcp:8090 tcp:8090 adb reverse tcp:9090 tcp:9090 adb reverse tcp:9091 tcp:9091 diff --git a/.github/workflows/mocha-ios.yml b/.github/workflows/mocha-ios.yml index d93c9459..65d05aee 100644 --- a/.github/workflows/mocha-ios.yml +++ b/.github/workflows/mocha-ios.yml @@ -41,6 +41,10 @@ jobs: uses: docker-practice/actions-setup-docker@1.0.12 timeout-minutes: 30 + - name: Install backup-server dependencies + working-directory: backup-server + run: npm i || npm i + - name: Run regtest setup working-directory: example/docker run: | diff --git a/backup-server/Dockerfile b/backup-server/Dockerfile index 957c2112..176a6a3e 100644 --- a/backup-server/Dockerfile +++ b/backup-server/Dockerfile @@ -1,4 +1,4 @@ -FROM node:20 +FROM node:20-alpine WORKDIR /app diff --git a/backup-server/package-lock.json b/backup-server/package-lock.json index f3f3908b..3a2a5eaf 100644 --- a/backup-server/package-lock.json +++ b/backup-server/package-lock.json @@ -1,12 +1,12 @@ { "name": "backup-server", - "version": "1.0.0", + "version": "0.0.129", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "backup-server", - "version": "1.0.0", + "version": "0.0.129", "license": "ISC", "dependencies": { "@tweedegolf/storage-abstraction": "^1.4.6", @@ -16,6 +16,10 @@ }, "devDependencies": { "ln-service": "^56.13.0" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" } }, "node_modules/@aws-crypto/crc32": { diff --git a/backup-server/package.json b/backup-server/package.json index fcebf4d3..615efd30 100644 --- a/backup-server/package.json +++ b/backup-server/package.json @@ -4,8 +4,8 @@ "description": "", "main": "index.js", "engines": { - "npm": ">=8.0.0 <9.0.0", - "node": ">=19.0.0" + "npm": ">=8.0.0", + "node": ">=18.0.0" }, "engineStrict" : true, "scripts": { diff --git a/example/Tests.tsx b/example/Tests.tsx index b9f5a901..3fd88fad 100644 --- a/example/Tests.tsx +++ b/example/Tests.tsx @@ -124,7 +124,7 @@ class Tests extends Component { // Adding an async hook before each test to allow the UI to update beforeEach(function () { // WFT it doesn't work ? - return new Promise((resolve) => setTimeout(resolve, 0)); + return new Promise((resolve) => setTimeout(resolve, 10)); }); // global.fs = require("react-native-fs"); // global.path = require("path-browserify"); diff --git a/example/docker/docker-compose.yml b/example/docker/docker-compose.yml index 398bdc31..4efc0696 100644 --- a/example/docker/docker-compose.yml +++ b/example/docker/docker-compose.yml @@ -193,7 +193,8 @@ services: ldk-backup-server: container_name: ldk-backup-server - image: synonymsoft/ldk-backup-server:1.0.0 + # image: synonymsoft/ldk-backup-server:1.0.0 + build: ../../backup-server/ expose: - '3003' ports: diff --git a/example/e2e/ldk.test.js b/example/e2e/ldk.test.js index 48f32c32..82ec8552 100644 --- a/example/e2e/ldk.test.js +++ b/example/e2e/ldk.test.js @@ -8,22 +8,19 @@ describe('LDK integration test', () => { await element(by.id('dev')).tap(); }); - it('should have heading', async () => { + it('should show "Running LDK" after starting up', async () => { await waitFor(element(by.text('react-native-ldk'))) .toBeVisible() - .withTimeout(20000); - }); + .withTimeout(60000); - it('should show "Running LDK" after starting up', async () => { - // await element(by.id('start')).tap(); await waitFor(element(by.text('Running LDK'))) .toBeVisible() - .withTimeout(20000); + .withTimeout(60000); await element(by.id('E2ETest')).tap(); await waitFor(element(by.text('e2e success'))) .toBeVisible() - .withTimeout(20000); + .withTimeout(60000); }); }); diff --git a/example/ldk/index.ts b/example/ldk/index.ts index f31f73dd..5db14d4a 100644 --- a/example/ldk/index.ts +++ b/example/ldk/index.ts @@ -109,13 +109,15 @@ export const setupLdk = async ( return err(storageRes.error); } - const res = await ldk.backupSetup({ - network: ldkNetwork(selectedNetwork), - seed: account.seed, - details: backupServerDetails!, - }); - if (res.isErr()) { - return err(res.error); + if (backupServerDetails) { + const res = await ldk.backupSetup({ + network: ldkNetwork(selectedNetwork), + seed: account.seed, + details: backupServerDetails, + }); + if (res.isErr()) { + return err(res.error); + } } const lmStart = await lm.start({ diff --git a/example/package.json b/example/package.json index f92a4d33..6476c025 100644 --- a/example/package.json +++ b/example/package.json @@ -72,7 +72,7 @@ "bitcoin-json-rpc": "^1.3.2", "chai-as-promised": "^7.1.1", "concurrently": "^8.2.0", - "detox": "20.7.0", + "detox": "20.17.0", "electrum-client": "github:BlueWallet/rn-electrum-client#47acb51149e97fab249c3f8a314f708dbee4fb6e", "eslint": "8.27.0", "eslint-config-prettier": "^8.5.0", diff --git a/example/tests/clightning.ts b/example/tests/clightning.ts index e9b83cd2..b0c1200b 100644 --- a/example/tests/clightning.ts +++ b/example/tests/clightning.ts @@ -1,12 +1,14 @@ -import { describe, it } from 'mocha'; +import lm, { ENetworks, ldk } from '@synonymdev/react-native-ldk'; import BitcoinJsonRpc from 'bitcoin-json-rpc'; +import { describe, it } from 'mocha'; import RNFS from 'react-native-fs'; -import lm, { ldk } from '@synonymdev/react-native-ldk'; import { CL, TestProfile, + backupServerDetails, initWaitForElectrumToSync, + skipRemoteBackups, sleep, wipeLdkStorage, } from './utils'; @@ -32,7 +34,7 @@ const clConfig = { macaroon: global.environment.clmacaroon, }; -describe('Clightning', function () { +describe.skip('Clightning', function () { this.retries(3); this.timeout(5 * 60 * 1000); // 5 minutes let waitForElectrum: any; @@ -77,15 +79,7 @@ describe('Clightning', function () { beforeEach(async () => { await waitForElectrum({ cl: true }); - // const electrumResponse = await connectToElectrum({}); - // if (electrumResponse.isErr()) { - // throw electrumResponse.error; - // } - // const account = await getAccount(); - profile = new TestProfile({ - // name: account.name, - // seed: account.seed, headerCallback: async (): Promise => { const syncRes = await lm.syncLdk(); if (syncRes.isErr()) { @@ -112,8 +106,6 @@ describe('Clightning', function () { // - open CL -> LDK channel // - make a few payments - // const account = await getAccount(); - await ldk.stop(); const lmStart = await lm.start({ ...profile.getStartParams(), @@ -132,6 +124,17 @@ describe('Clightning', function () { throw lmStart.error; } + if (!skipRemoteBackups) { + const backupRes = await ldk.backupSetup({ + network: ENetworks.regtest, + seed: profile.getAccount().seed, + details: backupServerDetails, + }); + if (backupRes.isErr()) { + throw backupRes.error; + } + } + const nodeId = await ldk.nodeId(); if (nodeId.isErr()) { throw nodeId.error; diff --git a/example/tests/eclair.ts b/example/tests/eclair.ts index 109bdd55..ab356c15 100644 --- a/example/tests/eclair.ts +++ b/example/tests/eclair.ts @@ -1,30 +1,18 @@ -import { describe, it } from 'mocha'; +import lm, { ENetworks, ldk } from '@synonymdev/react-native-ldk'; import BitcoinJsonRpc from 'bitcoin-json-rpc'; +import { describe, it } from 'mocha'; import RNFS from 'react-native-fs'; -import lm, { ldk } from '@synonymdev/react-native-ldk'; import { Eclair, TestProfile, + backupServerDetails, initWaitForElectrumToSync, + skipRemoteBackups, sleep, wipeLdkStorage, } from './utils'; -// import { -// broadcastTransaction, -// getBestBlock, -// getTransactionData, -// getTransactionPosition, -// } from '../ldk'; -// import { -// connectToElectrum, -// getScriptPubKeyHistory, -// subscribeToHeader, -// } from '../electrum';r -// import { getAccount, getAddress, ldkNetwork } from '../utils/helpers'; -// import { selectedNetwork } from '../utils/constants'; - const bitcoinURL = 'http://polaruser:polarpass@localhost:9091'; const eclairConfig = { host: 'localhost', @@ -112,21 +100,9 @@ describe('Eclair', function () { // - open Eclair -> LDK channel // - make a few payments - // const account = await getAccount(); - await ldk.stop(); const lmStart = await lm.start({ ...profile.getStartParams(), - // getBestBlock: getBestBlock, - // account: account, - // getAddress: getAddress, - // getScriptPubKeyHistory: getScriptPubKeyHistory, - // getFees: () => - // Promise.resolve({ highPriority: 10, normal: 5, background: 1 }), - // getTransactionData: getTransactionData, - // getTransactionPosition: getTransactionPosition, - // broadcastTransaction: broadcastTransaction, - // network: ldkNetwork(selectedNetwork), getFees: () => { return Promise.resolve({ onChainSweep: 30, @@ -143,6 +119,17 @@ describe('Eclair', function () { throw lmStart.error; } + if (!skipRemoteBackups) { + const backupRes = await ldk.backupSetup({ + network: ENetworks.regtest, + seed: profile.getAccount().seed, + details: backupServerDetails, + }); + if (backupRes.isErr()) { + throw backupRes.error; + } + } + const nodeId = await ldk.nodeId(); if (nodeId.isErr()) { throw nodeId.error; diff --git a/example/tests/lnd.ts b/example/tests/lnd.ts index d0d3ede1..30a8b891 100644 --- a/example/tests/lnd.ts +++ b/example/tests/lnd.ts @@ -2,7 +2,12 @@ import { expect } from 'chai'; import { describe, it } from 'mocha'; import BitcoinJsonRpc from 'bitcoin-json-rpc'; import RNFS from 'react-native-fs'; -import lm, { EEventTypes, TChannel, ldk } from '@synonymdev/react-native-ldk'; +import lm, { + EEventTypes, + ENetworks, + TChannel, + ldk, +} from '@synonymdev/react-native-ldk'; import * as bitcoin from 'bitcoinjs-lib'; import ElectrumClient from 'electrum-client'; import { Platform } from 'react-native'; @@ -12,12 +17,14 @@ import { TestProfile, getTxFeeRate, initWaitForElectrumToSync, + skipRemoteBackups, sleep, waitForLDKEvent, wipeLdkStorage, } from './utils'; import { Result } from '../utils/result'; import { getScriptHash } from '../utils/helpers'; +import { backupServerDetails } from './utils'; // import { // broadcastTransaction, @@ -89,31 +96,8 @@ describe('LND', function () { beforeEach(async () => { await waitForElectrum({ lnd: true }); - - // const electrumResponse = await connectToElectrum({}); - // if (electrumResponse.isErr()) { - // throw electrumResponse.error; - // } - // const account = await getAccount(); - - profile = new TestProfile({ - // name: account.name, - // seed: account.seed, - }); + profile = new TestProfile({}); await profile.init(); - - // const res = await ldk.backupSetup({ - // network: 'regtest', - // seed: profile.seed, - // details: { - // host: 'http://127.0.0.1:3003', - // serverPubKey: - // '0319c4ff23820afec0c79ce3a42031d7fef1dff78b7bdd69b5560684f3e1827675', - // }, - // }); - // if (res.isErr()) { - // throw res.error; - // } }); afterEach(async function () { @@ -136,9 +120,18 @@ describe('LND', function () { // - backup and restore LDK // - check if channel is still open - // const account = await getAccount(); + if (!skipRemoteBackups) { + const backupRes = await ldk.backupSetup({ + network: ENetworks.regtest, + seed: profile.getAccount().seed, + details: backupServerDetails, + }); + if (backupRes.isErr()) { + throw backupRes.error; + } + } - await ldk.stop(); + // await ldk.stop(); const lmStart = await lm.start({ ...profile.getStartParams(), // getBestBlock: getBestBlock, @@ -306,7 +299,10 @@ describe('LND', function () { throw claimableBalances1.error; } + return; + // backup LDK + // eslint-disable-next-line no-unreachable const backupResp = await lm.backupAccount({ account: profile.getAccount(), includeTransactionHistory: true, @@ -573,21 +569,19 @@ describe('LND', function () { // - force close channel from LDK // - check everything is ok - let fees = { - nonAnchorChannelFee: 5, - anchorChannelFee: 5, - maxAllowedNonAnchorChannelRemoteFee: 5, - channelCloseMinimum: 5, - minAllowedAnchorChannelRemoteFee: 5, - minAllowedNonAnchorChannelRemoteFee: 5, - onChainSweep: 5, - }; + if (!skipRemoteBackups) { + const backupRes = await ldk.backupSetup({ + network: ENetworks.regtest, + seed: profile.getAccount().seed, + details: backupServerDetails, + }); + if (backupRes.isErr()) { + throw backupRes.error; + } + } const lmStart = await lm.start({ ...profile.getStartParams(), - getFees: async () => { - return fees; - }, }); if (lmStart.isErr()) { throw lmStart.error; @@ -677,18 +671,18 @@ describe('LND', function () { EEventTypes.broadcast_transaction, ); - // set height fees and restart LDK so it catches up - fees = { - nonAnchorChannelFee: 30, - anchorChannelFee: 30, - maxAllowedNonAnchorChannelRemoteFee: 30, - channelCloseMinimum: 5, - minAllowedAnchorChannelRemoteFee: 5, - minAllowedNonAnchorChannelRemoteFee: 5, - onChainSweep: 30, - }; + // set high fees and restart LDK so it catches up + // fees = { + // nonAnchorChannelFee: 30, + // anchorChannelFee: 30, + // maxAllowedNonAnchorChannelRemoteFee: 30, + // channelCloseMinimum: 5, + // minAllowedAnchorChannelRemoteFee: 5, + // minAllowedNonAnchorChannelRemoteFee: 5, + // onChainSweep: 30, + // }; const syncRes0 = await lm.syncLdk(); - await lm.setFees(); + // await lm.setFees(); if (syncRes0.isErr()) { throw syncRes0.error; } @@ -734,9 +728,10 @@ describe('LND', function () { ({ amount_satoshis }) => amount_satoshis > 0, ); } + expect(claimableBalances1.value).to.have.length(1); expect(claimableBalances1.value[0]).to.include({ - claimable_amount_satoshis: 100001, + amount_satoshis: 100001, type: 'ClaimableAwaitingConfirmations', }); expect(claimableBalances1.value[0]) @@ -777,7 +772,7 @@ describe('LND', function () { ); await electrum.initElectrum({ client: 'get-balance', version: '1.4' }); const balance = await electrum.blockchainScripthash_getBalance( - getScriptHash(await profile.getAddress()), + getScriptHash((await profile.getAddress()).address), ); expect(balance) diff --git a/example/tests/unit.ts b/example/tests/unit.ts index d89550dc..87c5a140 100644 --- a/example/tests/unit.ts +++ b/example/tests/unit.ts @@ -1,9 +1,10 @@ +import lm, { ENetworks, ldk } from '@synonymdev/react-native-ldk'; import { expect } from 'chai'; import { describe, it } from 'mocha'; +import { Platform } from 'react-native'; import RNFS from 'react-native-fs'; -import lm, { ENetworks, ldk } from '@synonymdev/react-native-ldk'; -import { wipeLdkStorage } from './utils'; +import { backupServerDetails, wipeLdkStorage } from './utils'; describe('Unit', function () { this.timeout(1 * 60 * 1000); // 1 minute @@ -81,6 +82,7 @@ describe('Unit', function () { getTransactionPosition: async () => -1, broadcastTransaction: async () => '', network: ENetworks.regtest, + skipRemoteBackups: true, }); if (lmStart.isErr()) { @@ -165,33 +167,98 @@ describe('Unit', function () { } expect(nodesRes.value).to.be.an('array').that.is.empty; - const backupResp = await lm.backupAccount({ - account, - includeTransactionHistory: true, - }); - if (backupResp.isErr()) { - throw backupResp.error; - } - const br = backupResp.value; - expect(br.account.name).to.be.equal(account.name); - expect(br.account.seed).to.be.equal(account.seed); - expect(br.data.channel_manager).to.be.a('string').that.is.not.empty; - expect(br.data.channel_monitors).to.be.a('object').that.is.empty; - expect(br.data.peers).to.be.a('array').that.is.empty; - expect(br.data.unconfirmed_transactions).to.be.a('array').that.is.empty; - expect(br.data.broadcasted_transactions).to.be.a('array').that.is.empty; - expect(br.data.payment_ids).to.be.a('array').that.is.empty; - expect(br.data.spendable_outputs).to.be.a('array').that.is.empty; - expect(br.data.payments_claimed).to.be.a('array').that.is.empty; - expect(br.data.payments_sent).to.be.a('array').that.is.empty; - expect(br.data.bolt11_invoices).to.include(pr.value.to_str); - expect(br.data.timestamp).to.be.a('number'); - expect(br.package_version).to.be.a('string'); - expect(br.network).to.be.equal('regtest'); - const restart = await ldk.restart(); if (restart.isErr()) { throw restart.error; } }); + + it('can backup', async function () { + if (Platform.OS === 'android') { + return; + } + + const account = { + name: 'Mock2', + seed: '0000000000000000000000000000000000000000000000000000000000000001', + }; + + const backupRes = await ldk.backupSetup({ + network: ENetworks.regtest, + seed: account.seed, + details: backupServerDetails, + }); + if (backupRes.isErr()) { + throw backupRes.error; + } + + const startParams = { + getBestBlock: async () => ({ + hash: '6993d7c36204c9f369a7d9d80590c4e2a86b9641c327fbc276a534797ca50a2f', + height: 1, + hex: '000000205d1f3ece3bcb0a3530adaae2e4aa47f8aad434db95c6cb6d09d5ac99e3f6df4f0a19fe968830a06dcc49c9cca4acc532226e4c30b741802420e0fdd2b092eccdbf95cb64ffff7f2000000000', + }), + account, + getAddress: async () => ({ + address: 'bcrt1qtk89me2ae95dmlp3yfl4q9ynpux8mxjus4s872', + publicKey: + '0298720ece754e377af1b2716256e63c2e2427ff6ebdc66c2071c43ae80132ca32', + }), + getScriptPubKeyHistory: async () => [], + getFees: () => { + return Promise.resolve({ + nonAnchorChannelFee: 5, + anchorChannelFee: 5, + maxAllowedNonAnchorChannelRemoteFee: 5, + channelCloseMinimum: 5, + minAllowedAnchorChannelRemoteFee: 5, + minAllowedNonAnchorChannelRemoteFee: 5, + onChainSweep: 5, + }); + }, + getTransactionData: async () => ({ + header: '', + height: 0, + transaction: '', + vout: [], + }), + getTransactionPosition: async () => -1, + broadcastTransaction: async () => '', + network: ENetworks.regtest, + skipRemoteBackups: true, + }; + + const lmStart1 = await lm.start(startParams); + + if (lmStart1.isErr()) { + throw lmStart1.error; + } + + const fileListRes = await ldk.backupListFiles(); + if (fileListRes.isErr()) { + throw fileListRes.error; + } + + await ldk.stop(); + + const restoreRes = await lm.restoreFromRemoteServer({ + account: account, + serverDetails: backupServerDetails, + network: ENetworks.regtest, + overwrite: true, + }); + if (restoreRes.isErr()) { + throw restoreRes.error; + } + + const lmStart2 = await lm.start(startParams); + if (lmStart2.isErr()) { + throw lmStart2.error; + } + + const restartRes = await ldk.restart(); + if (restartRes.isErr()) { + throw restartRes.error; + } + }); }); diff --git a/example/tests/utils/index.ts b/example/tests/utils/index.ts index c58ba0e4..52093482 100644 --- a/example/tests/utils/index.ts +++ b/example/tests/utils/index.ts @@ -162,6 +162,7 @@ export const initWaitForElectrumToSync = async ( */ export const wipeLdkStorage = async (): Promise => { await ldk.stop(); + await sleep(10); if (lm.account.name === '') { return; @@ -190,7 +191,7 @@ export const wipeLdkStorage = async (): Promise => { } }; -export const getTxFeeRate = async (txid): Promise => { +export const getTxFeeRate = async (txid: string): Promise => { const electrum = new ElectrumClient( global.net, global.tls, @@ -229,6 +230,7 @@ export const waitForLDKEvent = ( } waiting = false; subs.remove(); + console.error(`WaitForLDKEvent ${event} timed out`); reject(`WaitForLDKEvent ${event} timed out`); }, timeout); @@ -243,3 +245,10 @@ export const waitForLDKEvent = ( }); }); }; + +export const skipRemoteBackups = false; +export const backupServerDetails = { + host: 'http://127.0.0.1:3003', + serverPubKey: + '0319c4ff23820afec0c79ce3a42031d7fef1dff78b7bdd69b5560684f3e1827675', +}; diff --git a/example/tests/utils/test-profile.ts b/example/tests/utils/test-profile.ts index ebe1baa0..4f0cc83a 100644 --- a/example/tests/utils/test-profile.ts +++ b/example/tests/utils/test-profile.ts @@ -1,9 +1,3 @@ -import ElectrumClient from 'electrum-client'; -import * as bip39 from 'bip39'; -import * as bip32 from 'bip32'; -import * as bitcoin from 'bitcoinjs-lib'; -import { randomBytes } from 'react-native-randombytes'; -import { Platform } from 'react-native'; import { IAddress, TAccount, @@ -13,7 +7,14 @@ import { TTransactionData, TTransactionPosition, } from '@synonymdev/react-native-ldk'; +import * as bip32 from 'bip32'; +import * as bip39 from 'bip39'; +import * as bitcoin from 'bitcoinjs-lib'; +import ElectrumClient from 'electrum-client'; +import { Platform } from 'react-native'; +import { randomBytes } from 'react-native-randombytes'; +import { skipRemoteBackups } from '.'; import { getAddressFromScriptPubKey, getNetwork, @@ -28,6 +29,7 @@ type TElectrumHeader = { export default class TestProfile { name: string; + enableBackups: boolean; electrum: ElectrumClient; seed: string; network: TAvailableNetworks; @@ -39,6 +41,7 @@ export default class TestProfile { this.seed = opts?.seed ?? randomBytes(32).toString('hex'); this.network = opts?.network ?? 'bitcoinRegtest'; this.headerCallback = opts?.headerCallback; + this.enableBackups = opts?.enableBackups ?? false; this.electrum = new ElectrumClient( global.net, global.tls, @@ -86,8 +89,8 @@ export default class TestProfile { network: ldkNetwork(this.network), // forceCloseOnStartup: { forceClose: true, broadcastLatestTx: false }, forceCloseOnStartup: undefined, - skipRemoteBackups: true, // trustedZeroConfPeers: [], + skipRemoteBackups, }; }; diff --git a/example/yarn.lock b/example/yarn.lock index 1d494ac3..8a80232d 100644 --- a/example/yarn.lock +++ b/example/yarn.lock @@ -1672,6 +1672,11 @@ minimatch "^3.1.2" strip-json-comments "^3.1.1" +"@flatten-js/interval-tree@^1.1.2": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@flatten-js/interval-tree/-/interval-tree-1.1.2.tgz#fcc891da48bc230392884be01c26fe8c625702e8" + integrity sha512-OwLoV9E/XM6b7bes2rSFnGNjyRy7vcoIHFTnmBR2WAaZTf0Fe4EX4GdA65vU1KgFAasti7iRSg2dZfYd1Zt00Q== + "@hapi/hoek@^9.0.0": version "9.3.0" resolved "https://registry.npmjs.org/@hapi/hoek/-/hoek-9.3.0.tgz" @@ -2404,8 +2409,9 @@ "@sinonjs/commons" "^3.0.0" "@synonymdev/react-native-ldk@../lib": - version "0.0.122" + version "0.0.127" dependencies: + bech32 "^2.0.0" bitcoinjs-lib "^6.0.2" "@tsconfig/react-native@^3.0.0": @@ -3380,6 +3386,16 @@ buffer@^6.0.3: base64-js "^1.3.1" ieee754 "^1.2.1" +bunyamin@^1.5.0: + version "1.5.2" + resolved "https://registry.yarnpkg.com/bunyamin/-/bunyamin-1.5.2.tgz#681db204c0b16531369d5c1f6c89dc8d760b7558" + integrity sha512-Xp2nfqk33zt3nX90OSTkLVOc5N+1zdR3MWvfLHoIrm3cGRkdxPTPYB9CCgrDV8oum5rbghJjAbmXFXOrRXvMtg== + dependencies: + "@flatten-js/interval-tree" "^1.1.2" + multi-sort-stream "^1.0.4" + stream-json "^1.7.5" + trace-event-lib "^1.3.1" + bunyan-debug-stream@^3.1.0: version "3.1.0" resolved "https://registry.npmjs.org/bunyan-debug-stream/-/bunyan-debug-stream-3.1.0.tgz" @@ -3397,6 +3413,18 @@ bunyan@^1.8.12: mv "~2" safe-json-stringify "~1" +bunyan@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/bunyan/-/bunyan-2.0.5.tgz#9dd056755220dddd8b5bb9cf76f3d0d766e96e71" + integrity sha512-Jvl74TdxCN6rSP9W1I6+UOUtwslTDqsSFkDqZlFb/ilaSvQ+bZAnXT/GT97IZ5L+Vph0joPZPhxUyn6FLNmFAA== + dependencies: + exeunt "1.1.0" + optionalDependencies: + dtrace-provider "~0.8" + moment "^2.19.3" + mv "~2" + safe-json-stringify "~1" + bytes@3.0.0: version "3.0.0" resolved "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz" @@ -3962,10 +3990,10 @@ detect-newline@^3.0.0: resolved "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz" integrity sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA== -detox@20.7.0: - version "20.7.0" - resolved "https://registry.yarnpkg.com/detox/-/detox-20.7.0.tgz#2bd6faa1503dbd25ffcbb8e85a35f63193f4be23" - integrity sha512-lBS//hl8NbusNx4E2Bucb8UciNPS9HhW1ejUIiEDgYuGQnuM33GcyQ6I4wXi4YUdrDYO2obpwxe/qLUg9DkVng== +detox@20.17.0: + version "20.17.0" + resolved "https://registry.yarnpkg.com/detox/-/detox-20.17.0.tgz#836d2ebb193572e1f4a97e44af343891a5658f7b" + integrity sha512-IasYgexfkrCoZuJTaqqKHQ2yflK+tnqifzdLwrp4hdTWlXUlG9j/YcM1Dn3ThSs3b6VNMtbSe6xoPkKD0oNiIQ== dependencies: ajv "^8.6.3" bunyan "^1.8.12" @@ -3974,16 +4002,17 @@ detox@20.7.0: chalk "^4.0.0" child-process-promise "^2.2.0" execa "^5.1.1" - find-up "^4.1.0" - fs-extra "^4.0.2" + find-up "^5.0.0" + fs-extra "^11.0.0" funpermaproxy "^1.1.0" glob "^8.0.3" ini "^1.3.4" + jest-environment-emit "^1.0.5" json-cycle "^1.3.0" lodash "^4.17.11" multi-sort-stream "^1.0.3" multipipe "^4.0.0" - node-ipc "^9.2.1" + node-ipc "9.2.1" proper-lockfile "^3.0.2" resolve-from "^5.0.0" sanitize-filename "^1.6.1" @@ -3998,8 +4027,8 @@ detox@20.7.0: trace-event-lib "^1.3.1" which "^1.3.1" ws "^7.0.0" - yargs "^16.0.3" - yargs-parser "^20.2.9" + yargs "^17.0.0" + yargs-parser "^21.0.0" yargs-unparser "^2.0.0" diff-sequences@^28.1.1: @@ -4459,6 +4488,11 @@ execa@^5.0.0, execa@^5.1.1: signal-exit "^3.0.3" strip-final-newline "^2.0.0" +exeunt@1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/exeunt/-/exeunt-1.1.0.tgz#af72db6f94b3cb75e921aee375d513049843d284" + integrity sha512-dd++Yn/0Fp+gtJ04YHov7MeAii+LFivJc6KqnJNfplzLVUkUDrfKoQDTLlCgzcW15vY5hKlHasWeIsQJ8agHsw== + exit@^0.1.2: version "0.1.2" resolved "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz" @@ -4679,14 +4713,14 @@ fs-extra@^0.22.1: jsonfile "^2.1.0" rimraf "^2.2.8" -fs-extra@^4.0.2: - version "4.0.3" - resolved "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.3.tgz" - integrity sha512-q6rbdDd1o2mAnQreO7YADIxf/Whx4AHBiRf6d+/cVT8h44ss+lHgxf1FemcqDnQt9X3ct4McHr+JMGlYSsK7Cg== +fs-extra@^11.0.0: + version "11.2.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-11.2.0.tgz#e70e17dfad64232287d01929399e0ea7c86b0e5b" + integrity sha512-PmDi3uwK5nFuXh7XDTlVnS17xJS7vW36is2+w3xcv8SVxiB4NyATf4ctkVY5bkSjX0Y4nbvZCq1/EjtEyr9ktw== dependencies: - graceful-fs "^4.1.2" - jsonfile "^4.0.0" - universalify "^0.1.0" + graceful-fs "^4.2.0" + jsonfile "^6.0.1" + universalify "^2.0.0" fs-extra@^8.1.0: version "8.1.0" @@ -5442,6 +5476,20 @@ jest-each@^29.7.0: jest-util "^29.7.0" pretty-format "^29.7.0" +jest-environment-emit@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/jest-environment-emit/-/jest-environment-emit-1.0.5.tgz#e6f33451f98b88ccd48e9e1188bb535880f03c1b" + integrity sha512-OsQ08AhYxkkyDBTIow+9ogNmJheQIGWQKp0Nku+1ToLWjAj2Pd6LmypN8HgUIqYHs4HFcqkQ25kaf1qExmoZpg== + dependencies: + bunyamin "^1.5.0" + bunyan "^2.0.5" + bunyan-debug-stream "^3.1.0" + funpermaproxy "^1.1.0" + lodash.merge "^4.6.2" + node-ipc "9.2.1" + strip-ansi "^6.0.0" + tslib "^2.5.3" + jest-environment-node@^29.2.1, jest-environment-node@^29.7.0: version "29.7.0" resolved "https://registry.yarnpkg.com/jest-environment-node/-/jest-environment-node-29.7.0.tgz#0b93e111dda8ec120bc8300e6d1fb9576e164376" @@ -5948,6 +5996,15 @@ jsonfile@^4.0.0: optionalDependencies: graceful-fs "^4.1.6" +jsonfile@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-6.1.0.tgz#bc55b2634793c679ec6403094eb13698a6ec0aae" + integrity sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ== + dependencies: + universalify "^2.0.0" + optionalDependencies: + graceful-fs "^4.1.6" + "jsx-ast-utils@^2.4.1 || ^3.0.0": version "3.3.3" resolved "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.3.tgz" @@ -6834,7 +6891,7 @@ ms@2.1.3: resolved "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz" integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== -multi-sort-stream@^1.0.3: +multi-sort-stream@^1.0.3, multi-sort-stream@^1.0.4: version "1.0.4" resolved "https://registry.npmjs.org/multi-sort-stream/-/multi-sort-stream-1.0.4.tgz" integrity sha512-hAZ8JOEQFbgdLe8HWZbb7gdZg0/yAIHF00Qfo3kd0rXFv96nXe+/bPTrKHZ2QMHugGX4FiAyET1Lt+jiB+7Qlg== @@ -6925,9 +6982,9 @@ node-int64@^0.4.0: resolved "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz" integrity sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw== -node-ipc@^9.2.1: +node-ipc@9.2.1: version "9.2.1" - resolved "https://registry.npmjs.org/node-ipc/-/node-ipc-9.2.1.tgz" + resolved "https://registry.yarnpkg.com/node-ipc/-/node-ipc-9.2.1.tgz#b32f66115f9d6ce841dc4ec2009d6a733f98bb6b" integrity sha512-mJzaM6O3xHf9VT8BULvJSbdVbmHUKRNOH7zDDkCrA1/T+CVjq2WVIDfLt0azZRXpgArJtl3rtmEozrbXPZ9GaQ== dependencies: event-pubsub "4.3.0" @@ -8189,6 +8246,13 @@ stream-json@^1.7.4: dependencies: stream-chain "^2.2.5" +stream-json@^1.7.5: + version "1.8.0" + resolved "https://registry.yarnpkg.com/stream-json/-/stream-json-1.8.0.tgz#53f486b2e3b4496c506131f8d7260ba42def151c" + integrity sha512-HZfXngYHUAr1exT4fxlbc1IOce1RYxp2ldeaf97LYCOPSoOqY/1Psp7iGvpb+6JIOgkra9zDYnPX01hGAHzEPw== + dependencies: + stream-chain "^2.2.5" + string-length@^4.0.1: version "4.0.2" resolved "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz" @@ -8459,6 +8523,11 @@ tslib@^2.1.0: resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.1.tgz#fd8c9a0ff42590b25703c0acb3de3d3f4ede0410" integrity sha512-t0hLfiEKfMUoqhG+U1oid7Pva4bbDPHYfJNiB7BiIjRkj1pyC++4N3huJfqY6aRH6VTB0rvtzQwjM4K6qpfOig== +tslib@^2.5.3: + version "2.6.2" + resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.2.tgz#703ac29425e7b37cd6fd456e92404d46d1f3e4ae" + integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q== + tsutils@^3.21.0: version "3.21.0" resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-3.21.0.tgz#b48717d394cea6c1e096983eed58e9d61715b623" @@ -8549,6 +8618,11 @@ universalify@^0.1.0: resolved "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz" integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== +universalify@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/universalify/-/universalify-2.0.1.tgz#168efc2180964e6386d061e094df61afe239b18d" + integrity sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw== + unpipe@~1.0.0: version "1.0.0" resolved "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz" @@ -8831,12 +8905,12 @@ yargs-parser@^18.1.2: camelcase "^5.0.0" decamelize "^1.2.0" -yargs-parser@^20.2.2, yargs-parser@^20.2.9: +yargs-parser@^20.2.2: version "20.2.9" resolved "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz" integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w== -yargs-parser@^21.1.1: +yargs-parser@^21.0.0, yargs-parser@^21.1.1: version "21.1.1" resolved "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz" integrity sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw== @@ -8851,7 +8925,7 @@ yargs-unparser@2.0.0, yargs-unparser@^2.0.0: flat "^5.0.2" is-plain-obj "^2.1.0" -yargs@16.2.0, yargs@^16.0.3, yargs@^16.2.0: +yargs@16.2.0, yargs@^16.2.0: version "16.2.0" resolved "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz" integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw== @@ -8881,10 +8955,10 @@ yargs@^15.1.0: y18n "^4.0.0" yargs-parser "^18.1.2" -yargs@^17.3.1: - version "17.6.2" - resolved "https://registry.npmjs.org/yargs/-/yargs-17.6.2.tgz" - integrity sha512-1/9UrdHjDZc0eOU0HxOHoS78C69UD3JRMvzlJ7S79S2nTaWRA/whGCTV8o9e/N/1Va9YIV7Q4sOxD8VV4pCWOw== +yargs@^17.0.0, yargs@^17.6.2, yargs@^17.7.2: + version "17.7.2" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" + integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w== dependencies: cliui "^8.0.1" escalade "^3.1.1" @@ -8894,10 +8968,10 @@ yargs@^17.3.1: y18n "^5.0.5" yargs-parser "^21.1.1" -yargs@^17.6.2, yargs@^17.7.2: - version "17.7.2" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" - integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w== +yargs@^17.3.1: + version "17.6.2" + resolved "https://registry.npmjs.org/yargs/-/yargs-17.6.2.tgz" + integrity sha512-1/9UrdHjDZc0eOU0HxOHoS78C69UD3JRMvzlJ7S79S2nTaWRA/whGCTV8o9e/N/1Va9YIV7Q4sOxD8VV4pCWOw== dependencies: cliui "^8.0.1" escalade "^3.1.1"