From 7b46e55621a439f64b8a99f4c721d951b4586dc6 Mon Sep 17 00:00:00 2001 From: Mathieu Hofman <86499+mhofman@users.noreply.github.com> Date: Wed, 15 Mar 2023 20:23:19 -0300 Subject: [PATCH] feat(runner): Enable starting testnet follower from state-sync (#105) --- runner/lib/main.js | 6 ++++ runner/lib/stats/types.d.ts | 1 + runner/lib/tasks/testnet.js | 58 +++++++++++++++++++++++++++++++------ 3 files changed, 56 insertions(+), 9 deletions(-) diff --git a/runner/lib/main.js b/runner/lib/main.js index 6f7df5e..9ce7548 100644 --- a/runner/lib/main.js +++ b/runner/lib/main.js @@ -279,6 +279,9 @@ const main = async (progName, rawArgs, powers) => { /** @type {string} */ let testnetOrigin; + /** @type {boolean | undefined} */ + let useStateSync; + const profile = argv.profile == null ? 'local' : argv.profile; switch (profile) { @@ -291,6 +294,7 @@ const main = async (progName, rawArgs, powers) => { case 'stage': makeTasks = makeTestnetTasks; testnetOrigin = argv.testnetOrigin || `https://${profile}.agoric.net`; + useStateSync = argv.useStateSync; break; default: throw new Error(`Unexpected profile option: ${profile}`); @@ -382,6 +386,7 @@ const main = async (progName, rawArgs, powers) => { metadata: { profile, testnetOrigin, + useStateSync, ...envInfo, testData: argv.testData, }, @@ -961,6 +966,7 @@ const main = async (progName, rawArgs, powers) => { chainOnly: globalChainOnly, withMonitor, testnetOrigin, + useStateSync, }; logPerfEvent('setup-tasks-start', setupConfig); ({ chainStorageLocation, clientStorageLocation } = diff --git a/runner/lib/stats/types.d.ts b/runner/lib/stats/types.d.ts index db92f04..9469aeb 100644 --- a/runner/lib/stats/types.d.ts +++ b/runner/lib/stats/types.d.ts @@ -161,6 +161,7 @@ export interface StageStats extends StageStatsInitData { export interface RunMetadata { readonly profile: string; readonly testnetOrigin?: string; + readonly useStateSync?: boolean | undefined; readonly agChainCosmosVersion?: unknown; readonly testData?: unknown; } diff --git a/runner/lib/tasks/testnet.js b/runner/lib/tasks/testnet.js index e27a2d1..ee08f99 100644 --- a/runner/lib/tasks/testnet.js +++ b/runner/lib/tasks/testnet.js @@ -124,7 +124,7 @@ export const makeTasks = ({ /** @type {Record} */ const additionChainEnv = {}; - /** @param {import("./types.js").TaskBaseOptions & {config?: {reset?: boolean, chainOnly?: boolean, withMonitor?: boolean, testnetOrigin?: string}}} options */ + /** @param {import("./types.js").TaskBaseOptions & {config?: {reset?: boolean, chainOnly?: boolean, withMonitor?: boolean, testnetOrigin?: string, useStateSync?: boolean}}} options */ const setupTasks = async ({ stdout, stderr, @@ -135,6 +135,7 @@ export const makeTasks = ({ chainOnly, withMonitor = true, testnetOrigin: testnetOriginOption, + useStateSync, } = {}, }) => { const { console, stdio } = getConsoleAndStdio( @@ -182,9 +183,6 @@ export const makeTasks = ({ const genesisPath = joinPath(chainStateDir, 'config', 'genesis.json'); if (!(await fsExists(genesisPath))) { - console.log('Fetching genesis'); - const genesis = await fetchAsJSON(`${testnetOrigin}/genesis.json`); - await childProcessDone( printerSpawn( sdkBinaries.cosmosChain, @@ -193,11 +191,6 @@ export const makeTasks = ({ ), ); - fs.writeFile( - joinPath(chainStateDir, 'config', 'genesis.json'), - JSON.stringify(genesis), - ); - await childProcessDone( printerSpawn( sdkBinaries.cosmosChain, @@ -227,6 +220,53 @@ export const makeTasks = ({ configP2p.seeds = seeds.join(','); configP2p.addr_book_strict = false; delete config.log_level; + + if (!useStateSync) { + console.log('Fetching genesis'); + const genesis = await fetchAsJSON(`${testnetOrigin}/genesis.json`); + + fs.writeFile( + joinPath(chainStateDir, 'config', 'genesis.json'), + JSON.stringify(genesis), + ); + } else { + console.log('Fetching state-sync info'); + /** @type {any} */ + const currentBlockInfo = await fetchAsJSON( + `http://${rpcAddrs[0]}/block`, + ); + + // `trustHeight` is the block height considered as the "root of trust" + // for state-sync. The node will attempt to find a snapshot offered for + // a block at or after this height, and will validate that block's hash + // using a light client with the configured RPC servers. + // We want to use a block height recent enough, but for which a snapshot + // exists since then. + const stateSyncInterval = + Number(process.env.AG_SETUP_COSMOS_STATE_SYNC_INTERVAL) || 2000; + const trustHeight = Math.max( + 1, + Number(currentBlockInfo.result.block.header.height) - + stateSyncInterval, + ); + + /** @type {any} */ + const trustedBlockInfo = await fetchAsJSON( + `http://${rpcAddrs[0]}/block?height=${trustHeight}`, + ); + const trustHash = trustedBlockInfo.result.block_id.hash; + + const configStatesync = /** @type {import('@iarna/toml').JsonMap} */ ( + config.statesync + ); + configStatesync.enable = true; + configStatesync.rpc_servers = rpcAddrs + .map((host) => `http://${host}`) + .join(','); + configStatesync.trust_height = trustHeight; + configStatesync.trust_hash = trustHash; + } + await fs.writeFile(configPath, TOML.stringify(config)); }