From 8dca0bde4709a9cbeab60ac88d0b544596615374 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 14:17:03 +0100 Subject: [PATCH 1/8] move logic to writeContract --- .../client-react/src/mud/setupNetwork.ts | 2 +- packages/common/src/createContract.ts | 95 +++----------- packages/common/src/createNonceManager.ts | 49 ++++---- packages/common/src/getNonceManager.ts | 21 ++++ packages/common/src/getNonceManagerId.ts | 15 +++ packages/common/src/writeContract.ts | 118 ++++++++++++++++++ 6 files changed, 193 insertions(+), 107 deletions(-) create mode 100644 packages/common/src/getNonceManager.ts create mode 100644 packages/common/src/getNonceManagerId.ts create mode 100644 packages/common/src/writeContract.ts diff --git a/examples/minimal/packages/client-react/src/mud/setupNetwork.ts b/examples/minimal/packages/client-react/src/mud/setupNetwork.ts index 895bf70553..04876200ac 100644 --- a/examples/minimal/packages/client-react/src/mud/setupNetwork.ts +++ b/examples/minimal/packages/client-react/src/mud/setupNetwork.ts @@ -16,7 +16,7 @@ export async function setupNetwork() { const clientOptions = { chain: networkConfig.chain, - transport: transportObserver(fallback([webSocket(), http()])), + transport: transportObserver(http()), pollingInterval: 1000, } as const satisfies ClientConfig; diff --git a/packages/common/src/createContract.ts b/packages/common/src/createContract.ts index b88924fc19..03406446cc 100644 --- a/packages/common/src/createContract.ts +++ b/packages/common/src/createContract.ts @@ -7,18 +7,13 @@ import { GetContractReturnType, Hex, PublicClient, - SimulateContractParameters, Transport, WalletClient, WriteContractParameters, getContract, } from "viem"; -import pRetry from "p-retry"; -import { createNonceManager } from "./createNonceManager"; -import { debug as parentDebug } from "./debug"; import { UnionOmit } from "./type-utils/common"; - -const debug = parentDebug.extend("createContract"); +import { WriteContractOptions, writeContract } from "./writeContract"; // copied from viem because this isn't exported // TODO: import from viem? @@ -81,88 +76,26 @@ export function createContract< }) as unknown as GetContractReturnType; if (contract.write) { - let nextWriteId = 0; - const nonceManager = createNonceManager({ - publicClient: publicClient as PublicClient, - address: walletClient.account.address, - }); - - // Replace write calls with our own proxy. Implemented ~the same as viem, but adds better handling of nonces (via queue + retries). + // Replace write calls with our own. Implemented ~the same as viem, but adds better handling of nonces (via queue + retries). contract.write = new Proxy( {}, { - get(_, functionName: string): GetContractReturnType["write"][string] { - async function prepareWrite( - options: WriteContractParameters - ): Promise> { - if (options.gas) { - debug("gas provided, skipping simulate", functionName, options); - return options as unknown as WriteContractParameters; - } - - debug("simulating write", functionName, options); - const { request } = await publicClient.simulateContract({ - ...options, - account: options.account ?? walletClient.account, - } as unknown as SimulateContractParameters); - - return request as unknown as WriteContractParameters; - } - - async function write(options: WriteContractParameters): Promise { - const preparedWrite = await prepareWrite(options); - - return await pRetry( - async () => { - if (!nonceManager.hasNonce()) { - await nonceManager.resetNonce(); - } - - const nonce = nonceManager.nextNonce(); - debug("calling write function with nonce", nonce, preparedWrite); - return await walletClient.writeContract({ - nonce, - ...preparedWrite, - }); - }, - { - retries: 3, - onFailedAttempt: async (error) => { - // On nonce errors, reset the nonce and retry - if (nonceManager.shouldResetNonce(error)) { - debug("got nonce error, retrying", error); - await nonceManager.resetNonce(); - return; - } - // TODO: prepareWrite again if there are gas errors? - throw error; - }, - } - ); - } - - return (...parameters) => { - const id = `${walletClient.chain.id}:${walletClient.account.address}:${nextWriteId++}`; - const { args, options } = < - { - args: unknown[]; - options: UnionOmit; - } - >getFunctionParameters(parameters as any); - - const request = { - address, + get(_, functionName: string) { + return ( + ...parameters: [ + args?: readonly unknown[], + options?: UnionOmit + ] + ) => { + const { args, options } = getFunctionParameters(parameters); + return writeContract(walletClient, { abi, + address, functionName, args, ...options, - }; - - const result = write(request); - - onWrite?.({ id, request, result }); - - return result; + onWrite, + } as unknown as WriteContractOptions); }; }, } diff --git a/packages/common/src/createNonceManager.ts b/packages/common/src/createNonceManager.ts index f9d3b8222c..e134726143 100644 --- a/packages/common/src/createNonceManager.ts +++ b/packages/common/src/createNonceManager.ts @@ -1,23 +1,18 @@ -import { - BaseError, - BlockTag, - Hex, - NonceTooHighError, - NonceTooLowError, - PublicClient, - TransactionExecutionError, -} from "viem"; +import { BaseError, BlockTag, Client, Hex, NonceTooHighError, NonceTooLowError } from "viem"; import { debug as parentDebug } from "./debug"; +import { getNonceManagerId } from "./getNonceManagerId"; +import { getTransactionCount } from "viem/actions"; const debug = parentDebug.extend("createNonceManager"); -type CreateNonceManagerOptions = { - publicClient: PublicClient; +export type CreateNonceManagerOptions = { + client: Client; address: Hex; blockTag?: BlockTag; + broadcastChannelName?: string; }; -type CreateNonceManagerResult = { +export type CreateNonceManagerResult = { hasNonce: () => boolean; nextNonce: () => number; resetNonce: () => Promise; @@ -25,22 +20,26 @@ type CreateNonceManagerResult = { }; export function createNonceManager({ - publicClient, + client, address, - blockTag, + blockTag = "latest", + broadcastChannelName, }: CreateNonceManagerOptions): CreateNonceManagerResult { const nonceRef = { current: -1 }; - const channel = - typeof BroadcastChannel !== "undefined" - ? // TODO: fetch chain ID or require it via types? - new BroadcastChannel(`mud:createNonceManager:${publicClient.chain?.id}:${address}`) - : null; + let channel: BroadcastChannel | null = null; - if (channel) { - channel.addEventListener("message", (event) => { - const nonce = JSON.parse(event.data); - debug("got nonce from broadcast channel", nonce); - nonceRef.current = nonce; + if (typeof BroadcastChannel !== "undefined") { + const channelName = broadcastChannelName + ? Promise.resolve(broadcastChannelName) + : getNonceManagerId({ client, address, blockTag }); + channelName.then((name) => { + channel = new BroadcastChannel(name); + // TODO: emit some sort of "connected" event so other channels can broadcast current nonce + channel.addEventListener("message", (event) => { + const nonce = JSON.parse(event.data); + debug("got nonce from broadcast channel", nonce); + nonceRef.current = nonce; + }); }); } @@ -56,7 +55,7 @@ export function createNonceManager({ } async function resetNonce(): Promise { - const nonce = await publicClient.getTransactionCount({ address, blockTag }); + const nonce = await getTransactionCount(client, { address, blockTag }); nonceRef.current = nonce; channel?.postMessage(JSON.stringify(nonceRef.current)); debug("reset nonce to", nonceRef.current); diff --git a/packages/common/src/getNonceManager.ts b/packages/common/src/getNonceManager.ts new file mode 100644 index 0000000000..46b450910e --- /dev/null +++ b/packages/common/src/getNonceManager.ts @@ -0,0 +1,21 @@ +import { CreateNonceManagerOptions, CreateNonceManagerResult, createNonceManager } from "./createNonceManager"; +import { getNonceManagerId } from "./getNonceManagerId"; + +const nonceManagers = new Map(); + +export async function getNonceManager({ + client, + address, + blockTag = "latest", +}: CreateNonceManagerOptions): Promise { + const id = await getNonceManagerId({ client, address, blockTag }); + + const existingNonceManager = nonceManagers.get(id); + if (existingNonceManager) { + return existingNonceManager; + } + + const nonceManager = createNonceManager({ client, address, blockTag }); + nonceManagers.set(id, nonceManager); + return nonceManager; +} diff --git a/packages/common/src/getNonceManagerId.ts b/packages/common/src/getNonceManagerId.ts new file mode 100644 index 0000000000..85220aae08 --- /dev/null +++ b/packages/common/src/getNonceManagerId.ts @@ -0,0 +1,15 @@ +import { BlockTag, Client, Hex, getAddress } from "viem"; +import { getChainId } from "viem/actions"; + +export async function getNonceManagerId({ + client, + address, + blockTag, +}: { + client: Client; + address: Hex; + blockTag: BlockTag; +}): Promise { + const chainId = client.chain?.id ?? (await getChainId(client)); + return `mud:createNonceManager:${chainId}:${getAddress(address)}:${blockTag}`; +} diff --git a/packages/common/src/writeContract.ts b/packages/common/src/writeContract.ts new file mode 100644 index 0000000000..bc32b14e45 --- /dev/null +++ b/packages/common/src/writeContract.ts @@ -0,0 +1,118 @@ +import { + Abi, + Account, + Chain, + Client, + Hex, + SimulateContractParameters, + Transport, + WriteContractParameters, + WriteContractReturnType, +} from "viem"; +import { simulateContract, writeContract as writeContract_ } from "viem/actions"; +import pRetry from "p-retry"; +import { debug as parentDebug } from "./debug"; +import { getNonceManager } from "./getNonceManager"; +import { parseAccount } from "viem/accounts"; + +const debug = parentDebug.extend("writeContract"); +let nextWriteId = 0; + +export type ContractWrite< + TAbi extends Abi | readonly unknown[] = Abi, + TFunctionName extends string = string, + TChain extends Chain | undefined = Chain, + TAccount extends Account | undefined = Account | undefined, + TChainOverride extends Chain | undefined = Chain | undefined +> = { + id: string; + request: WriteContractParameters; + result: Promise; +}; + +export type WriteContractOptions< + TAbi extends Abi | readonly unknown[] = Abi, + TFunctionName extends string = string, + TChain extends Chain | undefined = Chain, + TAccount extends Account | undefined = Account | undefined, + TChainOverride extends Chain | undefined = Chain | undefined +> = WriteContractParameters & { + onWrite?: (write: ContractWrite) => void; +}; + +export async function writeContract< + TChain extends Chain | undefined, + TAccount extends Account | undefined, + TAbi extends Abi | readonly unknown[], + TFunctionName extends string, + TChainOverride extends Chain | undefined +>( + client: Client, + { onWrite, ...request_ }: WriteContractOptions +): Promise { + const request = request_ as WriteContractParameters; + + const account_ = request.account ?? client.account; + if (!account_) { + // TODO: replace with viem AccountNotFoundError once its exported + throw new Error("No account provided"); + } + const account = parseAccount(account_); + + const nonceManager = await getNonceManager({ + client, + address: account.address, + }); + + async function prepareWrite(): Promise< + WriteContractParameters + > { + if (request.gas) { + debug("gas provided, skipping simulate", request); + return request; + } + + debug("simulating write", request); + const result = await simulateContract(client, { + ...request, + account, + } as unknown as SimulateContractParameters); + + return result.request as unknown as WriteContractParameters; + } + + async function write(): Promise { + const preparedWrite = await prepareWrite(); + + return await pRetry( + async () => { + if (!nonceManager.hasNonce()) { + await nonceManager.resetNonce(); + } + + const nonce = nonceManager.nextNonce(); + debug("calling write function with nonce", nonce, preparedWrite); + return await writeContract_(client, { nonce, ...preparedWrite }); + }, + { + retries: 3, + onFailedAttempt: async (error) => { + // On nonce errors, reset the nonce and retry + if (nonceManager.shouldResetNonce(error)) { + debug("got nonce error, retrying", error); + await nonceManager.resetNonce(); + return; + } + // TODO: prepareWrite again if there are gas errors? + throw error; + }, + } + ); + } + + const result = write(); + + onWrite?.({ id: `${nextWriteId++}`, request, result }); + + return result; +} From 52daba2f7987abd5b155bab1640622aff28f047f Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 16:10:37 +0100 Subject: [PATCH 2/8] revert example change --- examples/minimal/packages/client-react/src/mud/setupNetwork.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/minimal/packages/client-react/src/mud/setupNetwork.ts b/examples/minimal/packages/client-react/src/mud/setupNetwork.ts index 04876200ac..895bf70553 100644 --- a/examples/minimal/packages/client-react/src/mud/setupNetwork.ts +++ b/examples/minimal/packages/client-react/src/mud/setupNetwork.ts @@ -16,7 +16,7 @@ export async function setupNetwork() { const clientOptions = { chain: networkConfig.chain, - transport: transportObserver(http()), + transport: transportObserver(fallback([webSocket(), http()])), pollingInterval: 1000, } as const satisfies ClientConfig; From b0d5b7c5536ff4e25c84646700a3f4893df12b0b Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 16:20:15 +0100 Subject: [PATCH 3/8] rename to better align with viem --- packages/common/src/createContract.ts | 108 +------------------------- packages/common/src/getContract.ts | 99 +++++++++++++++++++++++ packages/common/src/index.ts | 8 +- packages/common/src/writeContract.ts | 4 +- 4 files changed, 111 insertions(+), 108 deletions(-) create mode 100644 packages/common/src/getContract.ts diff --git a/packages/common/src/createContract.ts b/packages/common/src/createContract.ts index 03406446cc..04404ea27d 100644 --- a/packages/common/src/createContract.ts +++ b/packages/common/src/createContract.ts @@ -1,106 +1,4 @@ -import { - Abi, - Account, - Address, - Chain, - GetContractParameters, - GetContractReturnType, - Hex, - PublicClient, - Transport, - WalletClient, - WriteContractParameters, - getContract, -} from "viem"; -import { UnionOmit } from "./type-utils/common"; -import { WriteContractOptions, writeContract } from "./writeContract"; +import { getContract } from "./getContract"; -// copied from viem because this isn't exported -// TODO: import from viem? -function getFunctionParameters(values: [args?: readonly unknown[], options?: object]): { - args: readonly unknown[]; - options: object; -} { - const hasArgs = values.length && Array.isArray(values[0]); - const args = hasArgs ? values[0]! : []; - const options = (hasArgs ? values[1] : values[0]) ?? {}; - return { args, options }; -} - -export type ContractWrite = { - id: string; - request: WriteContractParameters; - result: Promise; -}; - -export type CreateContractOptions< - TTransport extends Transport, - TAddress extends Address, - TAbi extends Abi, - TChain extends Chain, - TAccount extends Account, - TPublicClient extends PublicClient, - TWalletClient extends WalletClient -> = Required> & { - onWrite?: (write: ContractWrite) => void; -}; - -export function createContract< - TTransport extends Transport, - TAddress extends Address, - TAbi extends Abi, - TChain extends Chain, - TAccount extends Account, - TPublicClient extends PublicClient, - TWalletClient extends WalletClient ->({ - abi, - address, - publicClient, - walletClient, - onWrite, -}: CreateContractOptions< - TTransport, - TAddress, - TAbi, - TChain, - TAccount, - TPublicClient, - TWalletClient ->): GetContractReturnType { - const contract = getContract({ - abi, - address, - publicClient, - walletClient, - }) as unknown as GetContractReturnType; - - if (contract.write) { - // Replace write calls with our own. Implemented ~the same as viem, but adds better handling of nonces (via queue + retries). - contract.write = new Proxy( - {}, - { - get(_, functionName: string) { - return ( - ...parameters: [ - args?: readonly unknown[], - options?: UnionOmit - ] - ) => { - const { args, options } = getFunctionParameters(parameters); - return writeContract(walletClient, { - abi, - address, - functionName, - args, - ...options, - onWrite, - } as unknown as WriteContractOptions); - }; - }, - } - ); - } - - return contract as unknown as GetContractReturnType; -} +/** @deprecated use `getContract` instead */ +export const createContract = getContract; diff --git a/packages/common/src/getContract.ts b/packages/common/src/getContract.ts new file mode 100644 index 0000000000..8b2d9757bf --- /dev/null +++ b/packages/common/src/getContract.ts @@ -0,0 +1,99 @@ +import { + Abi, + Account, + Address, + Chain, + GetContractParameters, + GetContractReturnType, + PublicClient, + Transport, + WalletClient, + WriteContractParameters, + getContract as viem_getContract, +} from "viem"; +import { UnionOmit } from "./type-utils/common"; +import { WriteContractOptions, writeContract } from "./writeContract"; + +// copied from viem because this isn't exported +// TODO: import from viem? +function getFunctionParameters(values: [args?: readonly unknown[], options?: object]): { + args: readonly unknown[]; + options: object; +} { + const hasArgs = values.length && Array.isArray(values[0]); + const args = hasArgs ? values[0]! : []; + const options = (hasArgs ? values[1] : values[0]) ?? {}; + return { args, options }; +} + +export type GetContractOptions< + TTransport extends Transport, + TAddress extends Address, + TAbi extends Abi, + TChain extends Chain, + TAccount extends Account, + TPublicClient extends PublicClient, + TWalletClient extends WalletClient +> = Required> & { + onWrite?: WriteContractOptions["onWrite"]; +}; + +export function getContract< + TTransport extends Transport, + TAddress extends Address, + TAbi extends Abi, + TChain extends Chain, + TAccount extends Account, + TPublicClient extends PublicClient, + TWalletClient extends WalletClient +>({ + abi, + address, + publicClient, + walletClient, + onWrite, +}: GetContractOptions< + TTransport, + TAddress, + TAbi, + TChain, + TAccount, + TPublicClient, + TWalletClient +>): GetContractReturnType { + const contract = viem_getContract({ + abi, + address, + publicClient, + walletClient, + }) as unknown as GetContractReturnType; + + if (contract.write) { + // Replace write calls with our own. Implemented ~the same as viem, but adds better handling of nonces (via queue + retries). + contract.write = new Proxy( + {}, + { + get(_, functionName: string) { + return ( + ...parameters: [ + args?: readonly unknown[], + options?: UnionOmit + ] + ) => { + const { args, options } = getFunctionParameters(parameters); + return writeContract(walletClient, { + abi, + address, + functionName, + args, + ...options, + onWrite, + } as unknown as WriteContractOptions); + }; + }, + } + ); + } + + return contract as unknown as GetContractReturnType; +} diff --git a/packages/common/src/index.ts b/packages/common/src/index.ts index a15afa9877..b772a50ba8 100644 --- a/packages/common/src/index.ts +++ b/packages/common/src/index.ts @@ -1,11 +1,17 @@ export * from "./common"; export * from "./createBurnerAccount"; -export * from "./createContract"; export * from "./createNonceManager"; +export * from "./getContract"; export * from "./getBurnerPrivateKey"; +export * from "./getNonceManager"; +export * from "./getNonceManagerId"; export * from "./hexToResourceId"; export * from "./readHex"; export * from "./resourceIdToHex"; export * from "./resourceTypes"; export * from "./spliceHex"; export * from "./transportObserver"; +export * from "./writeContract"; + +/** @deprecated use `getContract` instead */ +export { createContract } from "./createContract"; diff --git a/packages/common/src/writeContract.ts b/packages/common/src/writeContract.ts index bc32b14e45..12d45e0ccf 100644 --- a/packages/common/src/writeContract.ts +++ b/packages/common/src/writeContract.ts @@ -9,7 +9,7 @@ import { WriteContractParameters, WriteContractReturnType, } from "viem"; -import { simulateContract, writeContract as writeContract_ } from "viem/actions"; +import { simulateContract, writeContract as viem_writeContract } from "viem/actions"; import pRetry from "p-retry"; import { debug as parentDebug } from "./debug"; import { getNonceManager } from "./getNonceManager"; @@ -92,7 +92,7 @@ export async function writeContract< const nonce = nonceManager.nextNonce(); debug("calling write function with nonce", nonce, preparedWrite); - return await writeContract_(client, { nonce, ...preparedWrite }); + return await viem_writeContract(client, { nonce, ...preparedWrite }); }, { retries: 3, From e31403d76d38df4f2900534b22c96692f3d456c0 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 08:26:31 -0700 Subject: [PATCH 4/8] Create rotten-beers-learn.md --- .changeset/rotten-beers-learn.md | 15 +++++++++++++++ 1 file changed, 15 insertions(+) create mode 100644 .changeset/rotten-beers-learn.md diff --git a/.changeset/rotten-beers-learn.md b/.changeset/rotten-beers-learn.md new file mode 100644 index 0000000000..b566ca4250 --- /dev/null +++ b/.changeset/rotten-beers-learn.md @@ -0,0 +1,15 @@ +--- +"@latticexyz/common": minor +--- + +- Moves contract write logic out of `createContract` into its own `writeContract` method so that it can be used outside of the contract instance, and for consistency with viem. +- Deprecates `createContract` in favor of `getContract` for consistency with viem. +- Reworks `createNonceManager`'s `BroadcastChannel` setup and moves out the notion of a "nonce manager ID" to `getNonceManagerId` so we can create an internal cache with `getNonceManager` for use in `writeContract`. + +If you were using the `createNonceManager` before, you'll just need to rename `publicClient` argument to `client`: + +```diff + const publicClient = createPublicClient({ ... }); +- const nonceManager = createNonceManager({ publicClient, ... }); ++ const nonceManager = createNonceManager({ client: publicClient, ... }); +``` From 069809b31cb19129875ecf35b2f989e015a5b089 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 16:28:09 +0100 Subject: [PATCH 5/8] add TODO --- packages/common/src/getContract.ts | 2 ++ packages/common/src/writeContract.ts | 2 ++ 2 files changed, 4 insertions(+) diff --git a/packages/common/src/getContract.ts b/packages/common/src/getContract.ts index 8b2d9757bf..634da2a305 100644 --- a/packages/common/src/getContract.ts +++ b/packages/common/src/getContract.ts @@ -38,6 +38,8 @@ export type GetContractOptions< onWrite?: WriteContractOptions["onWrite"]; }; +// TODO: migrate away from this approach once we can hook into viem: https://github.com/wagmi-dev/viem/discussions/1230 + export function getContract< TTransport extends Transport, TAddress extends Address, diff --git a/packages/common/src/writeContract.ts b/packages/common/src/writeContract.ts index 12d45e0ccf..e70e994848 100644 --- a/packages/common/src/writeContract.ts +++ b/packages/common/src/writeContract.ts @@ -40,6 +40,8 @@ export type WriteContractOptions< onWrite?: (write: ContractWrite) => void; }; +// TODO: migrate away from this approach once we can hook into viem's nonce management: https://github.com/wagmi-dev/viem/discussions/1230 + export async function writeContract< TChain extends Chain | undefined, TAccount extends Account | undefined, From 2013116f8f21796a12c3071869e8ee4df20bbad2 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 16:30:27 +0100 Subject: [PATCH 6/8] createContract -> getContract everywhere --- e2e/packages/client-vanilla/src/mud/setupNetwork.ts | 4 ++-- e2e/packages/test-data/generate-test-data.ts | 4 ++-- .../minimal/packages/client-phaser/src/mud/setupNetwork.ts | 4 ++-- .../minimal/packages/client-react/src/mud/setupNetwork.ts | 4 ++-- .../minimal/packages/client-vanilla/src/mud/setupNetwork.ts | 4 ++-- packages/cli/src/utils/utils/deployContract.ts | 4 ++-- packages/common/src/index.ts | 2 +- templates/phaser/packages/client/src/mud/createSystemCalls.ts | 2 +- templates/phaser/packages/client/src/mud/setupNetwork.ts | 4 ++-- templates/react/packages/client/src/mud/createSystemCalls.ts | 2 +- templates/react/packages/client/src/mud/setupNetwork.ts | 4 ++-- .../threejs/packages/client/src/mud/createSystemCalls.ts | 2 +- templates/threejs/packages/client/src/mud/setupNetwork.ts | 4 ++-- .../vanilla/packages/client/src/mud/createSystemCalls.ts | 2 +- templates/vanilla/packages/client/src/mud/setupNetwork.ts | 4 ++-- 15 files changed, 25 insertions(+), 25 deletions(-) diff --git a/e2e/packages/client-vanilla/src/mud/setupNetwork.ts b/e2e/packages/client-vanilla/src/mud/setupNetwork.ts index 34cb6e7329..17c1586374 100644 --- a/e2e/packages/client-vanilla/src/mud/setupNetwork.ts +++ b/e2e/packages/client-vanilla/src/mud/setupNetwork.ts @@ -4,7 +4,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver } from "@latticexyz/common"; import mudConfig from "contracts/mud.config"; export type SetupNetworkResult = Awaited>; @@ -26,7 +26,7 @@ export async function setupNetwork() { account: burnerAccount, }); - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/e2e/packages/test-data/generate-test-data.ts b/e2e/packages/test-data/generate-test-data.ts index 4b146a12c3..87a9e659e2 100644 --- a/e2e/packages/test-data/generate-test-data.ts +++ b/e2e/packages/test-data/generate-test-data.ts @@ -13,7 +13,7 @@ import { numberToHex, } from "viem"; import { mudFoundry } from "@latticexyz/common/chains"; -import { createContract } from "@latticexyz/common"; +import { getContract } from "@latticexyz/common"; import { storeEventsAbi } from "@latticexyz/store"; import { privateKeyToAccount } from "viem/accounts"; import IWorldAbi from "../contracts/out/IWorld.sol/IWorld.abi.json"; @@ -63,7 +63,7 @@ const walletClient = createWalletClient({ account, }); -const worldContract = createContract({ +const worldContract = getContract({ address: worldAddress, abi: IWorldAbi, publicClient, diff --git a/examples/minimal/packages/client-phaser/src/mud/setupNetwork.ts b/examples/minimal/packages/client-phaser/src/mud/setupNetwork.ts index 144ecd4a33..6c8e716ef3 100644 --- a/examples/minimal/packages/client-phaser/src/mud/setupNetwork.ts +++ b/examples/minimal/packages/client-phaser/src/mud/setupNetwork.ts @@ -4,7 +4,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; import { Subject, share } from "rxjs"; import mudConfig from "contracts/mud.config"; @@ -28,7 +28,7 @@ export async function setupNetwork() { }); const write$ = new Subject(); - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/examples/minimal/packages/client-react/src/mud/setupNetwork.ts b/examples/minimal/packages/client-react/src/mud/setupNetwork.ts index 895bf70553..1353c0803c 100644 --- a/examples/minimal/packages/client-react/src/mud/setupNetwork.ts +++ b/examples/minimal/packages/client-react/src/mud/setupNetwork.ts @@ -4,7 +4,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { ContractWrite, createBurnerAccount, createContract, transportObserver } from "@latticexyz/common"; +import { ContractWrite, createBurnerAccount, getContract, transportObserver } from "@latticexyz/common"; import { Subject, share } from "rxjs"; import mudConfig from "contracts/mud.config"; import { createClient as createFaucetClient } from "@latticexyz/faucet"; @@ -29,7 +29,7 @@ export async function setupNetwork() { }); const write$ = new Subject(); - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/examples/minimal/packages/client-vanilla/src/mud/setupNetwork.ts b/examples/minimal/packages/client-vanilla/src/mud/setupNetwork.ts index 144ecd4a33..6c8e716ef3 100644 --- a/examples/minimal/packages/client-vanilla/src/mud/setupNetwork.ts +++ b/examples/minimal/packages/client-vanilla/src/mud/setupNetwork.ts @@ -4,7 +4,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; import { Subject, share } from "rxjs"; import mudConfig from "contracts/mud.config"; @@ -28,7 +28,7 @@ export async function setupNetwork() { }); const write$ = new Subject(); - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/packages/cli/src/utils/utils/deployContract.ts b/packages/cli/src/utils/utils/deployContract.ts index 219032127f..e1bec26ebd 100644 --- a/packages/cli/src/utils/utils/deployContract.ts +++ b/packages/cli/src/utils/utils/deployContract.ts @@ -26,8 +26,8 @@ export async function deployContract(input: TxConfig & { nonce: number; contract throw new MUDError( `Error deploying ${contract.name}: invalid bytecode. Note that linking of public libraries is not supported yet, make sure none of your libraries use "external" functions.` ); - } else if (error?.message.includes("CreateContractLimit")) { - throw new MUDError(`Error deploying ${contract.name}: CreateContractLimit exceeded.`); + } else if (error?.message.includes("getContractLimit")) { + throw new MUDError(`Error deploying ${contract.name}: getContractLimit exceeded.`); } else throw error; } } diff --git a/packages/common/src/index.ts b/packages/common/src/index.ts index b772a50ba8..98fa9fdf7f 100644 --- a/packages/common/src/index.ts +++ b/packages/common/src/index.ts @@ -14,4 +14,4 @@ export * from "./transportObserver"; export * from "./writeContract"; /** @deprecated use `getContract` instead */ -export { createContract } from "./createContract"; +export { getContract } from "./getContract"; diff --git a/templates/phaser/packages/client/src/mud/createSystemCalls.ts b/templates/phaser/packages/client/src/mud/createSystemCalls.ts index 9a1c478041..b308bd73b2 100644 --- a/templates/phaser/packages/client/src/mud/createSystemCalls.ts +++ b/templates/phaser/packages/client/src/mud/createSystemCalls.ts @@ -24,7 +24,7 @@ export function createSystemCalls( * SetupNetworkResult, as defined in setupNetwork.ts * * - Out of this parameter, we only care about two fields: - * - worldContract (which comes from createContract, see + * - worldContract (which comes from getContract, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/phaser/packages/client/src/mud/setupNetwork.ts#L31). * - waitForTransaction (which comes from syncToRecs, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/phaser/packages/client/src/mud/setupNetwork.ts#L39). diff --git a/templates/phaser/packages/client/src/mud/setupNetwork.ts b/templates/phaser/packages/client/src/mud/setupNetwork.ts index 6fe7d00e2c..0b037665d0 100644 --- a/templates/phaser/packages/client/src/mud/setupNetwork.ts +++ b/templates/phaser/packages/client/src/mud/setupNetwork.ts @@ -9,7 +9,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; import { Subject, share } from "rxjs"; /* @@ -58,7 +58,7 @@ export async function setupNetwork() { /* * Create an object for communicating with the deployed World. */ - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/templates/react/packages/client/src/mud/createSystemCalls.ts b/templates/react/packages/client/src/mud/createSystemCalls.ts index 9858910ed2..d351f67040 100644 --- a/templates/react/packages/client/src/mud/createSystemCalls.ts +++ b/templates/react/packages/client/src/mud/createSystemCalls.ts @@ -18,7 +18,7 @@ export function createSystemCalls( * SetupNetworkResult, as defined in setupNetwork.ts * * - Out of this parameter, we only care about two fields: - * - worldContract (which comes from createContract, see + * - worldContract (which comes from getContract, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/react/packages/client/src/mud/setupNetwork.ts#L31). * - waitForTransaction (which comes from syncToRecs, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/react/packages/client/src/mud/setupNetwork.ts#L39). diff --git a/templates/react/packages/client/src/mud/setupNetwork.ts b/templates/react/packages/client/src/mud/setupNetwork.ts index 15d3555441..fe78721b3f 100644 --- a/templates/react/packages/client/src/mud/setupNetwork.ts +++ b/templates/react/packages/client/src/mud/setupNetwork.ts @@ -10,7 +10,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; import { Subject, share } from "rxjs"; @@ -60,7 +60,7 @@ export async function setupNetwork() { /* * Create an object for communicating with the deployed World. */ - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/templates/threejs/packages/client/src/mud/createSystemCalls.ts b/templates/threejs/packages/client/src/mud/createSystemCalls.ts index b87d86253d..782f61d94b 100644 --- a/templates/threejs/packages/client/src/mud/createSystemCalls.ts +++ b/templates/threejs/packages/client/src/mud/createSystemCalls.ts @@ -17,7 +17,7 @@ export function createSystemCalls( * SetupNetworkResult, as defined in setupNetwork.ts * * - Out of this parameter, we only care about two fields: - * - worldContract (which comes from createContract, see + * - worldContract (which comes from getContract, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/threejs/packages/client/src/mud/setupNetwork.ts#L31). * - waitForTransaction (which comes from syncToRecs, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/threejs/packages/client/src/mud/setupNetwork.ts#L39). diff --git a/templates/threejs/packages/client/src/mud/setupNetwork.ts b/templates/threejs/packages/client/src/mud/setupNetwork.ts index eb8a98ae84..274a46b80e 100644 --- a/templates/threejs/packages/client/src/mud/setupNetwork.ts +++ b/templates/threejs/packages/client/src/mud/setupNetwork.ts @@ -9,7 +9,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; import { Subject, share } from "rxjs"; /* @@ -58,7 +58,7 @@ export async function setupNetwork() { /* * Create an object for communicating with the deployed World. */ - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, diff --git a/templates/vanilla/packages/client/src/mud/createSystemCalls.ts b/templates/vanilla/packages/client/src/mud/createSystemCalls.ts index 9a364b4920..22449a324f 100644 --- a/templates/vanilla/packages/client/src/mud/createSystemCalls.ts +++ b/templates/vanilla/packages/client/src/mud/createSystemCalls.ts @@ -18,7 +18,7 @@ export function createSystemCalls( * SetupNetworkResult, as defined in setupNetwork.ts * * - Out of this parameter, we only care about two fields: - * - worldContract (which comes from createContract, see + * - worldContract (which comes from getContract, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L31). * - waitForTransaction (which comes from syncToRecs, see * https://github.com/latticexyz/mud/blob/26dabb34321eedff7a43f3fcb46da4f3f5ba3708/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L39). diff --git a/templates/vanilla/packages/client/src/mud/setupNetwork.ts b/templates/vanilla/packages/client/src/mud/setupNetwork.ts index 15d3555441..fe78721b3f 100644 --- a/templates/vanilla/packages/client/src/mud/setupNetwork.ts +++ b/templates/vanilla/packages/client/src/mud/setupNetwork.ts @@ -10,7 +10,7 @@ import { encodeEntity, syncToRecs } from "@latticexyz/store-sync/recs"; import { getNetworkConfig } from "./getNetworkConfig"; import { world } from "./world"; import IWorldAbi from "contracts/out/IWorld.sol/IWorld.abi.json"; -import { createBurnerAccount, createContract, transportObserver, ContractWrite } from "@latticexyz/common"; +import { createBurnerAccount, getContract, transportObserver, ContractWrite } from "@latticexyz/common"; import { Subject, share } from "rxjs"; @@ -60,7 +60,7 @@ export async function setupNetwork() { /* * Create an object for communicating with the deployed World. */ - const worldContract = createContract({ + const worldContract = getContract({ address: networkConfig.worldAddress as Hex, abi: IWorldAbi, publicClient, From 94c1d3c8551ab2f603f0d1ed2e66246421665963 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 16:39:08 +0100 Subject: [PATCH 7/8] type cast ugh --- packages/common/src/writeContract.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/common/src/writeContract.ts b/packages/common/src/writeContract.ts index e70e994848..4ce2a9353f 100644 --- a/packages/common/src/writeContract.ts +++ b/packages/common/src/writeContract.ts @@ -94,7 +94,7 @@ export async function writeContract< const nonce = nonceManager.nextNonce(); debug("calling write function with nonce", nonce, preparedWrite); - return await viem_writeContract(client, { nonce, ...preparedWrite }); + return await viem_writeContract(client, { nonce, ...preparedWrite } as typeof preparedWrite); }, { retries: 3, From 66c4e4bb30635700927165cb19e9c0ad39511280 Mon Sep 17 00:00:00 2001 From: Kevin Ingersoll Date: Tue, 3 Oct 2023 16:45:04 +0100 Subject: [PATCH 8/8] string replace too aggressive --- packages/cli/src/utils/utils/deployContract.ts | 4 ++-- packages/common/src/index.ts | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/cli/src/utils/utils/deployContract.ts b/packages/cli/src/utils/utils/deployContract.ts index e1bec26ebd..219032127f 100644 --- a/packages/cli/src/utils/utils/deployContract.ts +++ b/packages/cli/src/utils/utils/deployContract.ts @@ -26,8 +26,8 @@ export async function deployContract(input: TxConfig & { nonce: number; contract throw new MUDError( `Error deploying ${contract.name}: invalid bytecode. Note that linking of public libraries is not supported yet, make sure none of your libraries use "external" functions.` ); - } else if (error?.message.includes("getContractLimit")) { - throw new MUDError(`Error deploying ${contract.name}: getContractLimit exceeded.`); + } else if (error?.message.includes("CreateContractLimit")) { + throw new MUDError(`Error deploying ${contract.name}: CreateContractLimit exceeded.`); } else throw error; } } diff --git a/packages/common/src/index.ts b/packages/common/src/index.ts index 98fa9fdf7f..b772a50ba8 100644 --- a/packages/common/src/index.ts +++ b/packages/common/src/index.ts @@ -14,4 +14,4 @@ export * from "./transportObserver"; export * from "./writeContract"; /** @deprecated use `getContract` instead */ -export { getContract } from "./getContract"; +export { createContract } from "./createContract";