From 0ae9189ca60e86f7b12994bcc89bc196871d0e7c Mon Sep 17 00:00:00 2001 From: alvarius Date: Wed, 8 May 2024 13:56:17 +0100 Subject: [PATCH] fix(cli): function selector lookup during deploy (#2800) --- .changeset/blue-countries-drive.md | 8 +++ packages/cli/src/deploy/getFunctions.ts | 68 ++++++++++++++----------- 2 files changed, 45 insertions(+), 31 deletions(-) create mode 100644 .changeset/blue-countries-drive.md diff --git a/.changeset/blue-countries-drive.md b/.changeset/blue-countries-drive.md new file mode 100644 index 0000000000..4061d2d9ce --- /dev/null +++ b/.changeset/blue-countries-drive.md @@ -0,0 +1,8 @@ +--- +"@latticexyz/cli": patch +--- + +The deploy CLI now uses logs to find registered function selectors and their corresponding function signatures. +Previously only function signatures were fetched via logs and then mapped to function selectors via `getRecord` calls, +but this approach failed for namespaced function selectors of non-root system, +because the function signature table includes both the namespaced and non-namespaced signature but the function selector table only includes the namespaced selector that is registered on the world. diff --git a/packages/cli/src/deploy/getFunctions.ts b/packages/cli/src/deploy/getFunctions.ts index 3ee7b0462e..5508b49480 100644 --- a/packages/cli/src/deploy/getFunctions.ts +++ b/packages/cli/src/deploy/getFunctions.ts @@ -1,11 +1,9 @@ -import { Client, toFunctionSelector, parseAbiItem } from "viem"; +import { Client, parseAbiItem } from "viem"; import { WorldDeploy, WorldFunction, worldTables } from "./common"; import { debug } from "./debug"; import { storeSetRecordEvent } from "@latticexyz/store"; import { getLogs } from "viem/actions"; -import { decodeValueArgs } from "@latticexyz/protocol-parser/internal"; -import { getTableValue } from "./getTableValue"; -import { hexToResource } from "@latticexyz/common"; +import { decodeKey, decodeValueArgs } from "@latticexyz/protocol-parser/internal"; export async function getFunctions({ client, @@ -15,44 +13,52 @@ export async function getFunctions({ readonly worldDeploy: WorldDeploy; }): Promise { // This assumes we only use `FunctionSelectors._set(...)`, which is true as of this writing. - debug("looking up function signatures for", worldDeploy.address); - const logs = await getLogs(client, { + debug("looking up function selectors for", worldDeploy.address); + const selectorLogs = await getLogs(client, { strict: true, fromBlock: worldDeploy.deployBlock, toBlock: worldDeploy.stateBlock, address: worldDeploy.address, event: parseAbiItem(storeSetRecordEvent), - args: { tableId: worldTables.world_FunctionSignatures.tableId }, + args: { tableId: worldTables.world_FunctionSelectors.tableId }, + }); + + const selectors = selectorLogs.map((log) => { + return { + ...decodeValueArgs(worldTables.world_FunctionSelectors.valueSchema, log.args), + ...decodeKey(worldTables.world_FunctionSelectors.keySchema, log.args.keyTuple), + }; }); + debug("found", selectors.length, "function selectors for", worldDeploy.address); - const signatures = logs.map((log) => { - const value = decodeValueArgs(worldTables.world_FunctionSignatures.valueSchema, log.args); - return value.functionSignature; + // This assumes we only use `FunctionSignatures._set(...)`, which is true as of this writing. + debug("looking up function signatures for", worldDeploy.address); + const signatureLogs = await getLogs(client, { + strict: true, + fromBlock: worldDeploy.deployBlock, + toBlock: worldDeploy.stateBlock, + address: worldDeploy.address, + event: parseAbiItem(storeSetRecordEvent), + args: { tableId: worldTables.world_FunctionSignatures.tableId }, }); - debug("found", signatures.length, "function signatures for", worldDeploy.address); - // TODO: parallelize with a bulk getRecords - const functions = await Promise.all( - signatures.map(async (signature) => { - const selector = toFunctionSelector(signature); - const { systemId, systemFunctionSelector } = await getTableValue({ - client, - worldDeploy, - table: worldTables.world_FunctionSelectors, - key: { worldFunctionSelector: selector }, - }); - const { namespace, name } = hexToResource(systemId); - // TODO: find away around undoing contract logic (https://github.com/latticexyz/mud/issues/1708) - const systemFunctionSignature = namespace === "" ? signature : signature.replace(`${namespace}_${name}_`, ""); - return { - signature, - selector, - systemId, - systemFunctionSignature, - systemFunctionSelector, - }; + const selectorToSignature = Object.fromEntries( + signatureLogs.map((log) => { + return [ + decodeKey(worldTables.world_FunctionSignatures.keySchema, log.args.keyTuple).functionSelector, + decodeValueArgs(worldTables.world_FunctionSignatures.valueSchema, log.args).functionSignature, + ]; }), ); + debug("found", signatureLogs.length, "function signatures for", worldDeploy.address); + + const functions = selectors.map(({ worldFunctionSelector, systemFunctionSelector, systemId }) => ({ + selector: worldFunctionSelector, + signature: selectorToSignature[worldFunctionSelector], + systemFunctionSelector, + systemFunctionSignature: selectorToSignature[systemFunctionSelector], + systemId, + })); return functions; }