diff --git a/packages/jellyfish-api-core/__tests__/category/evm/evmFeatureFlag.test.ts b/packages/jellyfish-api-core/__tests__/category/evm/evmFeatureFlag.test.ts new file mode 100644 index 0000000000..3354287b31 --- /dev/null +++ b/packages/jellyfish-api-core/__tests__/category/evm/evmFeatureFlag.test.ts @@ -0,0 +1,311 @@ +import { MasterNodeRegTestContainer } from '@defichain/testcontainers' +import { ContainerAdapterClient } from '../../container_adapter_client' +import { TransferDomainType } from '../../../src/category/account' +import BigNumber from 'bignumber.js' + +describe('EVM feature flags', () => { + let dvmAddr: string, evmAddr: string, evmAddrTo: string + const container = new MasterNodeRegTestContainer() + const client = new ContainerAdapterClient(container) + + async function dvmToEvmTransferDomain (): Promise { + const txid = await client.account.transferDomain([ + { + src: { + address: dvmAddr, + amount: '5@DFI', + domain: TransferDomainType.DVM + }, + dst: { + address: evmAddr, + amount: '5@DFI', + domain: TransferDomainType.EVM + } + } + ]) + return txid + } + + async function evmToDvmTransferDomain (): Promise { + const txid = await client.account.transferDomain([ + { + src: { + address: evmAddr, + amount: '2@DFI', + domain: TransferDomainType.EVM + + }, + dst: { + address: dvmAddr, + amount: '2@DFI', + domain: TransferDomainType.DVM + } + } + ]) + return txid + } + + async function createEvmTx (nonce: number): Promise { + const evmTxHash = await client.evm.evmtx({ + from: evmAddr, + to: evmAddrTo, + value: new BigNumber(0.05), + gasPrice: 23, + gasLimit: 23000, + nonce + }) + return evmTxHash + } + + beforeAll(async () => { + await container.start() + await container.waitForWalletCoinbaseMaturity() + + dvmAddr = await container.getNewAddress('dfiAddress', 'legacy') + evmAddr = await container.getNewAddress('eth', 'eth') + evmAddrTo = await container.getNewAddress('eth', 'eth') + + await container.call('utxostoaccount', [{ [dvmAddr]: '100@0' }]) + await container.generate(1) + + await container.call('createtoken', [{ + symbol: 'BTC', + name: 'BTC', + isDAT: true, + mintable: true, + tradeable: true, + collateralAddress: dvmAddr + }]) + await container.generate(1) + }) + + afterAll(async () => { + await container.stop() + }) + + describe('all gov attrs for EVM is set to false', () => { + it('should still successfully get eth address', async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'false', + 'v0/params/feature/transferdomain': 'false', + 'v0/transferdomain/dvm-evm/enabled': 'false', + 'v0/transferdomain/evm-dvm/enabled': 'false' + } + }) + await container.generate(1) + + const newEvmAddr = await container.getNewAddress('eth', 'eth') + expect(newEvmAddr).toBeDefined() + expect(newEvmAddr).toMatch(/^0x[a-fA-F0-9]{40}$/gm) + expect(newEvmAddr.length).toStrictEqual(42) + }) + + it('should still successfully get eth balance', async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'true', + 'v0/params/feature/transferdomain': 'true', + 'v0/transferdomain/dvm-evm/enabled': 'true', + 'v0/transferdomain/evm-dvm/enabled': 'true' + } + }) + await container.generate(1) + + await dvmToEvmTransferDomain() + await container.generate(1) + + const withoutDvmBalances = await client.account.getTokenBalances({}, false) + const [withoutDvmBalance] = withoutDvmBalances[0].split('@') + + // disable EVM and try to get ETH balance + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'false', + 'v0/params/feature/transferdomain': 'false', + 'v0/transferdomain/dvm-evm/enabled': 'false', + 'v0/transferdomain/evm-dvm/enabled': 'false' + } + }) + await container.generate(1) + + const withEvmBalances = await client.account.getTokenBalances({}, false, { includeEth: true }) + const [withEvmBalance] = withEvmBalances[0].split('@') + expect(withEvmBalance).toBeDefined() + expect(new BigNumber(withoutDvmBalance)) + .toStrictEqual(new BigNumber(withEvmBalance).minus(5)) + }) + }) + + describe('all gov attrs for EVM is set to true', () => { + beforeAll(async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'true', + 'v0/params/feature/transferdomain': 'true', + 'v0/transferdomain/dvm-evm/enabled': 'true', + 'v0/transferdomain/evm-dvm/enabled': 'true' + } + }) + await container.generate(1) + }) + + it('should successfully transfer domain from DVM to EVM', async () => { + const txid = await dvmToEvmTransferDomain() + await container.generate(1) + expect(typeof txid).toStrictEqual('string') + expect(txid.length).toStrictEqual(64) + }) + + it('should successfully transfer domain from EVM to DVM', async () => { + const txid = await evmToDvmTransferDomain() + expect(typeof txid).toStrictEqual('string') + expect(txid.length).toStrictEqual(64) + }) + + it('should successfully create a new EVM transaction (EvmTx)', async () => { + // send fund to source evm address + await dvmToEvmTransferDomain() + await container.generate(1) + + const evmTxHash = await createEvmTx(0) + await container.generate(1) + + const blockHash: string = await client.blockchain.getBestBlockHash() + const txs = await client.blockchain.getBlock(blockHash, 1) + expect(txs.tx[1]).toStrictEqual(evmTxHash) + }) + }) + + describe('gov attr v0/params/feature/evm is set to false', () => { + beforeAll(async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'false', + 'v0/params/feature/transferdomain': 'true' + } + }) + await container.generate(1) + }) + + it('should fail DVM to EVM transferDomain', async () => { + const promise = dvmToEvmTransferDomain() + await expect(promise).rejects.toThrow('Cannot create tx, EVM is not enabled') + }) + + it('should fail EVM to DVM transferDomain', async () => { + const promise = evmToDvmTransferDomain() + await expect(promise).rejects.toThrow('Cannot create tx, EVM is not enabled') + }) + + it('should fail creation of new EVM transaction (EvmTx)', async () => { + const promise = createEvmTx(1) + await expect(promise).rejects.toThrow('Cannot create tx, EVM is not enabled') + }) + }) + + describe('gov attr v0/params/feature/transferdomain is set to false', () => { + beforeAll(async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'true', + 'v0/params/feature/transferdomain': 'false' + } + }) + await container.generate(1) + }) + + it('should fail DVM to EVM transferDomain', async () => { + const promise = dvmToEvmTransferDomain() + await expect(promise).rejects.toThrow('Cannot create tx, transfer domain is not enabled') + }) + + it('should fail EVM to DVM transferDomain', async () => { + const promise = evmToDvmTransferDomain() + await expect(promise).rejects.toThrow('Cannot create tx, transfer domain is not enabled') + }) + + it('should still successfully create new EVM transaction (EvmTx)', async () => { + const evmTxHash = await createEvmTx(1) + await container.generate(1) + + const blockHash: string = await client.blockchain.getBestBlockHash() + const txs = await client.blockchain.getBlock(blockHash, 1) + expect(txs.tx[1]).toStrictEqual(evmTxHash) + }) + }) + + describe('gov attr v0/transferdomain/dvm-evm/enabled is set to false', () => { + beforeAll(async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'true', + 'v0/params/feature/transferdomain': 'true', + 'v0/transferdomain/evm-dvm/enabled': 'true', + 'v0/transferdomain/dvm-evm/enabled': 'false' // disabled dvm-evm + } + }) + await container.generate(1) + }) + + it('should not allow DVM to EVM transferDomain', async () => { + const promise = dvmToEvmTransferDomain() + await expect(promise).rejects.toThrow('DVM to EVM is not currently enabled') + }) + + it('should allow EVM to DVM transferDomain', async () => { + const txid = await evmToDvmTransferDomain() + expect(typeof txid).toStrictEqual('string') + expect(txid.length).toStrictEqual(64) + }) + }) + + describe('gov attr v0/transferdomain/evm-dvm/enabled is set to false', () => { + beforeAll(async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'true', + 'v0/params/feature/transferdomain': 'true', + 'v0/transferdomain/dvm-evm/enabled': 'true', + 'v0/transferdomain/evm-dvm/enabled': 'false' // disabled evm-dvm + } + }) + await container.generate(1) + }) + + it('should not allow EVM to DVM transferDomain', async () => { + const promise = evmToDvmTransferDomain() + await expect(promise).rejects.toThrow('EVM to DVM is not currently enabled') + }) + + it('should allow DVM to EVM transferDomain', async () => { + const txid = await dvmToEvmTransferDomain() + expect(typeof txid).toStrictEqual('string') + expect(txid.length).toStrictEqual(64) + }) + }) + + describe('gov attrs v0/transferdomain/dvm-evm/enabled and v0/transferdomain/evm-dvm/enabled are both set to false', () => { + beforeAll(async () => { + await client.masternode.setGov({ + ATTRIBUTES: { + 'v0/params/feature/evm': 'true', + 'v0/params/feature/transferdomain': 'true', + 'v0/transferdomain/dvm-evm/enabled': 'false', // disabled dvm-evm + 'v0/transferdomain/evm-dvm/enabled': 'false' // disabled evm-dvm + } + }) + await container.generate(1) + }) + + it('should not allow EVM to DVM transferDomain', async () => { + const promise = evmToDvmTransferDomain() + await expect(promise).rejects.toThrow('EVM to DVM is not currently enabled') + }) + + it('should not allow DVM to EVM transferDomain', async () => { + const promise = dvmToEvmTransferDomain() + await expect(promise).rejects.toThrow('DVM to EVM is not currently enabled') + }) + }) +})