TypeScript 
viem is designed to be as type-safe as possible! Things to keep in mind:
- Types currently require using TypeScript v5.0.4 or greater.
- Changes to types in this repository are considered non-breaking and are usually released as patch semver changes (otherwise every type enhancement would be a major version!).
- It is highly recommended that you lock your viempackage version to a specific patch release and upgrade with the expectation that types may be fixed or upgraded between any release.
- The non-type-related public API of viemstill follows semver very strictly.
To ensure everything works correctly, make sure that your tsconfig.json has strict mode set to true:
{
  "compilerOptions": {
    "strict": true
  }
}Type Inference 
viem can infer types based on ABI and EIP-712 Typed Data definitions (powered by ABIType), giving you full end-to-end type-safety from your contracts to your frontend and incredible developer experience (e.g. autocomplete ABI function names and catch misspellings, strongly-typed ABI function arguments, etc.).
For this to work, you must either add const assertions to specific configuration parameters (more info on those below) or define them inline. For example, readContract's abi configuration parameter:
const result = client.readContract({
  abi: […], // <--- defined inline
})const abi = […] as const // <--- const assertion
const result = client.readContract({ abi })If type inference isn't working, it's likely you forgot to add a const assertion or define the configuration parameter inline.
TIP
Unfortunately TypeScript doesn't support importing JSON as const. Check out @wagmi/cli to help with this! It can automatically fetch ABIs from Etherscan, resolve ABIs from your Foundry/Hardhat projects, and much more.
Contract ABIs 
The following actions and utilities support type inference when you add a const assertion to abi or define abi inline:
Actions 
- createEventFilter
- watchEvent
- createContractEventFilter
- deployContract
- estimateContractGas
- multicall
- readContract
- simulateContract
- writeContract
- watchContractEvent
Utilities 
- decodeEventLog
- decodeFunctionResult
- encodeDeployData
- encodeErrorResult
- encodeEventTopics
- encodeFunctionData
- encodeFunctionResult
- getAbiItem
For example, readContract:
const result = client.readContract({
  //  ^? const data: bigint | undefined
  address: '0xecb504d39723b0be0e3a9aa33d646642d1051ee1',
  abi: [
    {
      name: 'balanceOf',
      type: 'function',
      stateMutability: 'view',
      inputs: [{ name: 'account', type: 'address' }],
      outputs: [{ type: 'uint256' }],
    },
    {
      name: 'totalSupply',
      type: 'function',
      stateMutability: 'view',
      inputs: [],
      outputs: [{ type: 'uint256' }],
    },
    {
      name: 'transfer',
      type: 'function',
      stateMutability: 'nonpayable',
      inputs: [
        { name: 'recipient', type: 'address' },
        { name: 'tokenId', type: 'uint256' },
      ],
      outputs: [{ type: 'bool' }],
    },
  ],
  functionName: 'balanceOf',
  // ^? (property) functionName?: "balanceOf" | "totalSupply" | undefined
  // Notice how "transfer" is not included since it is not a "read" function
  args: ['0x27a69ffba1e939ddcfecc8c7e0f967b872bac65c'],
  // ^? (property) args?: readonly [`0x${string}`] | undefined
})EIP-712 Typed Data 
Adding a const assertion to types or defining types inline adds type inference to signTypedData's value configuration parameter:
const result = client.signTypedData({
  domain: {
    name: 'Ether Mail',
    version: '1',
    chainId: 1,
    verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC',
  },
  types: {
    Person: [
      { name: 'name', type: 'string' },
      { name: 'wallet', type: 'address' },
    ],
    Mail: [
      { name: 'from', type: 'Person' },
      { name: 'to', type: 'Person' },
      { name: 'contents', type: 'string' },
    ],
  },
  value: {
    // ^? (parameter) value?: { name: string; wallet: `0x${string}` } | {
    //     from: {
    //         name: string;
    //         wallet: `0x${string}`;
    //     };
    //     to: {
    //         name: string;
    //         wallet: `0x${string}`;
    //     };
    //     contents: string;
    // } | undefined
    from: {
      name: 'Cow',
      wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826',
    },
    to: {
      name: 'Bob',
      wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB',
    },
    contents: 'Hello, Bob!',
  },
})Other 
The following utilities support type inference when you use const assertions or define arguments inline:
- decodeAbiParameters
- encodeAbiParameters
- encodePacked
- parseAbi
- parseAbiItem
- parseAbiParameter
- parseAbiParameters
Configuring Internal Types 
For advanced use-cases, you may want to configure viem's internal types. Most of viem's types relating to ABIs and EIP-712 Typed Data are powered by ABIType. See ABIType's documentation for more info on how to configure types.
window Polyfill 
By importing the viem/window Polyfill, the global window.ethereum will typed as an EIP1193Provider (including a fully-typed request function & typed events). It may be undefined in cases where no browser extension Wallet is detected, or if rendered on the server.
import 'viem/window';
if (typeof window !== 'undefined') {
  const transaction = await window.ethereum.request({
//      ^? const transaction: Transaction        
    method: 'eth_getTransactionByHash',
//  ^? (property) method: "eth_blockNumber" | "eth_call" | ...
    params: ['0x...']
//  ^? (property) params: [hash: Hash]
  })
}
