Action Functions, Events, and Triggers
This guide presents the three core concepts of Web3 Actions: triggers, events, and functions. You’ll learn how they work and how to use them to build your own Web3 Actions.
The three core components that make up a Web3 Action are:
- Functions: Custom JavaScript or TypeScript code you want to execute when an external event occurs. This is the core of your automation.
- Triggers: Pre-defined external events that your Web3 Action is configured to listen for. When the event occurs, the trigger instructs Tenderly to execute your custom code (function).
- Events (Trigger Types): Pre-defined external events that you can listen for by setting a trigger. When the event occurs, the trigger will call your custom code (functions).
Execution Types
Web3 Actions in Tenderly can be executed in two modes: Sequential and Parallel.

Sequential Execution
In sequential execution, actions are executed one by one in the order they were invoked. This is the default mode of execution. In this mode, each action waits for the previous action to complete before it begins execution. Here’s an example of an action configuration for a sequential execution:
our-org/our-cool-project:
runtime: v2
sources: actions
specs:
bestActionEver:
description: Does the best thing ever
+ execution_type: sequential
function: very/organized/file:bestActionEver
trigger:
...Parallel Execution
In parallel execution, actions are executed in parallel, which leads to higher throughput. In this mode, the order of execution is not guaranteed and there may be possible race conditions with action storage. Here’s an example of an action configuration for a parallel execution:
our-org/our-cool-project:
runtime: v2
sources: actions
specs:
bestActionEver:
description: Does the best thing ever
+ execution_type: parallel
function: very/organized/file:bestActionEver
trigger:
...Action functions
Web3 Action functions refer to the custom code written as standard JavaScript or TypeScript functions, but they must adhere to some rules.
- Functions must be asynchronous, returning
Promise<void> - Functions accept two parameters:
contextandevent - Functions must be a named export from the file
- Functions can be placed in any file under the actions root directory
Learn about the Web3 Actions project and file structure.
When a Web3 Action function is deployed, the Tenderly runtime will pass the following arguments during invocation:
- The
contextparameter that holds access to Storage and Secrets. - The
eventparameter that is an object with information answering the question “what just happened?”. This parameter contains data specific to the trigger type the Web3 Action function is listening for. The section Specifying triggers for external events goes into detail about external events.
The execution of Web3 Action functions is limited to 30 seconds. If your function takes longer than 30 seconds to execute, it will be terminated.
Here’s an example of a Web3 Action function written in TypeScript. The event parameter can be any one of the supported trigger types (events).
// File: actions/myCoolTsFile.ts
import {
ActionFn, Context,
Event, BlockEvent, PeriodicEvent, TransactionEvent, WebhookEvent
} from "@tenderly/actions";
// importing ethers available in Tenderly Runtime
import { ethers } from "ethers";
export const awesomeActionFunction: ActionFn = async (context: Context, event: Event) => {
// cast the event parameter to an appropriate type, based on the Trigger
// this function subscribed to.
// Of course, only one of these casts makes sense
const blockEvent = event as BlockEvent;
const periodicEvent = event as PeriodicEvent;
const transactionEvent = event as TransactionEvent;
const webhookEvent = event as WebhookEvent;
...
}Available libraries for dashboard-based actions
The Tenderly runtime comes pre-bundled with several javascript libraries that you can import when creating Web3 Actions via the Tenderly Dashboard. Available libraries include:
To import any of these libraries, use the require() function as you would with a standard npm-based project (without ES6).
Example import for Axios looks like this:
const axios = require('axios');Using npm libraries for CLI-based Web3 Actions
The project created by Tenderly CLI is in fact an npm module. You can install any npm package from your actions root folder.
External events and trigger types
When an external event happens, it triggers the execution of your custom code (functions). You can choose between four external events to listen for:
- Block event: A block is mined on a selected network.
- Periodic event: This trigger happens when certain time intervals pass or based on CRON expressions.
- Webhook event: An HTTP request is posted to the webhook URL (the Web3 Action exposes a webhook)
- Transaction event: A transaction matching given filter criteria is executed on a selected network.
When defining a Web3 Action, you’ll be effectively defining triggers that react to external events of interest for your project. In the context of trigger specification, we’ll be using the term trigger type.
You should write separate functions for each trigger type. Using the same function for different trigger types is not recommended.
Subscribing Web3 Action functions to events
In addition to defining your action function in JavaScript, you also need to provide the trigger configuration, which tells Tenderly what triggers it — the external event your function subscribes to.
When creating Web3 Actions via the Tenderly Dashboard, the creation flow in the UI will handle this part for you. Read the Dashboard Quickstart guide.
When working with code-based Web3 Actions, functions and their triggers must be defined in the tenderly.yaml file, which is generated by Tenderly CLI. Read the CLI Quickstart guide.
Example
In the example below, we’re declaring a Web3 Action called bestActionEver and referencing the function awesomeActionFunction that is exported from the actions/myCoolTsFile.ts file. This is the function that Tenderly will call when the Web3 Action gets triggered. The execution is controlled via execution_type, in this case it’s set to parallel.
account_id: ""
actions:
our-cool-org/our-cool-project:
runtime: v1
sources: actions
specs:
bestActionEver:
execution_type: parallel # or sequential
description: Does the best thing ever
function: myCoolTsFile:awesomeActionFunction
trigger: # trigger declaration start
type: ... # type
... # configuration map (external event specific)Specifying triggers for external events
In this section, we’ll examine the trigger declaration. For more information on writing tenderly.yaml file, reference this guide.
The trigger type and corresponding configurations are defined in the tenderly.yaml file. You first must define the trigger object, which has two mandatory properties:
- The
typeproperty, that specifies the trigger type, which can be:periodic | webhook | block | transaction - An object with a configuration specific to the selected trigger type
Below is a detailed explanation of the trigger-specific configurations for each trigger type.
Periodic event
A periodic event is used when you want your Web3 Action to be triggered at specific time intervals. It carries time: the time of invocation.
type PeriodicEvent = {
time: Date;
};The trigger declaration has the periodic type. It can be interval- or cron-based:
trigger:
type: periodic
periodic:
interval: 5mThe interval property can take any of the following values: 5m | 10m | 15m | 30m | 1h | 3h | 6h | 12h | 1d
Use the CRON-based periodic trigger if you need more granular control over when your Web3 Action gets executed:
trigger:
type: periodic
periodic:
cron: '5 */1 * * *'The cron property can be any valid CRON string.
Webhook event
Webhook-based Web3 Actions expose a custom webhook URL, making it possible to trigger them from external systems using a simple HTTP POST request.
The corresponding trigger type holds two properties: the time of invocation and any payload (a JSON object). Tenderly doesn’t perform any validation or inspections on your payload.
type WebhookEvent = {
time: Date;
payload: any;
};A trigger configuration example:
trigger:
type: webhook
webhook:
authenticated: trueIf authenticated is set to true, you must include the Tenderly Access Token with your request to be able to run the Web3 Action as the value of x-access-key. You can find the cURL of the exposed webhook in the Web3 Action overview in the Tenderly Dashboard.
curl -X POST \
-H "x-access-key: $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
https://api.tenderly.co/api/v1/actions/7acc7db2-00d6-4872-b2d2-d9430bd8fe7b/webhook \
-d '{"foo": "bar"}'Block event
The block event is used when you want to listen for the mining of blocks on one or several networks. You can make your Web3 Action “block-periodic” by specifying the number of mined blocks between two consecutive invocations.
type BlockEvent = {
blockHash: string;
blockNumber: number;
network: string;
};When declaring the block trigger, you can specify the following properties:
network: a single value or a list of network IDs you’re interested inblocks: the number of mined blocks between two consecutive executions of the Web3 Action. For example, execute the Web3 Action on every 100th block mined.
trigger:
type: block
block:
network:
- 1
- 8453
blocks: 100Transaction event
The transaction trigger type allows you to listen for specific transactions as they get executed on-chain.
To listen for transactions from a smart contract, the smart contract must be verified and added to your project in the Tenderly Dashboard. The CLI will throw a warning if this is not the case.
You can specify different conditions in the form of filters to zero in on transactions of interest. Your custom code will be invoked with that exact transaction payload passed through the event parameter.
type TransactionEvent = {
blockHash: string;
blockNumber: number;
from: string;
hash: string;
network: string;
to?: string;
logs: Array<{
address: string;
data: string;
topics: string[];
}>;
input: string;
value: string;
nonce: string;
gas: string;
gasUsed: string;
cumulativeGasUsed: string;
gasPrice: string;
gasTipCap: string;
gasFeeCap: string;
transactionHash: string;
};When listening for transaction events, you can also specify the following settings:
- Transaction mining status:
mined(the transaction has been mined) orconfirmed10(10 blocks are confirmed since the block that contains the transaction) - Filters: A list of conditions involving transaction payload using the
filterslist. Only transactions that match filters will trigger the execution of your code.
Filtering transactions
The filters list allows you to define triggering criteria using transaction properties to match specific transactions.
Each filter in the list is an object where all fields are AND-ed together. Every condition in the filter must be true for it to match. The filters list itself is OR-ed: a transaction matches if it satisfies any single filter in the list.
match = anyOf(
(network AND status AND from AND eventEmitted), # Filter 1
(network AND to) # Filter 2
)
Below is a list of all available filters you can combine to build criteria for transactions that will trigger the execution of your Web3 Action.
| Filter | Description |
|---|---|
*mandatory | |
| network* | The network ID from which the transaction originated |
| status | The transaction status: fail or success |
| from** | The address of the transaction originator |
| to** | The address of the transaction recipient |
| eventEmitted** | An event of a specific type, emitted by a particular contract |
| logEmitted** | Logs emitted by a particular contract, by matching the prefix of the raw log entry |
| function** | Direct call to the given function. Not applicable to internal calls between the contracts |
| value | The value that is transferred within the transaction in wei |
| gasUsed | The amount of gas used by the transaction in wei |
| gasLimit | The gas limit set for the transaction in wei |
| fee | The transaction fee in wei |
| contract | A smart contract involved in the transaction (see contract filter) |
A trigger must contain network and at least one of the following filters: from, to,
function, eventEmitted or logEmitted
An example of a transaction on Ethereum mainnet or Base sent to 0x236..fd62.
failedTransactionToMyContract:
execution_type: parallel
description: When sent to my contract fails
function: observingTransactions:failedAttempts
trigger:
type: transaction
transaction:
status:
- mined
filters:
# Transaction must be from Ethereum mainnet
- network: 1
# Transaction must have failed
status: fail
# Transaction must have been sent to this address
to: 0x2364259ACD20Bd2A8dEfDc628f4099302449fd62
# Or from Base
- network: 8453
# Transaction must have failed
status: fail
# Transaction must have been sent to this address
to: 0x2364259ACD20Bd2A8dEfDc628f4099302449fd62In this case, we’re interested only in transactions that are mined, matching either of these two filters:
- Filter 1: network 1 (Ethereum) AND status fail AND to
0x2364...fd62 - Filter 2: network 8453 (Base) AND status fail AND to
0x2364...fd62
Since we didn’t filter by any particular sender, all transactions coming to the mentioned recipient will result in the Web3 Action being triggered.
Filtering transactions by transaction fields
Among the common transaction fields, you can filter by the following properties having specific values.
from- filtering on a specific sender. Can be a single address or a list of addresses (OR-ed). Optional.to- filtering on a specific receiver. Can be a single address or a list of addresses (OR-ed). Optional.status- filtering on transaction status:successorfail. Can be a single value or a list (OR-ed). Optional.network- filtering by network. Can be a single chain ID or a list of chain IDs (OR-ed). Mandatory.contract.address- filtering only transactions involving the contract with this specific address. Optional.
All address fields must be 0x-prefixed, 40 hex characters (e.g. 0xf63c48626f874bf5604D3Ba9f4A85d5cE58f8019).
filters:
- network: # list of networks (OR-ed)
- 1
- 8453
status: # list of statuses (OR-ed)
- success
- fail
from: # list of sender addresses (OR-ed)
- 0x7ebB3Dca1C281b23D5B73175f10cA5A0a309B01F
- 0xD3a02149A236b2547Cc3C897Fb41C1a962f881AE
to: # list of recipient addresses (OR-ed)
- 0x003b3625cDcb5958E9709F4Ba8E340Cb0783DeaE
- 0x26997bd8473E0Dd0b37eB1711B7c1eE2354d78e4You can also query numerics related to exchanged value and gas, using relational operators. All of the following are in wei:
valuegasUsedgasLimitfee
Numeric fields accept either a single comparison object or a list of comparison objects. When a list is provided, comparisons are OR-ed. Within a single comparison object, all operators are AND-ed.
You can use any of the common comparison operators: eq, gte, gt, lt, lte. You can also use the not boolean flag to negate a comparison.
| Operator | Meaning |
|---|---|
eq | Equal to |
gt | Greater than |
gte | Greater than or equal to |
lt | Less than |
lte | Less than or equal to |
not | Boolean. Set to true to negate the comparison (e.g. match when value is not in range) |
Below is an example of a filter demonstrating filters using scalar values in the transaction payload.
filters:
- network: 1
to: 0x003b3625cDcb5958E9709F4Ba8E340Cb0783DeaE
# Single comparison object (operators AND-ed within)
value:
gte: 100
lte: 1000
# List of comparison objects (OR-ed between entries)
gasLimit:
- lt: 100
- gt: 1000
gasUsed:
eq: 9999
fee:
- lte: 100
- gte: 1000Using the not flag to negate a comparison:
filters:
- network: 1
to: 0x003b3625cDcb5958E9709F4Ba8E340Cb0783DeaE
# Matches when value is NOT between 100 and 1000
value:
gte: 100
lte: 1000
not: trueFiltering transactions using emitted EVM Events
Configuring Web3 Actions to listen for EVM events emitted by a transaction execution allows you to respond to significant changes logged by these events. You can achieve this by using the eventEmitted operator, which supports the following nested properties:
contract(mandatory): To specify the origin of the event.contract.address: The address of the contract that emitted the event.contract.invocation: Controls how the contract was called:direct,internal, orany(default). See contract filter for details.
name: The name of the event (requires a verified contract).id: The topic hash of the event signature (alternative toname). Useful for unverified contracts where the ABI is not available. For example, theTransfer(address,address,uint256)event has the topic hash0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef.not: Boolean. Set totrueto negate the entire event match, including parameters (trigger when the event is not emitted with the specified parameters).parameters: A list of conditions on decoded event argument values. All conditions are AND-ed: every condition must match. Each entry supports:name(required): The name of the event argument.string: String comparison. Accepts a plain string for exact match, or an object withexactandnotfields. See StringComparison for details.int: Integer comparison. Supports the same operators as numeric filters (eq,gt,gte,lt,lte). All operators within a singleintblock are AND-ed. Use thenotflag to negate the combined condition. See IntComparison for details.
There is no OR within a single eventEmitted entry’s parameters. All conditions are AND-ed.
For OR logic on the same parameter, use multiple eventEmitted entries (which are OR-ed in the outer list).
To use the eventEmitted filter with name or parameters, the contract must be verified and added to the project in the
Tenderly Dashboard. Use id (topic hash) for unverified contracts.
For the complete field reference, see EventFilter.
Below is an example of a Web3 Action that will be invoked when either TxSubmission or TxConfirmation event is emitted when the smart contract is executed at the address 0x418d..9d45 on Sepolia.
txSubmittedOrConfirmed:
description: When any of TxSubmission and TxConfirmation is emitted
execution_type: parallel
function: myCoolJsFile:myCoolFunction
trigger:
type: transaction
transaction:
status:
- mined
filters:
- network: 11155111
eventEmitted:
contract:
address: 0x418ebb95eaa40c119408143056cad984c6129d45
name: TxSubmission
- network: 11155111
eventEmitted:
contract:
address: 0x418ebb95eaa40c119408143056cad984c6129d45
name: TxConfirmationHere is an example filtering by decoded event parameters. All parameter conditions are AND-ed:
filters:
- network: 8453
eventEmitted:
contract:
address: 0x833589fcd6edb6e08f4c7c32d4f71b54bda02913
name: Transfer
parameters:
- name: from
string: "0x8F362C3dd5301Ce1b43ea3A278E3f12c1807C271"
- name: value
int:
gte: 1000000 # >= 1 USDC (6 decimals)Numeric range with negation on parameters:
filters:
- network: 1
eventEmitted:
contract:
address: 0x833589fcd6edb6e08f4c7c32d4f71b54bda02913
name: Transfer
parameters:
- name: value
int:
gte: 100
lte: 1000
not: true # matches when value < 100 OR value > 1000For OR logic on parameters, use multiple eventEmitted entries:
filters:
- network: 1
eventEmitted:
# Entry 1: value < 100
- contract:
address: 0x833589fcd6edb6e08f4c7c32d4f71b54bda02913
name: Transfer
parameters:
- name: value
int:
lt: 100
# Entry 2: value > 1000 (OR-ed with entry 1)
- contract:
address: 0x833589fcd6edb6e08f4c7c32d4f71b54bda02913
name: Transfer
parameters:
- name: value
int:
gt: 1000Using not on the event itself to negate the entire match:
filters:
- network: 1
eventEmitted:
contract:
address: 0x833589fcd6edb6e08f4c7c32d4f71b54bda02913
name: Transfer
parameters:
- name: from
string: "0x0000000000000000000000000000000000000000"
not: true # matches when no Transfer event from the zero address is foundFiltering transactions by emitted logs
Another way to listen for EVM events is to query them by the log topic.
The logEmitted filter can be a single object or a list (entries are OR-ed). It supports the following properties:
startsWith(mandatory): A list of topic values to match against the transaction’s log topics. Each entry is compared by exact equality (case-insensitive) to the corresponding topic. In practice these are full 32-byte topic hashes (0x-prefixed, 64 hex chars), but the field accepts any valid hex string.contract.address(optional): The source of the logs. When specified, only logs from this contract are matched.matchAny: Boolean. Whentrue, matching any single topic fromstartsWithis sufficient. Whenfalse(default), all topics must match.not: Boolean. Set totrueto negate the entire log match (trigger when the log is not emitted).
Using logEmitted enables you to use the event topic prefix in its raw form to filter for events.
This is useful if you’re setting up a Web3 Action on an unverified contract.
For the complete field reference, see LogFilter.
Below is an example of a Web3 Action that will be invoked when a transaction contains log entries starting with the given prefixes.
txEmittedSomeLogs:
description: When particular logs are emitted
function: observingTransactions:someLogsEmittedAction
execution_type: parallel
trigger:
type: transaction
transaction:
status:
- mined
filters:
- network: 11155111
logEmitted:
# Transaction must have emitted a log entry
contract:
address: 0x418ebb95eaa40c119408143056cad984c6129d45
startsWith:
# and topics of the log entry must start with either one of these
- 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef
- 0x0000000000000000000000000000000000000000000000000000000000000000Filtering transactions involving a specific contract
To filter for transactions that call a specific contract during their execution, you can use the optional contract filter.
The contract filter supports the following properties:
address: The address of the contract.invocation: Controls how the contract was called. Possible values:direct: the contract was called directly by the transaction sender (EOA). Use this when you only care about top-level calls to the contract.internal: the contract was called internally by another contract during execution (e.g. viaCALL,DELEGATECALL, orSTATICCALLopcodes). Use this to detect when your contract is invoked as part of a multi-contract interaction.any: matches both direct and internal calls. This is the default.
The invocation field is available wherever a contract object is used, including inside eventEmitted filters. For example, you can use invocation: internal on an eventEmitted contract to only match events emitted during internal calls. See Contract for the schema.
invocation is not supported on logEmitted. Only contract.address is used to restrict log matching; the invocation type is ignored.
In the example below, the first filter matches any successful transaction sent to 0x2364...fd62 on Sepolia. The second filter is more restrictive: only transactions that also involve an internal call to contract 0xad88...80d6 will trigger the Web3 Action.
contractInvocationAction:
description: When my contract is called directly or internally
execution_type: parallel
function: observingTransactions:contractCalls
trigger:
type: transaction
transaction:
status:
- mined
filters:
# Any successful transaction sent to this address
- network: 11155111
status: success
to: 0x2364259ACD20Bd2A8dEfDc628f4099302449fd62
# Only transactions that internally call this contract
- network: 11155111
status: success
to: 0x2364259ACD20Bd2A8dEfDc628f4099302449fd62
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6
invocation: internalFiltering transactions by function call
To filter for transactions that directly call a specific function, you can use the optional function filter. It requires the address of the contract contract.address and either the name or signature of the function.
The function filter supports the following properties:
contract(mandatory): To specify the target contract.contract.address: The address of the contract being called.contract.invocation: Controls how the function must be called:any(default — both direct and internal),direct(EOA-initiated only), orinternal(sub-calls only). See contract filter for details.
name: The name of the function (requires a verified contract).signature: The 4-byte function selector (e.g.0xa9059cbb). Useful for unverified contracts where the ABI is not available.not: Boolean. Set totrueto negate the entire function match, including parameters (trigger when the function is not called with the specified parameters).parameters: A list of conditions on decoded function input argument values. All conditions are AND-ed: every condition must match. Each entry supports:name(required): The name of the function argument.string: String comparison. Accepts a plain string for exact match, or an object withexactandnotfields. See StringComparison for details.int: Integer comparison. Supports the same operators as numeric filters (eq,gt,gte,lt,lte). All operators within a singleintblock are AND-ed. Use thenotflag to negate the combined condition. See IntComparison for details.
There is no OR within a single function entry’s parameters. All conditions are AND-ed.
For OR logic on the same parameter, use multiple function entries (which are OR-ed in the outer list).
To use the function filter with name or parameters, the contract must be verified and added to the project in the
Tenderly Dashboard. Use signature (4-byte selector) for unverified contracts.
You can specify function as a single object or as a list. When a list is provided, entries are OR-ed. The filter matches if any function in the list is called. For the complete field reference, see FunctionFilter.
Below is an example of a Web3 Action trigger responding to any call (direct or internal) to the verySpecialFunction of the contract deployed at 0xad88...80d6.
failedTransactionToMyContract:
description: When sent to my contract fails
execution_type: parallel
function: observingTransactions:failedAttempts
trigger:
type: transaction
transaction:
status:
- mined
filters:
# Transaction must be from Ethereum mainnet
- network: 1
# Transaction must have failed
status: fail
function:
# the function verySpecialFunction must have been called (direct or internal)
name: verySpecialFunction
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6You can also match by function selector and use multiple function entries:
filters:
- network: 1
function:
# Match by 4-byte selector (useful for unverified contracts)
- signature: "0xa9059cbb"
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6
# Match by name (requires verified contract)
- name: approve
not: true # negate: trigger when approve is NOT called
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6Here is an example filtering by decoded function input parameters. All parameter conditions are AND-ed:
filters:
- network: 1
function:
name: transfer
contract:
address: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
parameters:
- name: to
string: "0x8F362C3dd5301Ce1b43ea3A278E3f12c1807C271"
- name: amount
int:
gte: 1000000 # >= 1 USDC (6 decimals)String negation — trigger when the recipient is not a specific address:
filters:
- network: 1
function:
name: transfer
contract:
address: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
parameters:
- name: to
string:
exact: "0xa59a97Ab934aE02A1E0561ea0f4F6C275Fc148B4"
not: true # matches when 'to' is NOT this addressNumeric range with negation on parameters:
filters:
- network: 1
function:
name: transfer
contract:
address: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
parameters:
- name: amount
int:
gte: 100
lte: 1000
not: true # matches when amount < 100 OR amount > 1000For OR logic on parameters, use multiple function entries:
filters:
- network: 1
function:
# Entry 1: amount < 100
- name: transfer
contract:
address: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
parameters:
- name: amount
int:
lt: 100
# Entry 2: amount > 1000 (OR-ed with entry 1)
- name: transfer
contract:
address: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
parameters:
- name: amount
int:
gt: 1000To match only internal calls — for example, a transfer triggered by a router or flash-loan provider rather than directly by a user — set invocation: internal. This example triggers only when another contract internally calls transfer on USDC for at least 1,000 USDC:
filters:
- network: 1
function:
name: transfer
contract:
address: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
invocation: internal # sub-calls only — excludes direct EOA calls
parameters:
- name: amount
int:
gte: 1000000000 # >= 1,000 USDC (6 decimals)Complete transaction trigger reference
The following annotated YAML block shows every available field for transaction triggers in one place. Use it as a quick reference when building your trigger configuration. For type definitions, see the Trigger YAML schema reference below.
trigger:
type: transaction
transaction:
status:
- mined # or confirmed10
# At least one filter must match (OR logic).
# Within a filter, all fields are AND-ed.
filters:
- network: 1 # mandatory (single or list)
status: success # fail | success (single or list)
from: "0xDB6Fd484cFA46eeB73c71165C6C004B50309BbE1" # single or list of addresses
to: "0x2364259ACD20Bd2A8dEfDc628f4099302449fd62" # single or list of addresses
# --- Contract filter ---
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6
invocation: any # any | direct | internal
# --- Numeric comparisons (all in wei) ---
# Each can be a single object or a list (OR-ed between entries).
# Within a single object, all operators are AND-ed.
value:
gte: 1000000000000000000 # >= 1 ETH
not: false # set true to negate
gasUsed:
eq: 21000
gasLimit: # list form (OR-ed)
- lt: 100
- gt: 1000
fee:
- lte: 100
- gte: 1000
# --- Event filter (single object or list, OR-ed) ---
eventEmitted:
- contract:
address: 0x833589fcd6edb6e08f4c7c32d4f71b54bda02913
name: Transfer # match by event name (verified contract)
# id: "0xddf252..." # alternative: match by topic hash
not: false # negate entire event match
parameters: # filter by decoded event arguments (AND-ed)
- name: from
string: "0x8F362C3dd5301Ce1b43ea3A278E3f12c1807C271"
- name: value
int:
gte: 1000000 # >= 1 USDC (6 decimals)
# --- Log filter (single object or list, OR-ed) ---
logEmitted:
- startsWith:
- "0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef"
contract:
address: 0x418ebb95eaa40c119408143056cad984c6129d45
matchAny: false # true = match any single topic
not: false # negate entire log match
# --- Function filter (single object or list, OR-ed) ---
function:
- name: transfer # match by name (verified contract)
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6
invocation: any # any (default) | direct | internal
not: false # negate entire function + parameters match
parameters: # filter by decoded function arguments (AND-ed)
- name: to
string: "0x8F362C3dd5301Ce1b43ea3A278E3f12c1807C271" # short form
- name: from
string: # long form (with negation)
exact: "0xa59a97Ab934aE02A1E0561ea0f4F6C275Fc148B4"
not: true # negate this string match
- name: amount
int:
gte: 1000000 # >= 1 USDC (6 decimals)
# name and signature are mutually exclusive — each entry uses one or the other, not both
- signature: "0xa9059cbb" # match by 4-byte selector (0x + 8 hex chars)
not: true # negate this function match
contract:
address: 0xad881d3d06c7f168715b84b54f9d3e1ff27b80d6Trigger YAML schema reference
The schema below describes the structure and types of every field accepted by the trigger configuration. Use it alongside the annotated example above when building your tenderly.yaml.
Notation: ? = optional, | = one of, [] = list. Fields marked “single or list” accept either a scalar value or a YAML list.
All trigger types
trigger:
type: periodic | webhook | block | transaction
periodic: # when type = periodic
interval: string? # "5m" | "10m" | "15m" | "30m" | "1h" | "3h" | "6h" | "12h" | "1d"
cron: string? # standard cron expression (mutually exclusive with interval)
webhook: # when type = webhook
authenticated: bool? # default: true
block: # when type = block
network: Network # single chain ID or list of chain IDs
blocks: int # run every N blocks (must be > 0)
transaction: # when type = transaction
status: TransactionStatus[] # "mined" | "confirmed10"
filters: TransactionFilter[] # at least one filter requiredTransactionFilter
# All fields within a filter are AND-ed.
# Multiple filters in the list are OR-ed.
TransactionFilter:
network: Network # required. single chain ID or list (OR-ed)
status: Status # optional. single or list: "success" | "fail" (OR-ed)
from: Address # optional. single or list of addresses (OR-ed)
to: Address # optional. single or list of addresses (OR-ed)
value: IntComparison # optional. single object or list (OR-ed)
gasUsed: IntComparison # optional. single object or list (OR-ed)
gasLimit: IntComparison # optional. single object or list (OR-ed)
fee: IntComparison # optional. single object or list (OR-ed)
contract: Contract # optional. shared default — applied to function/eventEmitted that omit their own contract
function: FunctionFilter # optional. single object or list (OR-ed)
eventEmitted: EventFilter # optional. single object or list (OR-ed)
logEmitted: LogFilter # optional. single object or list (OR-ed)
# constraint: at least one of from, to, function, eventEmitted, or logEmitted is requiredAt least one of from, to, function, eventEmitted, or logEmitted must be present in each filter.
The top-level contract field acts as a shared default for the filter. It is automatically applied to any function or eventEmitted entry that does not specify its own contract, letting you avoid repeating the same address. It does not apply to logEmitted.
Shared types
Address: string # 0x-prefixed, 40 hex characters (case-insensitive)
Network: int | int[] # chain ID(s)
Status: string | string[] # "success" | "fail"
Contract:
address: Address # required
invocation: string? # "any" (default) | "direct" | "internal"
IntComparison: # single object or list of objects (OR-ed)
eq: int? # equal to
gt: int? # greater than
gte: int? # greater than or equal
lt: int? # less than
lte: int? # less than or equal
not: bool? # negate the combined condition
StringComparison: # accepts a plain string OR an object
# Short form: string: "0xdead..." → exact match, not = false
# Long form: string: { exact: "0xdead...", not: true }
exact: string? # exact match (case-insensitive)
not: bool? # negate the matchFunctionFilter
FunctionFilter: # single object or list (OR-ed)
contract: Contract # required
name: string? # function name (requires verified contract)
signature: string? # 4-byte selector, 0x-prefixed, 8 hex chars
not: bool? # negate entire function + parameters match
parameters: ParameterCondition[]?
# note: name and signature are mutually exclusive (use one or the other)
# note: parameters requires a verified contract (name must be set, not signature)
ParameterCondition: # all conditions in the list are AND-ed
name: string # required. function argument name
string: StringComparison? # string comparison (plain string or { exact, not })
int: IntComparison? # numeric comparisonEventFilter (eventEmitted)
EventFilter: # single object or list (OR-ed)
contract: Contract # required
name: string? # event name (requires verified contract)
id: string? # topic hash, 0x-prefixed, 64 hex chars
not: bool? # negate entire event + parameters match
parameters: ParameterCondition[]?
# note: name and id are mutually exclusive (use one or the other)
# note: see ParameterCondition definition under FunctionFilter aboveLogFilter (logEmitted)
LogFilter: # single object or list (OR-ed)
startsWith: string[] # required. hex strings (typically 0x + 64 hex chars for full topic hashes)
contract: Contract? # optional. restrict to logs from this contract address
# note: invocation is not supported for logEmitted; only contract.address is used
matchAny: bool? # true = match any topic; false (default) = match all
not: bool? # negate entire log match