-
Notifications
You must be signed in to change notification settings - Fork 16
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
93 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
import yargs from "yargs"; | ||
import { ApiPromise } from '@polkadot/api'; | ||
import { getApiFor, NETWORK_YARGS_OPTIONS } from "../utils/networks.ts"; | ||
|
||
|
||
const argv = yargs(process.argv.slice(2)) | ||
.usage("Usage: $0") | ||
.version("1.0.0") | ||
.options({ | ||
...NETWORK_YARGS_OPTIONS, | ||
start: { | ||
type: "number", | ||
description: "Block number to start", | ||
default: 0 | ||
}, | ||
end: { | ||
type: "number", | ||
description: "Block number to end", | ||
default: 10000000 | ||
}, | ||
key: { | ||
type: "string", | ||
description: "Storage key to look for", | ||
required: true | ||
}, | ||
}).argv; | ||
|
||
async function dichotomicSearch( | ||
api: ApiPromise, | ||
startBlock: number, | ||
endBlock: number, | ||
storageKey: string, | ||
): Promise<number | null> { | ||
let left = startBlock; | ||
let right = endBlock; | ||
|
||
// Get the value of the storage key at a specific block | ||
async function getStorageValueAtBlock(blockNumber: number): Promise<any> { | ||
const blockHash = await api.rpc.chain.getBlockHash(blockNumber); | ||
const storageValue = (await api.rpc.state.getStorage(storageKey, blockHash.toString())) as any; | ||
console.log(`[${blockNumber}: ${blockHash.toString()}] value: ${storageValue.toHex()} `); | ||
return storageValue.toHex(); | ||
} | ||
|
||
const initialValue = await getStorageValueAtBlock(startBlock); | ||
console.log(`Initial value: ${initialValue}`); | ||
console.log(`Starting dichotomic search from block ${startBlock} to ${endBlock}`); | ||
|
||
// Perform binary search | ||
while (left <= right) { | ||
const mid = Math.floor((left + right) / 2); | ||
const midValue = await getStorageValueAtBlock(mid); | ||
|
||
if (midValue !== initialValue) { | ||
// If the value changed, narrow the search range to the left | ||
right = mid - 1; | ||
} else { | ||
// If the value is the same, narrow the search range to the right | ||
left = mid + 1; | ||
} | ||
} | ||
|
||
// Check if the change was found | ||
if (left <= endBlock) { | ||
return left; | ||
} | ||
|
||
return null; // No change found within the range | ||
} | ||
|
||
async function main() { | ||
const api = await getApiFor(argv); | ||
|
||
const startBlock = argv.start; | ||
const endBlock = Math.min(argv.end, (await api.rpc.chain.getHeader()).number.toNumber()); | ||
const storageKey = argv.key | ||
|
||
console.log("===============================================") | ||
console.log("Warning, this will only work if a single change of value happened during the range") | ||
console.log("===============================================") | ||
|
||
const result = await dichotomicSearch(api, startBlock, endBlock, storageKey); | ||
|
||
if (result !== null) { | ||
console.log(`Change detected at block: ${result}`); | ||
} else { | ||
console.log('No change detected in the specified range.'); | ||
} | ||
|
||
await api.disconnect(); | ||
} | ||
|
||
main().catch(console.error); |