immunx-agent-tools v0.0.3
General Agent Module
Description
This module contains some common approaches for building Immunx Agents. You will also find some tools for writing tests for these agents. These approaches can be composed for creating more complex agents or used only for checking without returning their findings.
Installation
- Using npm: npm i immunx-agent-tools
or
- Clone the repo
- npm install
- npm run test
Handlers
Handlers are approaches for dealing with block and transaction events. They can either be integrated into a sensor's logic to make it easier to get specific data based on transactions or blocks or used like Immunx sensor generators through.
Each handler gets specific data, which is called metadata (related to what would be relevant for the alert metadata),
from a transaction or block event. This data can be returned to be processed externally by calling
handler.metadata(event), but the handler can also receive a callback that creates a finding from its metadata. In
this case, it can both return findings through handler.handle(event) and also generate the Immunx sensor handlers through
handler.getHandleBlock() and handler.getHandleTransaction().
Handler is an abstract base class and each specific handler extends it. The common interface is:
- Handler(options): Each handler has a specific set of options. The only common field between all the specific options is the optional- onFinding, that defines how a finding will be generated based on the metadata.- Handler, in this case, is a specific handler, not the- Handlerbase class.
- metadata(event): This method returns a promise to an array of metadata objects related to a transaction or block or, if there's no implementation for a specific event (e.g. a handler only gets information from transactions, not blocks) it will return a promise that resolves to- null.
- handle(event, onFinding?): This method handles a transaction or block event and returns a list of findings.- onFindingwill override- options.onFindingif both were specified.
- getHandleBlock(onFinding?): This method returns a- immunx-agent- HandleBlockhandle callback.- onFindingwill override- options.onFindingif both were specified.
- getHandleTransaction(onFinding?): This method returns a- immunx-agent- HandleTransactionhandle callback.- onFindingwill override- options.onFindingif both were specified.
Each handler's options and metadata interfaces can be accessed through Handler.Options and Handler.Metadata
(Handler, again, in this case, being a specific handler, not the Handler base class).
BlacklistedAddresses
This handler detects transactions that contain addresses from a list provided by the user.
How to use it
import { Finding, FindingSeverity, FindingType, TransactionEvent } from "immunx-agent";
import { handlers, createAddress } from "immunx-agent-tools";
const blacklistedAddressesHandler = new handlers.BlacklistedAddresses({
  addresses: [createAddress("0x0")],
  onFinding(metadata) {
    return Finding.from({
      name: "Blacklisted Address",
      description: "A transaction involving a blacklisted address was found",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {},
      addresses: metadata.addresses,
    });
  },
});
const handleTransaction = blacklistedAddressesHandler.getHandleTransaction();
const handleBlock = blacklistedAddressesHandler.getHandleBlock();
// or
async function handleTransaction(txEvent: TransactionEvent): Promise<Finding[]> {
  const findings = await blacklistedAddressesHandler.handle(txEvent);
  // or
  const metadataList = await blacklistedAddressesHandler.metadata(txEvent);
  const findingsFromMetadata = (metadataList || []).map((metadata) => {
    return Finding.from({
      name: "Blacklisted Address",
      description: "A transaction involving a blacklisted address was found",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {},
      addresses: metadata.addresses,
    });
  });
  // ...
}Options
- addresses: Blacklisted addresses. A finding should be generated if any of them is involved in a transaction.
Metadata
- addresses: Blacklisted addresses that were involved in a transaction.
Erc20Transfers
This handler detects ERC20 token transfers in transactions.
How to use it
import { Finding, FindingSeverity, FindingType, TransactionEvent } from "immunx-agent";
import { handlers, createAddress } from "immunx-agent-tools";
const erc20TransfersHandler = new handlers.Erc20Transfers({
  emitter: createAddress("0x0"),
  from: createAddress("0x1"),
  to: createAddress("0x2"),
  amountThreshold: "10000", // or (amount) => amount.gte("100")
  onFinding(metadata) {
    return Finding.from({
      name: "Large ERC20 transfer",
      description: "A large ERC20 transfer was detected",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {
        from: metadata.from,
        to: metadata.to,
        amount: metadata.amount.toString(),
      },
    });
  },
});
const handleTransaction = erc20TransfersHandler.getHandleTransaction();
const handleBlock = erc20TransfersHandler.getHandleBlock();
// or
async function handleTransaction(txEvent: TransactionEvent): Promise<Finding[]> {
  const findings = await erc20TransfersHandler.handle(txEvent);
  // or
  const metadataList = await erc20TransfersHandler.metadata(txEvent);
  const findingsFromMetadata = (metadataList || []).map((metadata) => {
    return Finding.from({
      name: "Large ERC20 transfer",
      description: "A large ERC20 transfer was detected",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {
        token: metadata.emitter,
        from: metadata.from,
        to: metadata.to,
        amount: metadata.amount.toString(),
      },
    });
  });
  // ...
}Options
- emitter: Token address, emitter of the- Transferevents that will be listened.
- from: Transfer sender.
- to: Transfer receiver.
- amountThreshold: Determines a filter based on the transfer amount. Can be either a value, like- "1000"(doesn't consider the token's decimal places, same- uint256representation as in the contract), case in which the transfer event will be filtered out when it amount less than that value, or a callback that defines whether the amount should lead to a finding or not.
Metadata
- emitter: Token address.
- from: Transfer sender.
- to: Transfer receiver.
- amount: Transfer amount.
EthTransfers
This handler detects ether transfers in transactions.
How to use it
import { Finding, FindingSeverity, FindingType, TransactionEvent } from "immunx-agent";
import { handlers, createAddress } from "immunx-agent-tools";
const ethTransfersHandler = new handlers.EthTransfers({
  from: createAddress("0x0"),
  to: createAddress("0x1"),
  valueThreshold: "10000", // or (value) => value.gte("100")
  onFinding(metadata) {
    return Finding.from({
      name: "Large ether transfer",
      description: "A large ether transfer was detected",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {
        from: metadata.from,
        to: metadata.to,
        value: metadata.value.toString(),
      },
    });
  },
});
const handleTransaction = ethTransfersHandler.getHandleTransaction();
const handleBlock = ethTransfersHandler.getHandleBlock();
// or
async function handleTransaction(txEvent: TransactionEvent): Promise<Finding[]> {
  const findings = await ethTransfersHandler.handle(txEvent);
  // or
  const metadataList = await ethTransfersHandler.metadata(txEvent);
  const findingsFromMetadata = (metadataList || []).map((metadata) => {
    return Finding.from({
      name: "Large ether transfer",
      description: "A large ether transfer was detected",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {
        from: metadata.from,
        to: metadata.to,
        value: metadata.value.toString(),
      },
    });
  });
  // ...
}Options
- from: Transfer sender.
- to: Transfer receiver.
- valueThreshold: Determines a filter based on the transfer amount. Can be either a value, like- "1000"(in wei), case in which the transfer event will be filtered out when it amount less than that value, or a callback that defines whether the amount should lead to a finding or not.
Metadata
- from: Transfer sender.
- to: Transfer receiver.
- value: Transferred value in wei.
TraceCalls
This handler parses and detects specific calls in transactions traces.
How to use it
import { Finding, FindingSeverity, FindingType, TransactionEvent } from "immunx-agent";
import { handlers, createAddress } from "immunx-agent-tools";
const traceCallsHandler = new handlers.TraceCalls({
  signatures: ["function func(uint256 param) returns (uint256 resp)"],
  from: createAddress("0x0"),
  to: createAddress("0x1"),
  includeErrors: false,
  filterByArguments(args) {
    return args.param.eq(0);
  },
  filterByOutput(output) {
    return output.resp.eq(1);
  },
  filter(metadata) {
    return metadata.trace.traceAddress.length === 1;
  },
  onFinding(metadata) {
    return Finding.from({
      name: "Func called in traces",
      description: "A func call was detected in the transaction's traces",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {
        from: metadata.from,
        to: metadata.to,
        error: metadata.error,
        param: metadata.args.param.toString(),
        resp: metadata.output.resp.toString(),
      },
    });
  },
});
const handleTransaction = traceCallsHandler.getHandleTransaction();
const handleBlock = traceCallsHandler.getHandleBlock();
// or
async function handleTransaction(txEvent: TransactionEvent): Promise<Finding[]> {
  const findings = await traceCallsHandler.handle(txEvent);
  // or
  const metadataList = await traceCallsHandler.metadata(txEvent);
  const findingsFromMetadata = (metadataList || []).map((metadata) => {
    return Finding.from({
      name: "Func called in traces",
      description: "A func call was detected in the transaction's traces",
      severity: FindingSeverity.Info,
      type: FindingType.Info,
      metadata: {
        from: metadata.from,
        to: metadata.to,
        error: metadata.error,
        param: metadata.args.param.toString(),
        resp: metadata.output.resp.toString(),
      },
    });
  });
  // ...
}Options
- signatures: Function signatures to be monitored. Also used in decoding.
- from: Call sender.
- to: Call receiver.
- includeErrors: Whether calls that had an error should be included or not (by default, falsy).
- filterByArguments: Callback (same signature as a- array.filter(cb)callback) to filter calls by arguments.
- filterByOutput: Callback (same signature as a- array.filter(cb)callback) to filter calls by returned values.
- filter: Callback (same signature as a- array.filter(cb)callback) to filter calls by metadata.
Metadata
- from: Call sender.
- to: Call receiver.
- trace: Trace object.
- error: Whether there was an error during the call or not.
- output: Call result. Will be- nullif- erroris- true.
As well as ethers.utils.TransactionDescription's fields:
- functionFragment: Function fragment from the signature.
- name: Function name.
- args: Function arguments.
- signature: Function signature.
- sighash: Function sighash.
- value: Transaction value in wei.
Utils
Address Handling
These are utility functions to create and manipulate addresses.
- padAddress(address): Simply pads left a hex string with zeroes so it fits the expected length.
- createAddress(address): Pads the provided address and ensures it is lowercase.
- createChecksumAddress(address): Pads the provided address and ensures it is in checksum format.
- toChecksumAddress(address): Formats a valid address (case-insensitive) in checksum format.
TestTransactionEvent
This is a helper class for creating TransactionEvents using the fluent interface pattern.
How to use it
import { TestTransactionEvent } from "immunx-agent-tools/lib/test";
const txEvent: TransactionEvent = new TestTransactionEvent().setFrom(address1).setTo(address2);There are multiple methods you can use for creating the exact TransactionEvent you want:
- setFrom(address)This method sets the- transaction.fromfield in the event.
- setTo(address)This method sets the- transaction.tofield in the event.
- setGas(value)This method sets the- transaction.gasfield in the event.
- setGasPrice(value)This method sets the- transaction.gasPricefield in the event.
- setValue(value)This method sets the- transaction.valuefield in the event.
- setData(data)This method sets the- transaction.datafield in the event.
- setGasUsed(value)This method sets the- receipt.gasUsedfield in the event.
- setStatus(status)This method sets the- receipt.statusfield in the event.
- setTimestamp(timestamp)This method sets the- block.timestampfield in the event.
- setBlock(block)This method sets the- block.numberfield in the event.
- addEventLog(eventSignature, address, data, topics)This method adds a log to the- receipt.logsfield. The only mandatory argument is the- eventSignature,- addressargument is the zero address by default,- topicsis a spread list with the indexed event arguments, and- datais the empty string by default.- The - keccak256hash of the signature is added at the beginning of the- topicslist automatically.
- addAnonymousEventLog(address, data, topics)This method adds a log to the- receipt.logsfield.- addressargument is the zero address by default,- topicsis a spread list with the indexed event arguments, and- datais the empty string by default.
- addInterfaceEventLog(event, address, inputs)This method adds a log to the- receipt.logsfield.- eventis an- ethers.utils.EventFragmentinstance,- addressargument is the zero address by default,- inputsargument is an array of event parameter values and is an empty array by default.
- addInvolvedAddresses(addresses)This method adds a spread list of addresses to- addressesfield.
- addTraces(traceProps)This method adds a list of- Traceobjects at the end of- tracesfield in the event. The traces are created from the- tracePropsspread list.- TracePropsis a TS object with the following optional fields- { function, to, from, arguments, output, value, traceAddress }.
TestBlockEvent
This is a helper class for creating BlockEvents using the fluent interface pattern.
How to use it
import { TestBlockEvent } from "immunx-agent-tools/lib/test";
const blockEvent: BlockEvent = new TestBlockEvent().setHash(blockHash).setNumber(blockNumber);There are multiple methods you can use for creating the exact BlockEvent you want:
- setHash(blockHash)This method sets the- block.hashfield in the event.
- setParentHash(blockHash)This method sets the- block.parentHashfield in the event.
- setNumber(blockNumber)This method sets the- block.numberfield in the event.
- addTransactions(txns)This method adds the hashes of a spread list of transaction events at the end of- block.transactionsfield in the event.
- addTransactionsHashes(hashes)This method adds a hashes spread list to the end of- block.transactionsfield in the event.
runBlock
This is a helper function to simulate the execution of run block cli command when the sensor has implemented a handleTransaction and a handleBlock.
How to use it
import { runBlock } from "immunx-agent-tools/lib/test";
async myFunction(params) => {
  ...
  const findings: Findings[] = await runBlock(sensor, block, tx1, tx2, tx3, ..., txN);
  ...
};Parameters description:
- sensor: It is a JS object with two properties,- handleTransactionand- handleBlock.
- block: It is the- BlockEventthat the sensor will handle.
- tx#: These are the- TransactionEventobjects asociated with the- BlockEventthat the sensor will handle.
MockEthersProvider
This is a helper class for mocking the ethers.providers.Provider class.
Basic usage:
import { MockEthersProvider } from "immunx-agent-tools/lib/test";
import { createAddress } from "immunx-agent-tools";
import { utils, Contract } from "ethers";
const iface: utils.Interface =  new utils.Interface([
  "function myAwesomeFunction(uint256 param1, string param2) extenal view returns (unit8 id, uint256 val)"
]);
const address: string = createAddress("0xf00");
const data: string = createAddress("0xda7a");
const mockProvider: MockEthersProvider = new MockEthersProvider()
  .addCallTo(
    address, 20, iface,
    "myAwesomeFunction",
    { inputs: [10, "tests"], outputs: [1, 2000] },
  )
  .addStorage(address, 5, 15, utils.defaultAbiCoder.encode(["address"], [data]));How to use it
This mock provides some methods to set up the values that the provider should return:
- addCallTo(contract, block, iface, id, { inputs, outputs }). This method prepares a call to the- contractaddress at the specified- block, where- ifaceis the- ethers.utils.Interfaceobject relative to the contract,- idis the identifier of the function to call,- inputsare the parameters passed in the call and- outputsare the values the call should return.
- addCallFrom(contract, from, block, iface, id, { inputs, outputs }). Similar to- addCallTobut only the- fromwill be able to call the function.
- addStorage(contract, slot, block, result). This method prepares the value stored in the specific- slotof- contractaddress in the given- blockto be- result.
- addBlock(blockNumber, block). This method prepares the block with number- blockNumberto be- block.
- setLatestBlock(block). This method allows you to set up what the number of the latest block in the provider is.
- addSigner(addr). This function prepares a valid signer for the given address that uses the provider being used.
- addLogs(logs). This method allows you to add entries to the logs record that will be filtered in- getLogsif the filter specified wasn't yet added in- addFilteredLogs.
- setNetwork(chainId, ensAddress?, name?). This method allows you to set up the network information (- chainId,- ensAddressand- name) that will be returned when there's a call to- getNetwork.
- clear(). This function clears all the mocked data.
All the data you set in the provider will be used until the clear function is called.
MockEthersSigner
This is a helper class for mocking the ethers.providers.JsonRpcSigner class. This class extends MockEthersProvider.
Basic usage:
import { MockEthersProvider, MockEthersSigner } from "immunx-agent-tools/lib/test";
import { createAddress } from "immunx-agent-tools";
import { utils, Contract } from "ethers";
const iface: utils.Interface =  new utils.Interface([
  "function myAwesomeFunction(uint256 param1, string param2)"
]);
const address: string = createAddress("0xf00");
const contract: string = createAddress("0xda0");
const mockProvider: MockEthersProvider = new MockEthersProvider();
const mockSigner: MockEthersSigner = new MockEthersSigner(mockProvider)
  .setAddress(from)
  .allowTransaction(
    address, contract, iface,
    "myAwesomeFunction", [20, "twenty"]
    { confirmations: 42 }, // receipt data
  )How to use it
This mock provides some methods to set up the values that the signer should return:
- setAddress(address). This method sets the address that the signer can sign.
- allowTransaction(from, to, iface, id, inputs). This method prepares a txn sent to- toand signed from- from. The transaction is meant to call the method- idtaken from the- ifaceof the- tocontract passing the- inputsas parameters.- receiptwill be the receipt returned by the transaction.
- denyTransaction(from, to, iface, id, inputs, msg). Same conditions of- allowTransactionbut in this case the transaction will be reverted with- msgmessage.
All the data you set in the signer will be used until the clear function is called.
NetworkManager
This is a tool to help with storing data relative to the network the sensor will be running at.
Basic usage:
import { NetworkManager } from "immunx-agent-tools";
interface NetworkData {
  address: string;
  num: number;
}
const data: Record<number, NetworkData> = {
  // ChainID 1
  1: {
    address: "address1",
    num: 1,
  },
  42: {
    address: "address42",
    num: 2,
  },
};
const provider = getEthersProvider();
const networkManager = new NetworkManager(data);
await networkManager.init(provider);
networkManager.get("address"); // "address1" if the ChainID is 1, "address42" if the ChainID is 42How to use it
- NetworkManager(networkData, chainId?): Sets the network data and creates a- NetworkManagerinstance. If- chainIdis specified, it won't need- NetworkManager.init()to be initialized. Throws an error if there is no entry for- chainIdin- networkData.
- getNetworkMap(): Gets the network map passed as argument in the constructor as read-only.
- getNetwork(): Gets the instance's active ChainID.
- setNetwork(chainId): Sets the instance's active ChainID. Throws an error if there is no entry for- chainIdin- networkData.
- init(provider): Retrieves network data from the provider and sets the active ChainID. Throws an error if there is no entry for that ChainID in- networkData.
- get(key): Gets the value of the field- keyin the active network's data record. Throws an error if- NetworkManagerwas not yet initialized, i.e. the ChainID was not specified in the constructor and- NetworkManager.init()or- NetworkManager.setNetwork()were not called.
ProviderCache
This is a class that can create a proxy to a provider which then caches call results and avoids cached calls being repeated both later and in the same block or transaction.
Basic usage:
import { ProviderCache, createAddress } from "immunx-agent-tools";
import { ethers, getEthersProvider } from "immunx-agent";
const provider = getEthersProvider();
const cachedProvider = ProviderCache.createProxy(provider);
const address = createAddress("0x0");
const iface: ethers.ContractInterface = [];
// the cached provider can be used as a regular provider
const contract = new ethers.Contract(address, iface, cachedProvider);How to use it
- ProviderCache.createProxy(provider, cacheByBlockTag?): Creates a proxy to a provider that caches call results. If- cacheByBlockTagis set to- false, then the call is cached without taking the block tag into account, useful for cases where some data can't change between blocks. By default,- cacheByBlockTagis set to- true, thus the call result cache takes into account the block tag in which it's called.
- ProviderCache.clear(): Clears the internal cache.
- ProviderCache.set(options): Sets options specified by- options.- options.blockDataCacheSize?: If it's defined, the block data cache (used when- cacheByBlockTagis- true) is cleared if it exists and its max size is changed to the value specified.
- options.immutableDataCacheSize?: if it's defined, the immutable data cache (used when- cacheByBlockTagis- false) is cleared if it exists and its max size is changed to the value specified.
 
CachedContract
This is a shortcut class that extends ethers.Contract but uses a cached provider from ProviderCache. Creating a CachedContract by calling new CachedContract(address, iface, provider, cacheByBlockTag?) is equivalent to creating an ethers.Contract by calling new ethers.Contract(address, iface, ProviderCache.createProxy(provider, cacheByBlockTag?)). There's also some utility methods.
Basic usage:
import { CachedContract, createAddress } from "immunx-agent-tools";
import { getEthersProvider } from "immunx-agent";
const provider = getEthersProvider();
const address = createAddress("0x0");
const iface: ethers.ContractInterface = [];
const cachedContract = new CachedContract(address, iface, provider, true);
// it can also be created from an existing ethers.Contract
const contract = new ethers.Contract(address, iface, provider);
const cachedContractfromContract = CachedContract.from(contract, true);How to use it
- CachedContract(addressOrName, contractInterface, signerOrProvider, cacheByBlockTag?): Creates a new- CachedContractinstance with address- addressOrName, interface- contractInterfaceand a- ProviderCacheproxy to the provider- providerwith the specified- cacheByBlockTagoption. By default,- cacheByBlockTagis set to- true. Throws if- providertype is not an extension of- ethers.providers.BaseProvider.
- from(contract, cacheByBlockTag?): Creates a new- CachedContractinstance from- contract, an- ethers.Contractinstance by collecting its fields and calling the constructor. A wrapper to- new CachedContract(contract.address, contract.interface, contract.provider, cacheByBlockTag?). By default,- cacheByBlockTagis set to- true. Throws if- contracthas a signer.
- clear(): A shortcut to- ProviderCache.clear(). Clears the- ProviderCacheglobal cache.
MulticallProvider
This is an ethers provider-like interface built on top of ethers-multicall, but it also supports specifying a block
tag for a call, using Multicall2 features and making grouped calls.
The calls are decoded using ethers, so each return data has the same structure as a call made by itself through an ethers.Contract.
Supported chains (by default):
- Ethereum Mainnet
- Ropsten Testnet
- Rinkeby Testnet
- Görli Testnet
- Kovan Testnet
- Binance Smart Chain
- Binance Smart Chain Testnet
- Gnosis
- Huobi ECO Chain Mainnet
- Polygon Mainnet
- Fantom Opera
- Arbitrum One
- Avalanche
- Mumbai Testnet
Other chains can also be supported by finding a deployed Multicall2 contract address and calling
MulticallProvider.setMulticall2Addresses({ [chainId]: multicall2Address }). Default addresses can also be overriden.
Basic usage:
import { getEthersProvider } from "immunx-agent";
import { MulticallProvider, MulticallContract, createAddress } from "immunx-agent-tools";
const provider = getEthersProvider();
const multicallProvider = new MulticallProvider(provider);
const token = new MulticallContract(createAddress("0x0"), [
  "function balanceOf(address account) external view returns (uint256)",
  "function allowance(address owner, address spender) external view returns (uint256)",
]);
async function initialize() {
  // fetches the provider network and loads an appropriate Multicall2 address
  // throws if the network is not supported
  await multicallProvider.init();
}
async function getBalances() {
  const addresses = [createAddress("0x1"), createAddress("0x2"), createAddress("0x3")];
  const calls = addresses.map(address => token.balanceOf(address));
  const blockTag = 1;
  const [success, balancesAll] = await multicallProvider.all(calls, blockTag); // [success, [balance0, balance1, balance2]]
  // or
  const balancesTryAll = await multicallProvider.tryAll(calls, blockTag); // [{ success, returnData: balance0 }, { success, returnData: balance1 }, { success, returnData: balance2 }]
  // or
  const [successGrouped, balancesGrouped] = await multicallProvider.groupAll(
    addresses.map(address => [token.balanceOf(address), token.allowance(address, createAddress("0x4"))])
  ); // [success, [[balance0, allowance0], [balance1, allowance1], [balance2, allowance2]]]
  // or
  const balancesGroupTryAll = await multicallProvider.groupTryAll(
    addresses.map(address => [token.balanceOf(address), token.allowance(address, createAddress("0x4"))])
  ); // [
  // [{ success, returnData: balance0 }, { success, returnData: allowance0 }],
  // [{ success, returnData: balance1 }, { success, returnData: allowance1 }],
  // [{ success, returnData: balance2 }, { success, returnData: allowance2 }],
  //]
}How to use it
- MulticallProvider(provider, chainId?): Creates a- MulticallProviderinstance through an ethers provider- provider. If- chainIdis specified, it's not necessary to call- init()before making calls.
- MulticallProvider.setMulticall2Addresses(addresses): Allows overriding and adding support to more networks by specifying a valid- Multicall2contract address to it.
- init(): Fetches the provider's chain ID and and loads a- Multicall2contract address. If there's no known address for that network, it throws an error.
- all(calls, blockTag?, batchSize?): Performs the calls in- blockTagwith- batchSizesized multicalls and requires success of all of them. By default,- batchSizeis- 50.
- tryAll(calls, blockTag?, batchSize?): Performs the calls in- blockTagwith- batchSizesized multicalls and doesn't require their success, returning a flag for each of them that indicates whether they were successful or not. By default,- batchSizeis- 50.
- groupAll(calls, blockTag?, batchSize?): Works in the same way as- all(), but allows specifying groups of calls (e.g.- [[call0, call1], [call2, call3]]) and keeps that same structure in the returned data.
- groupTryAll(calls, blockTag?, batchSize?): Works in the same way as- tryAll(), but allows specifying groups of calls (e.g.- [[call0, call1], [call2, call3]]) and keeps that same structure in the returned data.