From 6877cdd35a5437f89fb5288a39ef7e3c325d03d1 Mon Sep 17 00:00:00 2001 From: Nadai2010 Date: Thu, 17 Oct 2024 20:28:34 +0100 Subject: [PATCH] Fix in write hooks parser (useScaffoldMultiwriteContracts and useScaffoldWriteContracts and useScaffoldReadContracts) add update file contract.ts --- .../useScaffoldMultiWriteContract.ts | 44 +++-- .../scaffold-stark/useScaffoldReadContract.ts | 2 +- .../useScaffoldWriteContract.ts | 173 ++++++++++-------- .../nextjs/utils/scaffold-stark/contract.ts | 48 +++-- 4 files changed, 154 insertions(+), 113 deletions(-) diff --git a/packages/nextjs/hooks/scaffold-stark/useScaffoldMultiWriteContract.ts b/packages/nextjs/hooks/scaffold-stark/useScaffoldMultiWriteContract.ts index 16728c86..e6661477 100644 --- a/packages/nextjs/hooks/scaffold-stark/useScaffoldMultiWriteContract.ts +++ b/packages/nextjs/hooks/scaffold-stark/useScaffoldMultiWriteContract.ts @@ -11,7 +11,7 @@ import { UseScaffoldWriteConfig, } from "~~/utils/scaffold-stark/contract"; import { useSendTransaction, useNetwork, Abi } from "@starknet-react/core"; -import { InvocationsDetails } from "starknet"; +import { Contract as StarknetJsContract, InvocationsDetails } from "starknet"; import { notification } from "~~/utils/scaffold-stark"; import { useMemo } from "react"; import { useTransactor } from "./useTransactor"; @@ -44,30 +44,38 @@ export const useScaffoldMultiWriteContract = < contractName as ContractName ] as Contract; - const abiFunction = getFunctionsByStateMutability( - contract?.abi || [], - "external", - ).find((fn) => fn.name === functionName); + // TODO: see if we still need this + // const abiFunction = getFunctionsByStateMutability( + // contract?.abi || [], + // "external", + // ).find((fn) => fn.name === functionName); + + // we convert to starknetjs contract instance here since deployed data may be undefined if contract is not deployed + const contractInstance = new StarknetJsContract( + contract.abi, + contract.address, + ); return { - contractAddress: contract?.address, - entrypoint: functionName, - calldata: - abiFunction && unParsedArgs && contract - ? parseFunctionParams({ - abiFunction, - isRead: false, - inputs: unParsedArgs as any[], - isReadArgsParsing: false, - abi: contract.abi, - }).flat() - : [], + ...contractInstance.populate(functionName, unParsedArgs as any[]), + + // TODO: see if we still need this + // calldata: + // abiFunction && unParsedArgs && contract + // ? parseFunctionParams({ + // abiFunction, + // isRead: false, + // inputs: unParsedArgs as any[], + // isReadArgsParsing: false, + // abi: contract.abi, + // }).flat() + // : [], }; }); } else { return []; } - }, [calls]); + }, [calls, targetNetwork.network]); // TODO add custom options const sendTransactionInstance = useSendTransaction({ diff --git a/packages/nextjs/hooks/scaffold-stark/useScaffoldReadContract.ts b/packages/nextjs/hooks/scaffold-stark/useScaffoldReadContract.ts index b9b9431c..fae3d663 100644 --- a/packages/nextjs/hooks/scaffold-stark/useScaffoldReadContract.ts +++ b/packages/nextjs/hooks/scaffold-stark/useScaffoldReadContract.ts @@ -37,4 +37,4 @@ export const useScaffoldReadContract = < }) as Omit, "data"> & { data: AbiFunctionOutputs | undefined; }; -}; \ No newline at end of file +}; diff --git a/packages/nextjs/hooks/scaffold-stark/useScaffoldWriteContract.ts b/packages/nextjs/hooks/scaffold-stark/useScaffoldWriteContract.ts index 61f41df7..7632c9e8 100644 --- a/packages/nextjs/hooks/scaffold-stark/useScaffoldWriteContract.ts +++ b/packages/nextjs/hooks/scaffold-stark/useScaffoldWriteContract.ts @@ -1,4 +1,4 @@ -import { useEffect, useMemo } from "react"; +import { useCallback, useEffect, useMemo } from "react"; import { useTargetNetwork } from "./useTargetNetwork"; import { useDeployedContractInfo, @@ -12,8 +12,14 @@ import { parseFunctionParams, UseScaffoldWriteConfig, } from "~~/utils/scaffold-stark/contract"; -import { useSendTransaction, useNetwork, Abi } from "@starknet-react/core"; +import { + useSendTransaction, + useNetwork, + Abi, + useContract, +} from "@starknet-react/core"; import { notification } from "~~/utils/scaffold-stark"; +import { Contract as StarknetJsContract } from "starknet"; type UpdatedArgs = Parameters< ReturnType["sendAsync"] @@ -45,93 +51,100 @@ export const useScaffoldWriteContract = < [deployedContractData?.abi, functionName], ); - const parsedParams = useMemo(() => { - if (args && abiFunction && deployedContractData) { - const parsed = parseFunctionParams({ - abiFunction, - abi: deployedContractData.abi, - inputs: args as any[], - isRead: false, - isReadArgsParsing: false, - }).flat(Infinity); - return parsed; - } - return []; - }, [args, abiFunction, deployedContractData]); + // TODO: see if we need this bit later + // const parsedParams = useMemo(() => { + // if (args && abiFunction && deployedContractData) { + // const parsed = parseFunctionParams({ + // abiFunction, + // abi: deployedContractData.abi, + // inputs: args as any[], + // isRead: false, + // isReadArgsParsing: true, + // }).flat(Infinity); + // return parsed; + // } + // return []; + // }, [args, abiFunction, deployedContractData]); - const sendTransactionInstance = useSendTransaction({ - calls: deployedContractData - ? [ - { - contractAddress: deployedContractData?.address, - entrypoint: functionName, - calldata: parsedParams, - }, - ] - : [], - }); + // leave blank for now since default args will be called by the trigger function anyway + const sendTransactionInstance = useSendTransaction({}); - const sendContractWriteTx = async (params?: { - args?: UseScaffoldWriteConfig["args"]; - }) => { - // if no args supplied, use the one supplied from hook - let newArgs = params?.args; - if (Object.keys(newArgs || {}).length <= 0) { - newArgs = args; - } + const sendContractWriteTx = useCallback( + async (params?: { + args?: UseScaffoldWriteConfig["args"]; + }) => { + // if no args supplied, use the one supplied from hook + let newArgs = params?.args; + if (Object.keys(newArgs || {}).length <= 0) { + newArgs = args; + } - if (!deployedContractData) { - console.error( - "Target Contract is not deployed, did you forget to run `yarn deploy`?", + if (!deployedContractData) { + console.error( + "Target Contract is not deployed, did you forget to run `yarn deploy`?", + ); + return; + } + if (!chain?.id) { + console.error("Please connect your wallet"); + return; + } + if (chain?.id !== targetNetwork.id) { + console.error("You are on the wrong network"); + return; + } + + // TODO: see if we need this back, keeping this here + // let newParsedParams = + // newArgs && abiFunction && deployedContractData + // ? parseFunctionParams({ + // abiFunction, + // abi: deployedContractData.abi, + // inputs: newArgs as any[], + // isRead: false, + // isReadArgsParsing: false, + // }) + // : parsedParams; + + // we convert to starknetjs contract instance here since deployed data may be undefined if contract is not deployed + const contractInstance = new StarknetJsContract( + deployedContractData.abi, + deployedContractData.address, ); - return; - } - if (!chain?.id) { - console.error("Please connect your wallet"); - return; - } - if (chain?.id !== targetNetwork.id) { - console.error("You are on the wrong network"); - return; - } - let newParsedParams = - newArgs && abiFunction && deployedContractData - ? parseFunctionParams({ - abiFunction, - abi: deployedContractData.abi, - inputs: newArgs as any[], - isRead: false, - isReadArgsParsing: false, - }).flat(Infinity) - : parsedParams; - const newCalls = [ - { - contractAddress: deployedContractData.address, - entrypoint: functionName, - calldata: newParsedParams, - }, - ]; + const newCalls = deployedContractData + ? [contractInstance.populate(functionName, newArgs as any[])] + : []; - if (sendTransactionInstance.sendAsync) { - try { - // setIsMining(true); - return await sendTxnWrapper(() => - sendTransactionInstance.sendAsync(newCalls as any[]), - ); - } catch (e: any) { - throw e; - } finally { - // setIsMining(false); + if (sendTransactionInstance.sendAsync) { + try { + // setIsMining(true); + return await sendTxnWrapper(() => + sendTransactionInstance.sendAsync(newCalls as any[]), + ); + } catch (e: any) { + throw e; + } finally { + // setIsMining(false); + } + } else { + notification.error("Contract writer error. Try again."); + return; } - } else { - notification.error("Contract writer error. Try again."); - return; - } - }; + }, + [ + args, + chain?.id, + deployedContractData, + functionName, + sendTransactionInstance, + sendTxnWrapper, + targetNetwork.id, + ], + ); return { ...sendTransactionInstance, sendAsync: sendContractWriteTx, }; -}; \ No newline at end of file +}; diff --git a/packages/nextjs/utils/scaffold-stark/contract.ts b/packages/nextjs/utils/scaffold-stark/contract.ts index eb64030b..6016009e 100644 --- a/packages/nextjs/utils/scaffold-stark/contract.ts +++ b/packages/nextjs/utils/scaffold-stark/contract.ts @@ -239,22 +239,16 @@ export type ExtractAbiFunctionNamesScaffold< TAbiStateMutability extends AbiStateMutability = AbiStateMutability, > = ExtractAbiFunctionsScaffold["name"]; -// helper function will only take from interfaces : //TODO: see if we can make it more generic // helper function will only take from interfaces : //TODO: see if we can make it more generic export type ExtractAbiFunctionsScaffold< TAbi extends Abi, TAbiStateMutability extends AbiStateMutability = AbiStateMutability, > = Extract< ExtractAbiInterfaces["items"][number], - | { - state_mutability: TAbiStateMutability; - } - | Extract< - TAbi[number], - { - type: "function"; - } - > + { + type: "function"; + state_mutability: TAbiStateMutability; + } >; // helper function will only take from interfaces : //TODO: see if we can make it more generic @@ -334,6 +328,31 @@ export type AbiFunctionOutputs< TFunctionName extends string, > = ExtractAbiFunctionScaffold["outputs"]; +/*export type AbiEventInputs> = ExtractAbiEvent< + TAbi, + TEventName +>["inputs"]; + +type IndexedEventInputs< + TContractName extends ContractName, + TEventName extends ExtractAbiEventNames>, +> = Extract, TEventName>[number], { indexed: true }>; + +export type EventFilters< + TContractName extends ContractName, + TEventName extends ExtractAbiEventNames>, +> = IsContractDeclarationMissing< + any, + IndexedEventInputs extends never + ? never + : { + [Key in IsContractDeclarationMissing< + any, + IndexedEventInputs["name"] + >]?: AbiParameterToPrimitiveType, { name: Key }>>; + } +>;*/ + export type UseScaffoldEventHistoryConfig< TContractName extends ContractName, TEventName extends ExtractAbiEventNames>, @@ -764,13 +783,11 @@ export function parseFunctionParams({ args: inputs, }); - console.debug({ formattedInputs }); - formattedInputs.forEach((inputItem) => { const { type: inputType, value: inputValue } = inputItem; parsedInputs.push( - parseParamWithType(inputType, inputValue, isRead, !!isReadArgsParsing), + deepParseValues(inputValue, isRead, inputType, !!isReadArgsParsing), ); }); @@ -822,7 +839,10 @@ function formatInputForParsing({ _formatInput(structValue, argIndex, variants as AbiParameter[]), ]; }); - return { type: structName, value: { variant: Object.fromEntries } }; + return { + type: structName, + value: { variant: Object.fromEntries(formattedEntries) }, + }; } const { members } = structDef as AbiStruct;