Skip to content

Commit

Permalink
add migration scripts
Browse files Browse the repository at this point in the history
  • Loading branch information
Maksandre committed Jun 9, 2023
1 parent 9b5209a commit 078a748
Show file tree
Hide file tree
Showing 9 changed files with 490 additions and 0 deletions.
12 changes: 12 additions & 0 deletions .github/workflows/forkless-update-data.yml
Original file line number Diff line number Diff line change
Expand Up @@ -251,6 +251,18 @@ jobs:
if: success() || failure()
run: cat './forkless-parachain-upgrade-data-logs.${{ matrix.network }}/forkless-data.log'

- name: Run AppPromo migration
if: success() || failure()
working-directory: tests
run: |
yarn install
cd src/migrations/942057-appPromotion
/home/ubuntu/.cargo/bin/chainql --tla-str=chainUrl=ws://127.0.0.1:9944 stakersParser.jsonnet > output.json
npx ts-node --esm lockedToFreeze.ts
env:
WS_RPC: ws://127.0.0.1:9944
SUPERUSER_SEED: //Alice

- name: Stop running containers
if: always() # run this step always
run: docker-compose -f ".docker/docker-compose.forkless-data.${{ matrix.network }}.yml" down --volumes
Expand Down
1 change: 1 addition & 0 deletions tests/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,7 @@
"chai-like": "^1.1.1",
"csv-writer": "^1.6.0",
"find-process": "^1.4.7",
"lossless-json": "^2.0.9",
"solc": "0.8.17",
"web3": "1.10.0"
},
Expand Down
25 changes: 25 additions & 0 deletions tests/src/migrations/942057-appPromotion/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
## Stakers Data Loading

Set the environment variable (WS_RPC). For example, ws://localhost:9944. Execute the following command:

chainql --tla-str=chainUrl=<WS_RPC> stakersParser.jsonnet > output.json

where <WS_RPC> - is the network address.

Example for Opal:
:

chainql --tla-str=chainUrl=wss://eu-ws-opal.unique.network:443 stakersParser.jsonnet > output.json

To install chainql, execute the following command:


cargo install chainql

## Execute offchain migration

To run, you need to add an environment variable (SUPERUSER_SEED) with the sudo key seed.

Run the script by executing the following command:

npx ts-node lockedToFreeze.ts
74 changes: 74 additions & 0 deletions tests/src/migrations/942057-appPromotion/afterMaintenance.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
// Copyright 2019-2022 Unique Network (Gibraltar) Ltd.
// This file is part of Unique Network.

// Unique Network is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// Unique Network is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.

// You should have received a copy of the GNU General Public License
// along with Unique Network. If not, see <http://www.gnu.org/licenses/>.

import {IKeyringPair} from '@polkadot/types/types';
import {ApiPromise} from '@polkadot/api';
import {expect, itSub, Pallets, requirePalletsOrSkip, usingPlaygrounds} from '../../util';
import {main as testedScript} from './correctStateAfterMaintenance';

async function maintenanceEnabled(api: ApiPromise): Promise<boolean> {
return (await api.query.maintenance.enabled()).toJSON() as boolean;
}



describe('Integration Test: Maintenance mode & App Promo', () => {
let superuser: IKeyringPair;

before(async function() {
await usingPlaygrounds(async (helper, privateKey) => {
requirePalletsOrSkip(this, helper, [Pallets.Maintenance]);
superuser = await privateKey('//Alice');
});
});

describe('Test AppPromo script for check state after Maintenance mode', () => {
before(async function () {
await usingPlaygrounds(async (helper) => {
if(await maintenanceEnabled(helper.getApi())) {
console.warn('\tMaintenance mode was left enabled BEFORE the test suite! Disabling it now.');
await expect(helper.getSudo().executeExtrinsic(superuser, 'api.tx.maintenance.disable', [])).to.be.fulfilled;
}
});
});
itSub('Can find and fix inconsistent state', async({helper}) => {
const api = helper.getApi();

await helper.executeExtrinsic(superuser, 'api.tx.sudo.sudo', [api.tx.system.setStorage([
['0x42b67acb8bd223c60d0c8f621ffefc0ae280fa2db99bd3827aac976de75af95f5153cb1f00942ff401000000',
'0x04d43593c715fdd31c61141abd04a99fd6822c8558854ccde39a5684e7a56da27d000010632d5ec76b0500000000000000'],
['0x42b67acb8bd223c60d0c8f621ffefc0ae280fa2db99bd3827aac976de75af95f9eb2dcce60f37a2702000000',
'0x04d43593c715fdd31c61141abd04a99fd6822c8558854ccde39a5684e7a56da27d000010632d5ec76b0500000000000000'],
['0xc2261276cc9d1f8598ea4b6a74b15c2fb1c0eb12e038e5c7f91e120ed4b7ebf1de1e86a9a8c739864cf3cc5ec2bea59fd43593c715fdd31c61141abd04a99fd6822c8558854ccde39a5684e7a56da27d',
'0x046170707374616b656170707374616b65000020c65abc8ed70a00000000000000'],
])]);

expect((await api.query.appPromotion.pendingUnstake(1)).toJSON()).to.be.deep.equal([['5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY', '0x00000000000000056bc75e2d63100000']]);
expect((await api.query.appPromotion.pendingUnstake(2)).toJSON()).to.be.deep.equal([['5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY', '0x00000000000000056bc75e2d63100000']]);
await testedScript();

expect((await api.query.appPromotion.pendingUnstake(1)).toJSON()).to.be.deep.equal([]);
expect((await api.query.appPromotion.pendingUnstake(2)).toJSON()).to.be.deep.equal([]);

});

itSub('(!negative test!) Only works when Maintenance mode is disabled', async({helper}) => {
await expect(helper.getSudo().executeExtrinsic(superuser, 'api.tx.maintenance.enable', [])).to.be.fulfilled;
await expect(testedScript()).to.be.rejectedWith('The network is still in maintenance mode');
await expect(helper.getSudo().executeExtrinsic(superuser, 'api.tx.maintenance.disable', [])).to.be.fulfilled;
});
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
import {usingPlaygrounds} from '../../util';



const WS_ENDPOINT = 'ws://localhost:9944';
const DONOR_SEED = '//Alice';

export const main = async(options: { wsEndpoint: string; donorSeed: string } = {
wsEndpoint: WS_ENDPOINT,
donorSeed: DONOR_SEED,
}) => {
await usingPlaygrounds(async (helper, privateKey) => {
const api = helper.getApi();

if((await api.query.maintenance.enabled()).valueOf()) {
throw Error('The network is still in maintenance mode');
}

const pendingBlocks = (
await api.query.appPromotion.pendingUnstake.entries()
).map(([k, _v]) =>
k.args[0]);

const currentBlock = await api.query.system.number();

const filteredBlocks = pendingBlocks.filter((b) => currentBlock.gt(b));

if(filteredBlocks.length != 0) {
console.log(
'During maintenance mode, %d block(s) were not processed',
filteredBlocks.length,
);
} else {
console.log('Nothing to change');
return;
}

const skippedBlocks = chunk(filteredBlocks, 10);

const signer = await privateKey(options.donorSeed);

const txs = skippedBlocks.map((b) =>
api.tx.sudo.sudo(api.tx.appPromotion.forceUnstake(b)));


const promises = txs.map((tx) => () => helper.signTransaction(signer, tx));

await Promise.allSettled(promises.map((p) => p()));

const failedBlocks: bigint[] = [];
let isSuccess = true;

for(const b of filteredBlocks) {
if(((await api.query.appPromotion.pendingUnstake(b)).toJSON() as any[]).length != 0) {
failedBlocks.push(b.toBigInt());
isSuccess = false;
}
}

if(isSuccess) {
console.log('Done. %d block(s) were processed.', filteredBlocks.length);
} else {
throw new Error(`Something went wrong. Block(s) have not been processed: ${failedBlocks}`);
}


}, options.wsEndpoint);
};

const chunk = <T>(arr: T[], size: number) =>
Array.from({length: Math.ceil(arr.length / size)}, (_: any, i: number) =>
arr.slice(i * size, i * size + size));
Loading

0 comments on commit 078a748

Please sign in to comment.