From bf9b3c613c9d29fca31530758f95a3c46766bd2b Mon Sep 17 00:00:00 2001 From: Fuxing Loh <4266087+fuxingloh@users.noreply.github.com> Date: Mon, 8 Mar 2021 09:22:57 +0800 Subject: [PATCH] testcontainers for defichain js ecosystem (#39) * removed main/test net workflow * testcontainers for defichain js ecosystem * check in .idea code styles * automatically pull docker image for testing * cleanup code debt * added escape-hatch, unable to find open handles * added DeFiDRpcError for easier downstream error surfacing --- .github/workflows/ci.yml | 50 +-- .gitignore | 2 + .idea/README.md | 21 ++ .idea/codeStyles/Project.xml | 12 + .idea/codeStyles/codeStyleConfig.xml | 5 + .idea/dictionaries/fuxing.xml | 58 +++- README.md | 19 +- jest.config.js | 3 +- package-lock.json | 15 +- package.json | 3 +- packages/testcontainers/README.md | 114 ++++++- .../__tests__/chains/container.test.ts | 58 ++++ .../chains/reg_test_container.test.ts | 94 ++++++ .../__tests__/testcontainers.test.ts | 56 +++- .../testcontainers/__tests__/testkeys.test.ts | 27 ++ packages/testcontainers/jest.config.js | 3 +- .../testcontainers/src/chains/container.ts | 316 ++++++++++++++++++ .../src/chains/main_net_container.ts | 12 + .../src/chains/reg_test_container.ts | 105 ++++++ .../src/chains/test_net_container.ts | 16 + packages/testcontainers/src/testcontainers.ts | 15 +- packages/testcontainers/src/testkeys.ts | 99 ++++++ 22 files changed, 1038 insertions(+), 65 deletions(-) create mode 100644 .idea/README.md create mode 100644 .idea/codeStyles/Project.xml create mode 100644 .idea/codeStyles/codeStyleConfig.xml create mode 100644 packages/testcontainers/__tests__/chains/container.test.ts create mode 100644 packages/testcontainers/__tests__/chains/reg_test_container.test.ts create mode 100644 packages/testcontainers/__tests__/testkeys.test.ts create mode 100644 packages/testcontainers/src/chains/container.ts create mode 100644 packages/testcontainers/src/chains/main_net_container.ts create mode 100644 packages/testcontainers/src/chains/reg_test_container.ts create mode 100644 packages/testcontainers/src/chains/test_net_container.ts create mode 100644 packages/testcontainers/src/testkeys.ts diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index e1256917ba..96e0f5914a 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -1,4 +1,3 @@ -# Shift-left testing to mitigation upstream risk name: CI on: @@ -9,7 +8,7 @@ on: jobs: build: - name: Build + name: Build & Test runs-on: ubuntu-latest steps: - uses: actions/checkout@v2 @@ -21,50 +20,5 @@ jobs: - run: npm run build - run: npm run standard - - run: npx --no-install jest --ci --coverage + - run: npx --no-install jest --ci --coverage --forceExit - run: npx codecov - - reg-net: - name: RegNet - runs-on: ubuntu-latest - needs: [ build ] - steps: - - uses: actions/checkout@v2 - - uses: actions/setup-node@v2 - with: - node-version: '15' - - # TODO(fuxingloh): reg net testing - - run: | - echo "::error::not yet implemented" - exit 1 - - test-net: - name: TestNet - runs-on: ubuntu-latest - needs: [ build ] - steps: - - uses: actions/checkout@v2 - - uses: actions/setup-node@v2 - with: - node-version: '15' - - # TODO(fuxingloh): test net testing - - run: | - echo "::error::not yet implemented" - exit 1 - - main-net: - name: MainNet - runs-on: ubuntu-latest - needs: [ reg-net, test-net ] - steps: - - uses: actions/checkout@v2 - - uses: actions/setup-node@v2 - with: - node-version: '15' - - # TODO(fuxingloh): main net testing? - - run: | - echo "::error::not yet implemented" - exit 1 diff --git a/.gitignore b/.gitignore index 979b6f1edd..a4bf8054dd 100644 --- a/.gitignore +++ b/.gitignore @@ -2,9 +2,11 @@ .idea/* !.idea/inspectionProfiles/ !.idea/dictionaries/ +!.idea/codeStyles/ !.idea/vcs.xml !.idea/modules.xml !.idea/*.iml +!.idea/README.md # Yarn .yarn/* diff --git a/.idea/README.md b/.idea/README.md new file mode 100644 index 0000000000..d5f97872b5 --- /dev/null +++ b/.idea/README.md @@ -0,0 +1,21 @@ +# `.idea` + +If you are reading this and wondering why `.idea/` files are commited to GitHub. Well... + +### `.idea/dictionaries/*` + +> IntelliJ IDEA helps you make sure that all your source code, including variable names, textual strings, comments, literals, and commit messages, is spelt correctly. For this purpose, IntelliJ IDEA provides a dedicated Typo inspection which is enabled by default. + +1. Spellcheck everything, code are no expections +2. Adopt the principles of fixing all red/orange/yellow/wavy errors. +3. Treat all text as code, because they are and can introduces bugs. `client.call('generatetoaddress')` + vs `client.call('generatetoadress')` +4. Lastly, share that same improved developer experience with all IntelliJ IDEA users. + +### `.idea/codeStyles/*` + +Although ts-standard --fix the code styles, the provided code style is provided to match ts-standard defaults. + +### `.idea/*.xml` + +For convenience and better developer onboarding experience as suggested by IntelliJ and @fuxingloh. diff --git a/.idea/codeStyles/Project.xml b/.idea/codeStyles/Project.xml new file mode 100644 index 0000000000..c90e46612b --- /dev/null +++ b/.idea/codeStyles/Project.xml @@ -0,0 +1,12 @@ + + + + diff --git a/.idea/codeStyles/codeStyleConfig.xml b/.idea/codeStyles/codeStyleConfig.xml new file mode 100644 index 0000000000..79ee123c2b --- /dev/null +++ b/.idea/codeStyles/codeStyleConfig.xml @@ -0,0 +1,5 @@ + + + + \ No newline at end of file diff --git a/.idea/dictionaries/fuxing.xml b/.idea/dictionaries/fuxing.xml index b8b0a53530..d230bca44d 100644 --- a/.idea/dictionaries/fuxing.xml +++ b/.idea/dictionaries/fuxing.xml @@ -1,3 +1,57 @@ - - \ No newline at end of file + + + accounttoutxos + amkheight + bayfrontgardensheight + bayfrontheight + bayfrontmarinaheight + bech + clarkequayheight + createmasternode + dakotaheight + debugexclude + defi + defichain + defid + devnet + dockerode + dummypos + fuxingloh + generatetoaddress + getaddressinfo + getbalance + getblockcount + getblockhash + getmintinginfo + getnewaddress + getreceivedbyaddress + gettransaction + getunconfirmedbalance + importprivkey + jsonrpc + libevent + listaccounts + logtimemicros + mainnet + masternode + nblocks + nmasternode + nospv + printtoconsole + priv + regtest + rewardaddress + rpcallowip + rpcbind + rpcpassword + rpcuser + segwit + sendtoaddress + testcontainers + txnotokens + uacomment + utxostoaccount + + + diff --git a/README.md b/README.md index 002dcf473c..10b75491e2 100644 --- a/README.md +++ b/README.md @@ -64,18 +64,29 @@ npm install ### Testing `jest.config.js` is set up at the root project level as well as at each sub module. You can run jest at root to test all -modules or individually at each sub module. If you use IntelliJ IDEA, you can right click any file to test it -individually and have it reported to the IDE. +modules or individually at each sub module. By default, only regtest chain are used for normal testing. If you use +IntelliJ IDEA, you can right click any file to test it individually and have it reported to the IDE. Docker is required to run the tests as [`@defichain/testcontainers`](./packages/testcontainers) will automatically spin -up `regtest` instances for testing. +up `regtest` instances for testing. The number of containers it will spin up concurrently is dependent on your +jest `--maxConcurrency` count. -Coverage is collected at merge with `codecov`; more testing 🚀 less 🐛 = 😎 +Coverage is collected at pr merge with `codecov`; more testing 🚀 less 🐛 = 😎 ```shell jest ``` +### Publishing + +`"version": "0.0.0"` is used because publishing will be done automatically +by [GitHub releases](https://github.com/DeFiCh/jellyfish/releases) with connected workflows. On +release `types: [ published, prereleased ]`, GitHub Action will automatically build all packages in this repo and +publish it into npm. + +* release are tagged as `@latest` +* prerelease are tagged as `@next` (please use this cautiously) + ### IntelliJ IDEA IntelliJ IDEA is the IDE of choice for writing and maintaining this library. IntelliJ's files are included for diff --git a/jest.config.js b/jest.config.js index 9814ca2f36..bac1eacb35 100644 --- a/jest.config.js +++ b/jest.config.js @@ -1,5 +1,6 @@ module.exports = { projects: [ '/packages/*' - ] + ], + testTimeout: 240000 } diff --git a/package-lock.json b/package-lock.json index 17f5ad6ed4..59d3ae6673 100644 --- a/package-lock.json +++ b/package-lock.json @@ -18,7 +18,8 @@ "size-limit": "^4.9.2", "ts-jest": "^26.5.2", "ts-standard": "^10.0.0", - "typescript": "^4.2.2" + "typescript": "^4.2.2", + "wait-for-expect": "^3.0.2" } }, "node_modules/@babel/code-frame": { @@ -29922,6 +29923,12 @@ "node": ">=10" } }, + "node_modules/wait-for-expect": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/wait-for-expect/-/wait-for-expect-3.0.2.tgz", + "integrity": "sha512-cfS1+DZxuav1aBYbaO/kE06EOS8yRw7qOFoD3XtjTkYvCvh3zUvNST8DXK/nPaeqIzIv3P3kL3lRJn8iwOiSag==", + "dev": true + }, "node_modules/walker": { "version": "1.0.7", "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.7.tgz", @@ -54286,6 +54293,12 @@ "xml-name-validator": "^3.0.0" } }, + "wait-for-expect": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/wait-for-expect/-/wait-for-expect-3.0.2.tgz", + "integrity": "sha512-cfS1+DZxuav1aBYbaO/kE06EOS8yRw7qOFoD3XtjTkYvCvh3zUvNST8DXK/nPaeqIzIv3P3kL3lRJn8iwOiSag==", + "dev": true + }, "walker": { "version": "1.0.7", "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.7.tgz", diff --git a/package.json b/package.json index a3feb120dc..3950df7915 100644 --- a/package.json +++ b/package.json @@ -34,7 +34,8 @@ "size-limit": "^4.9.2", "ts-jest": "^26.5.2", "ts-standard": "^10.0.0", - "typescript": "^4.2.2" + "typescript": "^4.2.2", + "wait-for-expect": "^3.0.2" }, "lint-staged": { "*.{ts,js}": [ diff --git a/packages/testcontainers/README.md b/packages/testcontainers/README.md index c93c88c4f7..9051d9f9ca 100644 --- a/packages/testcontainers/README.md +++ b/packages/testcontainers/README.md @@ -1,3 +1,115 @@ # @defichain/testcontainers - +Similar to [testcontainers](https://www.testcontainers.org/) in the Java ecosystem, this package provides a lightweight, +throwaway instances of `regtest`, `testnet` or `mainnet` provisioned automatically in Docker container. +`@defichain/testcontainers` encapsulate on top of `defi/defichain:v1.x` and directly interface with the Docker REST API. + +With `@defichain/testcontainers`, it allows the JS developers to: + +1. End-to-end test their application without hassle of setting up the toolchain +2. Run parallel tests as port number and container are dynamically generated on demand +3. Supercharge our CI workflow; run locally, anywhere or CI (as long as it has Docker installed) +4. Supercharge your `@defichain/jellyfish` implementation with 100% day 1 compatibility (mono repo!) +5. Bring quality and reliability to dApps on the DeFiChain JS ecosystem + +## Usage Example + +Install as dev only as you don't need this in production. **Please don't use this in production!** + +```shell +npm i -D @defichain/testcontainers +``` + +Use your favourite jest runner and start building dApps! + +### Basic `RegTestContainer` setup + +```js +import { RegTestDocker } from '@defichain/testcontainers' +import { getClient } from 'somewhere' + +describe('reg test', () => { + const node = new RegTestDocker() + + beforeEach(async () => { + await node.start({ + user: 'foo', + password: 'bar', + }) + await node.ready() + }) + + afterEach(async () => { + await node.stop() + }) + + it('should getmintinginfo and chain should be regtest', async () => { + const url = await node.getRpcUrl() + const client = getClient(url) + const result = await client.call('getmintinginfo', []) + expect(result.chain).toBe('regtest') + }) +}) +``` + +### `MasterNodeRegTestContainer` with auto-minting + +```js +import { MasterNodeRegTestContainer } from '@defichain/testcontainers' +import waitForExpect from "wait-for-expect"; + +describe('master node pos minting', () => { + const container = new MasterNodeRegTestContainer() + + beforeEach(async () => { + await container.start() + await container.waitForReady() + }) + + afterEach(async () => { + await container.stop() + }) + + it('should wait until coinbase maturity with spendable balance', async () => { + await container.generate(100) + + await waitForExpect(async () => { + const info = await container.getMintingInfo() + expect(info.blocks).toBeGreaterThan(100) + }) + + // perform utxostoaccount rpc + const address = await container.getNewAddress() + const payload: { [key: string]: string } = {} + payload[address] = "100@0" + await container.call("utxostoaccount", [payload]) + }) +}) +``` + +### Endpoint? + +```js +const container = new MasterNodeRegTestContainer() +const rpcURL = await container.getCachedRpcUrl() + +// they are dynmaically assigned to host, you can run multiple concurrent tests! +const port = await container.getPort('8555/tcp') +``` + +### Included `container.call('method', [])` for convenience RPC calls + +```js +const container = new MasterNodeRegTestContainer() +await container.start() +await container.waitForReady() + +// raw universal calls +const { blocks } = await container.call('getmintinginfo') +const address = await container.call('getnewaddress', ['label', 'legacy']) + +// basic included types +const count = await container.getBlockCount() +const info = await container.getMintingInfo() +const newAddress = await container.getNewAddress() +``` diff --git a/packages/testcontainers/__tests__/chains/container.test.ts b/packages/testcontainers/__tests__/chains/container.test.ts new file mode 100644 index 0000000000..ce1d61e956 --- /dev/null +++ b/packages/testcontainers/__tests__/chains/container.test.ts @@ -0,0 +1,58 @@ +import { RegTestContainer } from '../../src/chains/reg_test_container' +import { DeFiDContainer, DeFiDRpcError, StartOptions } from '../../src/chains/container' + +describe('container error handling', () => { + let container: DeFiDContainer + + afterEach(async () => { + try { + await container?.stop() + } catch (ignored) { + } + }) + + it('should error immediately if required container is not yet started', async () => { + container = new RegTestContainer() + return await expect(container.getRpcPort()) + .rejects.toThrow(/container not yet started/) + }) + + it('should error rpc as DeFiDRpcError', async () => { + container = new RegTestContainer() + await container.start() + await container.waitForReady() + return await expect(container.call('invalid')) + .rejects.toThrowError(DeFiDRpcError) + }) + + it('should get error: container might have crashed if invalid Cmd is present', async () => { + class InvalidCmd extends RegTestContainer { + public getCmd (opts: StartOptions): string[] { + return [ + ...super.getCmd(opts), + '-invalid=123' + ] + } + } + + container = new InvalidCmd() + await container.start() + return expect(container.waitForReady(3000)) + .rejects.toThrow(/Unable to find rpc port, the container might have crashed/) + }) + + it('should get error: container not found if container is stopped', async () => { + container = new RegTestContainer() + await container.start() + await container.stop() + return await expect(container.getRpcPort()) + .rejects.toThrow(/\(HTTP code 404\) no such container - No such container:/) + }) + + it('should fail fast if wait for is set to 100ms', async () => { + container = new RegTestContainer() + await container.start() + await expect(container.waitForReady(100)) + .rejects.toThrow(/DeFiDContainer docker not ready within given timeout of/) + }) +}) diff --git a/packages/testcontainers/__tests__/chains/reg_test_container.test.ts b/packages/testcontainers/__tests__/chains/reg_test_container.test.ts new file mode 100644 index 0000000000..82d2796e01 --- /dev/null +++ b/packages/testcontainers/__tests__/chains/reg_test_container.test.ts @@ -0,0 +1,94 @@ +import { + MasterNodeRegTestContainer, + RegTestContainer +} from '../../src/chains/reg_test_container' +import waitForExpect from 'wait-for-expect' +import { GenesisKeys } from '../../src/testkeys' + +describe('regtest', () => { + const container = new RegTestContainer() + + beforeAll(async () => { + await container.start() + await container.waitForReady() + }) + + afterAll(async () => { + await container.stop() + }) + + it('should be block 0 for getmintinginfo rpc', async () => { + const info = await container.getMintingInfo() + expect(info.blocks).toBe(0) + expect(info.chain).toBe('regtest') + }) + + it('should be block 0 for getblockcount rpc', async () => { + const count = await container.getBlockCount() + expect(count).toBe(0) + }) + + describe('address', () => { + it('should be able to getnewaddress', async () => { + const address = await container.getNewAddress() + expect(address.length).toBe(44) + }) + + it('should be able to getnewaddress with label and as p2sh-segwit', async () => { + const address = await container.getNewAddress('not-default', 'p2sh-segwit') + expect(address.length).toBe(35) + }) + + it('should be able to getnewaddress with label and as legacy', async () => { + const address = await container.getNewAddress('not-default', 'legacy') + expect(address.length).toBe(34) + }) + }) +}) + +describe('master node pos minting', () => { + const container = new MasterNodeRegTestContainer() + + beforeEach(async () => { + await container.start() + await container.waitForReady() + }) + + afterEach(async () => { + await container.stop() + }) + + it('should auto generate coin in master node mode', async () => { + await waitForExpect(async () => { + const info = await container.getMintingInfo() + expect(info.blocks).toBeGreaterThan(3) + }) + }) + + it('should wait until coinbase maturity with spendable balance', async () => { + const key = GenesisKeys[2].operator + await container.generate(10, key.address) + await container.generate(100) + + await waitForExpect(async () => { + const info = await container.getMintingInfo() + expect(info.blocks).toBeGreaterThan(100) + }) + + await container.generate(3) + + await waitForExpect(async () => { + const balance = await container.call('getbalance') + expect(balance).toBeGreaterThan(150) + }) + }) + + it('should be able to perform amk rpc feature', async () => { + await container.generate(105) + + const address = await container.getNewAddress() + const payload: { [key: string]: string } = {} + payload[address] = '100@0' + await container.call('utxostoaccount', [payload]) + }) +}) diff --git a/packages/testcontainers/__tests__/testcontainers.test.ts b/packages/testcontainers/__tests__/testcontainers.test.ts index a5e9970f50..bd44f6f608 100644 --- a/packages/testcontainers/__tests__/testcontainers.test.ts +++ b/packages/testcontainers/__tests__/testcontainers.test.ts @@ -1,5 +1,55 @@ -import { getName } from '../src/testcontainers' +import { MainNetContainer, TestNetContainer, RegTestContainer } from '../src/testcontainers' -it('should getName testcontainers', () => { - expect(getName()).toBe('testcontainers') +describe('main', () => { + const container = new MainNetContainer() + + beforeEach(async () => { + await container.start() + await container.waitForReady() + }) + + afterEach(async () => { + await container.stop() + }) + + it('should be able to getmintinginfo and chain should be main', async () => { + const { chain } = await container.getMintingInfo() + expect(chain).toBe('main') + }) +}) + +describe('test', () => { + const container = new TestNetContainer() + + beforeEach(async () => { + await container.start() + await container.waitForReady() + }) + + afterEach(async () => { + await container.stop() + }) + + it('should be able to getmintinginfo and chain should be test', async () => { + const { chain } = await container.getMintingInfo() + expect(chain).toBe('test') + }) +}) + +describe('regtest', () => { + const container = new RegTestContainer() + + beforeEach(async () => { + await container.start() + await container.waitForReady() + }) + + afterEach(async () => { + await container.stop() + }) + + it('should be able to getmintinginfo and chain should be regtest', async () => { + const { chain } = await container.getMintingInfo() + expect(chain).toBe('regtest') + }) }) diff --git a/packages/testcontainers/__tests__/testkeys.test.ts b/packages/testcontainers/__tests__/testkeys.test.ts new file mode 100644 index 0000000000..cd2b904841 --- /dev/null +++ b/packages/testcontainers/__tests__/testkeys.test.ts @@ -0,0 +1,27 @@ +import { RegTestContainer } from '../src/chains/reg_test_container' +import { GenesisKeys } from '../src/testkeys' + +describe('genesis keys', () => { + const container = new RegTestContainer() + + beforeEach(async () => { + await container.start() + await container.waitForReady() + }) + + afterEach(async () => { + await container.stop() + }) + + it('should be able to import all priv key with valid address', async () => { + for (const key of GenesisKeys) { + await container.call('importprivkey', [key.operator.privKey]) + await container.call('importprivkey', [key.owner.privKey]) + } + + for (const key of GenesisKeys) { + await container.call('getaddressinfo', [key.operator.address]) + await container.call('getaddressinfo', [key.owner.address]) + } + }) +}) diff --git a/packages/testcontainers/jest.config.js b/packages/testcontainers/jest.config.js index 440ec8dbe9..b9f5804e1e 100644 --- a/packages/testcontainers/jest.config.js +++ b/packages/testcontainers/jest.config.js @@ -7,5 +7,6 @@ module.exports = { '^.+\\.ts$': 'ts-jest' }, verbose: true, - clearMocks: true + clearMocks: true, + testTimeout: 120000 } diff --git a/packages/testcontainers/src/chains/container.ts b/packages/testcontainers/src/chains/container.ts new file mode 100644 index 0000000000..a61d25b1d0 --- /dev/null +++ b/packages/testcontainers/src/chains/container.ts @@ -0,0 +1,316 @@ +import Dockerode, { DockerOptions, Container, ContainerInfo } from 'dockerode' +import fetch from 'node-fetch' +import JSONBig from 'json-bigint' + +/** + * Types of network as per https://github.com/DeFiCh/ain/blob/bc231241/src/chainparams.cpp#L825-L836 + */ +type Network = 'mainnet' | 'testnet' | 'devnet' | 'regtest' + +/** + * Mandatory options to start defid with + */ +export interface StartOptions { + // TODO(fuxingloh): change to cookie based auth soon + user?: string + password?: string +} + +const defaultStartOptions = { + user: 'testcontainers-user', + password: 'testcontainers-password' +} + +/** + * Generate a name for a new docker container with network type and random number + */ +function generateName (network: Network): string { + const rand = Math.floor(Math.random() * 10000000) + return `${DeFiDContainer.PREFIX}-${network}-${rand}` +} + +/** + * Clean up stale nodes are nodes that are running for 1 hour + */ +async function cleanUpStale (docker: Dockerode): Promise { + /** + * Same prefix and created more than 1 hour ago + */ + const isStale = (containerInfo: ContainerInfo): boolean => { + if (containerInfo.Names.filter((value) => value.startsWith(DeFiDContainer.PREFIX)).length > 0) { + return containerInfo.Created + 60 * 60 < Date.now() / 1000 + } + + return false + } + + /** + * Stop container that are running, remove them after + */ + const tryStopRemove = async (containerInfo: ContainerInfo): Promise => { + const container = docker.getContainer(containerInfo.Id) + if (containerInfo.State === 'running') { + await container.stop() + } + await container.remove() + } + + return await new Promise((resolve, reject) => { + docker.listContainers({ all: 1 }, (error, result) => { + if (error instanceof Error) { + return reject(error) + } + + const promises = (result ?? []) + .filter(isStale) + .map(tryStopRemove) + + Promise.all(promises).finally(resolve) + }) + }) +} + +/** + * Pull DeFiDContainer.image from DockerHub + */ +async function pullImage (docker: Dockerode): Promise { + return await new Promise((resolve, reject) => { + docker.pull(DeFiDContainer.image, {}, (error, result) => { + if (error instanceof Error) { + reject(error) + return + } + docker.modem.followProgress(result, () => { + resolve() + }) + }) + }) +} + +/** + * DeFiChain defid node managed in docker + */ +export abstract class DeFiDContainer { + /* eslint-disable @typescript-eslint/no-non-null-assertion, no-void */ + public static readonly PREFIX = 'defichain-testcontainers-' + public static readonly image = 'defi/defichain:1.5.0' + + protected readonly docker: Dockerode + protected readonly network: Network + + protected container?: Container + protected startOptions?: StartOptions + protected cachedRpcUrl?: string + + protected constructor (network: Network, options?: DockerOptions) { + this.docker = new Dockerode(options) + this.network = network + } + + /** + * Require container, else error exceptionally. + * Not a clean design, but it keep the complexity of this implementation low. + */ + protected requireContainer (): Container { + if (this.container !== undefined) { + return this.container + } + throw new Error('container not yet started') + } + + /** + * Convenience Cmd builder with StartOptions + */ + protected getCmd (opts: StartOptions): string[] { + return [ + 'defid', + '-printtoconsole', + '-rpcallowip=172.17.0.0/16', + '-rpcbind=0.0.0.0', + `-rpcuser=${opts.user!}`, + `-rpcpassword=${opts.password!}` + ] + } + + /** + * Always pull a version of DeFiDContainer.image, + * Create container and start it immediately + */ + async start (startOptions: StartOptions = {}): Promise { + await pullImage(this.docker) + this.startOptions = Object.assign(defaultStartOptions, startOptions) + this.container = await this.docker.createContainer({ + name: generateName(this.network), + Image: DeFiDContainer.image, + Tty: true, + Cmd: this.getCmd(this.startOptions), + HostConfig: { + PublishAllPorts: true + } + }) + await this.container.start() + } + + /** + * Get host machine port + * + * @param name of ExposedPorts e.g. '80/tcp' + */ + async getPort (name: string): Promise { + const container = this.requireContainer() + + return await new Promise((resolve, reject) => { + container.inspect(function (err, data) { + if (err instanceof Error) { + return reject(err) + } + + if (data?.NetworkSettings.Ports[name] !== undefined) { + return resolve(data.NetworkSettings.Ports[name][0].HostPort) + } + + return reject(new Error('Unable to find rpc port, the container might have crashed')) + }) + }) + } + + /** + * Get host machine port used for defid rpc + */ + public abstract getRpcPort (): Promise + + /** + * Get host machine url used for defid rpc calls with auth + */ + async getCachedRpcUrl (): Promise { + if (this.cachedRpcUrl === undefined) { + const port = await this.getRpcPort() + const user = this.startOptions!.user! + const password = this.startOptions!.password! + this.cachedRpcUrl = `http://${user}:${password}@127.0.0.1:${port}/` + } + return this.cachedRpcUrl + } + + /** + * Utility rpc function for the current node, for convenience sake. + * This is not error checked, it will just return the raw result. + * @throws DeFiDRpcError for rpc call errors + */ + async call (method: string, params: any = []): Promise { + const url = await this.getCachedRpcUrl() + const response = await fetch(url, { + method: 'POST', + body: JSONBig.stringify({ + jsonrpc: '1.0', + id: Math.floor(Math.random() * 10000000000000000), + method: method, + params: params + }) + }) + const text = await response.text() + const { result, error } = JSONBig.parse(text) + + // surface as DeFiDRpcError for downstream type checking + if (error !== null) { + throw new DeFiDRpcError(result) + } + + return result + } + + /** + * Convenience method to getmintinginfo, typing mapping is non exhaustive + */ + async getMintingInfo (): Promise<{ blocks: number, chain: string }> { + return await this.call('getmintinginfo', []) + } + + /** + * Convenience method to getblockcount, typing mapping is non exhaustive + */ + async getBlockCount (): Promise { + return await this.call('getblockcount', []) + } + + /** + * Wait for rpc to be ready, default to 15000ms + */ + private async waitForRpc (timeout = 15000): Promise { + const expiredAt = Date.now() + timeout + + return await new Promise((resolve, reject) => { + const checkReady = (): void => { + this.cachedRpcUrl = undefined + this.getMintingInfo().then(() => { + resolve() + }).catch(err => { + if (expiredAt < Date.now()) { + reject(new Error(`DeFiDContainer docker not ready within given timeout of ${timeout}ms.\n${err.message as string}`)) + } + + setTimeout(() => void checkReady(), 200) + }) + } + + checkReady() + }) + } + + /** + * Wait for everything to be ready, override for additional hooks + */ + async waitForReady (timeout = 15000): Promise { + return await this.waitForRpc(timeout) + } + + /** + * tty into docker + */ + async exec (opts: { Cmd: string[] }): Promise { + return await new Promise((resolve, reject) => { + const container = this.requireContainer() + container.exec({ + Cmd: opts.Cmd + }, (error, exec) => { + if (error instanceof Error) { + reject(error) + } else { + exec?.start({}) + .then(() => resolve()) + .catch(reject) + } + }) + }) + } + + /** + * Stop and remove the current node. + * + * This method will also automatically stop and removes nodes that are stale. + * Stale nodes are nodes that are running for more than 1 hour + */ + async stop (): Promise { + try { + await this.container?.stop() + } finally { + try { + await this.container?.remove() + } finally { + await cleanUpStale(this.docker) + } + } + } +} + +/** + * RPC error from container + */ +export class DeFiDRpcError extends Error { + readonly payload: any + + constructor (payload: any) { + super('DeFiD RPC error from container') + this.payload = payload + } +} diff --git a/packages/testcontainers/src/chains/main_net_container.ts b/packages/testcontainers/src/chains/main_net_container.ts new file mode 100644 index 0000000000..440877dd93 --- /dev/null +++ b/packages/testcontainers/src/chains/main_net_container.ts @@ -0,0 +1,12 @@ +import { DockerOptions } from 'dockerode' +import { DeFiDContainer } from './container' + +export class MainNetContainer extends DeFiDContainer { + constructor (options?: DockerOptions) { + super('mainnet', options) + } + + async getRpcPort (): Promise { + return await this.getPort('8554/tcp') + } +} diff --git a/packages/testcontainers/src/chains/reg_test_container.ts b/packages/testcontainers/src/chains/reg_test_container.ts new file mode 100644 index 0000000000..d35c264c87 --- /dev/null +++ b/packages/testcontainers/src/chains/reg_test_container.ts @@ -0,0 +1,105 @@ +import { DockerOptions } from 'dockerode' +import { DeFiDContainer, StartOptions } from './container' +import { GenesisKeys, MasterNodeKey } from '../testkeys' + +export class RegTestContainer extends DeFiDContainer { + constructor (options?: DockerOptions) { + super('regtest', options) + } + + protected getCmd (opts: StartOptions): string[] { + return [...super.getCmd(opts), + '-regtest=1', + '-txnotokens=0', + '-logtimemicros', + '-amkheight=0', + '-bayfrontheight=1', + '-bayfrontgardensheight=2', + '-clarkequayheight=3', + '-dakotaheight=4' + ] + } + + async getNewAddress (label: string = '', addressType: 'legacy' | 'p2sh-segwit' | 'bech32' | string = 'bech32'): Promise { + return await this.call('getnewaddress', [label, addressType]) + } + + async getRpcPort (): Promise { + return await this.getPort('19554/tcp') + } +} + +/** + * RegTest with MasterNode preconfigured + */ +export class MasterNodeRegTestContainer extends RegTestContainer { + private readonly masternodeKey: MasterNodeKey + + constructor (masternodeKey: MasterNodeKey = GenesisKeys[0], options?: DockerOptions) { + super(options) + this.masternodeKey = masternodeKey + } + + /** + * Additional debug options turned on for traceability. + */ + protected getCmd (opts: StartOptions): string[] { + return [ + ...super.getCmd(opts), + '-dummypos=1', + '-nospv' + ] + } + + /** + * It is set to auto mint every 1 second by default in regtest. + * https://github.com/DeFiCh/ain/blob/6dc990c45788d6806ea/test/functional/test_framework/test_node.py#L160-L178 + */ + async generate (nblocks: number, address: string = this.masternodeKey.operator.address, maxTries: number = 1000000): Promise { + const mintedHashes: string[] = [] + + for (let minted = 0, tries = 0; minted < nblocks && tries < maxTries; tries++) { + const result = await this.call('generatetoaddress', [1, address, 1]) + + if (result === 1) { + minted += 1 + const count = await this.call('getblockcount') + const hash = await this.call('getblockhash', [count]) + mintedHashes.push(hash) + } + } + + return mintedHashes + } + + /** + * This will automatically import the necessary private key for master to mint tokens + */ + async waitForReady (timeout: number = 15000): Promise { + await super.waitForReady(timeout) + + // import keys for master node + await this.call('importprivkey', [ + this.masternodeKey.operator.privKey, 'coinbase', true + ]) + await this.call('importprivkey', [ + this.masternodeKey.owner.privKey, 'coinbase', true + ]) + + // configure the masternode + const fileContents = + 'gen=1' + '\n' + + 'spv=1' + '\n' + + `masternode_operator=${this.masternodeKey.operator.address}` + '\n' + + `masternode_owner=${this.masternodeKey.owner.address}` + + await this.exec({ + Cmd: ['bash', '-c', `echo "${fileContents}" > ~/.defi/defi.conf`] + }) + + // restart and wait for ready + await this.container?.stop() + await this.container?.start() + await super.waitForReady(timeout) + } +} diff --git a/packages/testcontainers/src/chains/test_net_container.ts b/packages/testcontainers/src/chains/test_net_container.ts new file mode 100644 index 0000000000..fdf12b2309 --- /dev/null +++ b/packages/testcontainers/src/chains/test_net_container.ts @@ -0,0 +1,16 @@ +import { DockerOptions } from 'dockerode' +import { DeFiDContainer, StartOptions } from './container' + +export class TestNetContainer extends DeFiDContainer { + constructor (options?: DockerOptions) { + super('testnet', options) + } + + protected getCmd (opts: StartOptions): string[] { + return [...super.getCmd(opts), '-testnet=1'] + } + + async getRpcPort (): Promise { + return await this.getPort('18554/tcp') + } +} diff --git a/packages/testcontainers/src/testcontainers.ts b/packages/testcontainers/src/testcontainers.ts index d76bda4325..d7f62b7d95 100644 --- a/packages/testcontainers/src/testcontainers.ts +++ b/packages/testcontainers/src/testcontainers.ts @@ -1,3 +1,12 @@ -export function getName (): string { - return 'testcontainers' -} +export { DockerOptions } from 'dockerode' + +export { StartOptions, DeFiDContainer, DeFiDRpcError } from './chains/container' +export { MainNetContainer } from './chains/main_net_container' +export { TestNetContainer } from './chains/test_net_container' +export { + RegTestContainer, MasterNodeRegTestContainer +} from './chains/reg_test_container' + +export { + KeyPair, MasterNodeKey, GenesisKeys +} from './testkeys' diff --git a/packages/testcontainers/src/testkeys.ts b/packages/testcontainers/src/testkeys.ts new file mode 100644 index 0000000000..a8fcdbfb7a --- /dev/null +++ b/packages/testcontainers/src/testkeys.ts @@ -0,0 +1,99 @@ +export interface KeyPair { + address: string + privKey: string +} + +export interface MasterNodeKey { + owner: KeyPair + operator: KeyPair +} + +/** + * As per: + * https://github.com/DeFiCh/ain/blob/6dc990c45788d6806ea/src/chainparams.cpp#L664-L677 + * https://github.com/DeFiCh/ain/blob/6dc990c45788d6806ea/test/functional/test_framework/test_node.py#L121-L132 + * + * 2 first and 2 last of genesis MNs acts as foundation members + */ +export const GenesisKeys: MasterNodeKey[] = [ + { + owner: { + address: 'mwsZw8nF7pKxWH8eoKL9tPxTpaFkz7QeLU', + privKey: 'cRiRQ9cHmy5evDqNDdEV8f6zfbK6epi9Fpz4CRZsmLEmkwy54dWz' + }, + operator: { + address: 'mswsMVsyGMj1FzDMbbxw2QW3KvQAv2FKiy', + privKey: 'cPGEaz8AGiM71NGMRybbCqFNRcuUhg3uGvyY4TFE1BZC26EW2PkC' + } + }, + { + owner: { + address: 'msER9bmJjyEemRpQoS8YYVL21VyZZrSgQ7', + privKey: 'cSCmN1tjcR2yR1eaQo9WmjTMR85SjEoNPqMPWGAApQiTLJH8JF7W' + }, + operator: { + address: 'mps7BdmwEF2vQ9DREDyNPibqsuSRZ8LuwQ', + privKey: 'cVNTRYV43guugJoDgaiPZESvNtnfnUW19YEjhybihwDbLKjyrZNV' + } + }, + { + owner: { + address: 'myF3aHuxtEuqqTw44EurtVs6mjyc1QnGUS', + privKey: 'cSXiqwTiYzECugcvCT4PyPKz2yKaTST8HowFVBBjccZCPkX6wsE9' + }, + operator: { + address: 'mtbWisYQmw9wcaecvmExeuixG7rYGqKEU4', + privKey: 'cPh5YaousYQ92tNd9FkiiS26THjSVBDHUMHZzUiBFbtGNS4Uw9AD' + } + }, + { + owner: { + address: 'mwyaBGGE7ka58F7aavH5hjMVdJENP9ZEVz', + privKey: 'cVA52y8ABsUYNuXVJ17d44N1wuSmeyPtke9urw4LchTyKsaGDMbY' + }, + operator: { + address: 'n1n6Z5Zdoku4oUnrXeQ2feLz3t7jmVLG9t', + privKey: 'cV9tJBgAnSfFmPaC6fWWvA9StLKkU3DKV7eXJHjWMUENQ8cKJDkL' + } + }, + { + owner: { + address: 'mgsE1SqrcfUhvuYuRjqy6rQCKmcCVKNhMu', + privKey: 'cRJyBuQPuUhYzN5F2Uf35958oK9AzZ5UscRfVmaRr8ktWq6Ac23u' + }, + operator: { + address: 'mzqdipBJcKX9rXXxcxw2kTHC3Xjzd3siKg', + privKey: 'cQYJ87qk39i3uFsXBZ2EkwdX1h72q1RQcX9V8X7PPydFPgujxrCy' + } + }, + { + owner: { + address: 'mud4VMfbBqXNpbt8ur33KHKx8pk3npSq8c', + privKey: 'cPjeCNka7omVbKKfywPVQyBig9eopBHy6eJqLzrdJqMP4DXApkcb' + }, + operator: { + address: 'mk5DkY4qcV6CUpuxDVyD3AHzRq5XK9kbRN', + privKey: 'cV6Hjhutf11RvFHaERkp52QNynm2ifNmtUfP8EwRRMg6NaaQsHTe' + } + }, + { + owner: { + address: 'bcrt1qyrfrpadwgw7p5eh3e9h3jmu4kwlz4prx73cqny', + privKey: 'cR4qgUdPhANDVF3bprcp5N9PNW2zyogDx6DGu2wHh2qtJB1L1vQj' + }, + operator: { + address: 'bcrt1qmfvw3dp3u6fdvqkdc0y3lr0e596le9cf22vtsv', + privKey: 'cVsa2wQvCjZZ54jGteQ8qiQbQLJQmZSBWriYUYyXbcaqUJFqK5HR' + } + }, + { + owner: { + address: 'bcrt1qyeuu9rvq8a67j86pzvh5897afdmdjpyankp4mu', + privKey: 'cUX8AEUZYsZxNUh5fTS7ZGnF6SPQuTeTDTABGrp5dbPftCga2zcp' + }, + operator: { + address: 'bcrt1qurwyhta75n2g75u2u5nds9p6w9v62y8wr40d2r', + privKey: 'cUp5EVEjuAGpemSuejP36TWWuFKzuCbUJ4QAKJTiSSB2vXzDLsJW' + } + } +]