Advanced
Safe Transaction Service

Safe Transaction Service

The Safe Transaction Service tracks transactions sent via Safe contracts. It indexes these transactions using events (L2 chains) and tracing (L1 chains) mechanisms.

Key Features

  • Blockchain indexing: Executed transactions, configuration changes, ERC-20/721 transfers, and on-chain confirmations are automatically indexed from the blockchain.
  • Off-chain transaction signatures: Transactions can be sent to the service, enabling off-chain signature collection. This feature helps inform owners about pending transactions awaiting confirmation for execution.
  • Off-chain messages: The service can collect off-chain signatures to confirm messages following EIP-1271 (opens in a new tab).
  • Transaction decoding: The service keeps getting source and ABIs from contracts that interact with Safe to decode these interactions.

Technology stack overview

The Safe Transaction Service is a Django (opens in a new tab) app written in Python with the following architecture:

Transaction Service architecture

Blockchain indexing

The Safe Transaction Service can index automatically executed transactions, configuration changes, ERC-20/721 transfers, and on-chain confirmations. The indexer is running on worker-indexer by different periodic tasks (opens in a new tab).

ERC-20 and ERC-721 are indexed using eth_getLogs (opens in a new tab) filtered by the Transfer topic keccak('Transfer(address,address,uint256)').

Safe creation, executed transactions, configuration changes, and on-chain confirmations are indexed differently depending on whether the chain is L1 or L2.

For L1 chains, the indexer calls tracing methods. For the oldest blocks, trace_filter (opens in a new tab) is used filtering by singleton address of Safe contracts, and for the latest blocks trace_block (opens in a new tab) is used, as trace_filter takes longer to return updated information. trace_block will be used if the block depth is lower than ETH_INTERNAL_TXS_NUMBER_TRACE_BLOCKS. The environment variables indexing uses are defined here (opens in a new tab).

For L2 chains, the indexing is by events with the eth_getLogs (opens in a new tab) method with the corresponding topics.

From Safe creation, the Transaction Service stores each contract change on the SafeStatus model as nonce, owners, etc. The latest and current status of a Safe is stored as SafeLastStatus for easy database access and optimization.

Endpoints

The following endpoints show the current indexing status of the Safe Transaction Service:

  • GET /v1/about/indexing/

Response example:


_10
{
_10
"currentBlockNumber": 9773327, // Last block on the blockchain
_10
"erc20BlockNumber": 9773326, // Last block indexed for erc20/721 events
_10
"erc20Synced": true,
_10
"masterCopiesBlockNumber": 9773327, // Last block indexed for executed transactions, ether transfers, configuration changes, etc.
_10
"masterCopiesSynced": true,
_10
"synced": true
_10
}

Reorgs handling

Every block is marked as not confirmed during indexing unless it has some depth (configured via the ETH_REORG_BLOCKS environment variable). Unconfirmed blocks are checked periodically to see if the blockchain blockHash for that number changed before it reaches the desired number of confirmations. If that's the case, all blocks from that block and related transactions are deleted, and indexing is restarted to the last confirmed block.

Note: No off-chain signatures, transactions, or messages are lost in this process. Only on-chain data is removed.

Off-chain transaction signatures

The Safe Transaction Service can collect off-chain transaction signatures, allowing the owners to share their signatures to reach the required threshold before executing a transaction and spending less gas than on-chain approvals.

Endpoints

The following endpoints let us propose a transaction and collect every confirmation (off-chain signatures):

  • POST /v1/safes/{address}/multisig-transactions/: Creates a new transaction and requires at least one signature.
  • POST /v1/multisig-transactions/{safe_tx_hash}/confirmations/: Adds a new confirmation and needs the safe_tx_hash.
  • GET /v1/multisig-transactions/{safe_tx_hash}/: Returns all the multi-signature transaction information.
  • GET /v1/multisig-transactions/{safe_tx_hash}/confirmations/: Returns the list of all confirmations to a multi-signature transaction.

The following sequence diagram shows a use case for a Safe shared by Alice and Bob where at least one confirmation for each one is required:

What's the safe_tx_hash?

safe_tx_hash is the unique identifier for a Safe transaction and is calculated using the EIP-712 (opens in a new tab) standard:
keccak256(0x19 || 0x1 || domainSeparator || safeTxHashStruct)
where safeTxHashStruct is the hashStruct of a Safe transaction.

The following Guides show how to create and execute Safe transactions from the Safe Transaction Service.

Off-chain messages

The Safe Transaction Service can collect the necessary off-chain signatures to confirm a message using EIP-1271 (opens in a new tab).
The message can be a string (EIP-191 is used to get the hash) or an object EIP-712.

Endpoints

  • GET /v1/safes/{address}/messages/: Returns the messages created for the given Safe address.
  • POST /v1/safes/{address}/messages/: Creates a message with at least one signature.
  • GET /v1/messages/{message_hash}/: Returns a message for a given message hash.
  • POST /v1/messages/{message_hash}/signatures/: Adds another signature to the message with the given message hash.

The following sequence diagram shows a use case for a Safe shared by Alice and Bob where at least one signature for each one is required to confirm a message fully:

The following Guides show how to create and sign messages from the Safe Transaction Service.

Transaction decoder

The Safe Transaction Service can decode contract interactions. To achieve it, the service periodically gets source and ABIs from different sources like Sourcify, Etherscan, and Blockscout using the safe-eth-py library.

The detection of contract interactions is done in a periodic task executed every hour for multisig-transaction and module-transactions or every six hours for multisend-transactions on worker-contracts-tokens. For every new contract, the service tries to download the source, and the ABI requests it first to Sourcify, then Etherscan, and as a last chance, Blockscout. It's important to know that not all these data sources are supported or configured for every network on safe-eth-py.

Supported and configured networks on safe-eth-py:

Endpoints

  • POST /v1/data-decoder/ decodes a transaction data passed on the body for a to contract address.

Example transaction decoder


_10
curl -X 'POST' \
_10
'https://safe-transaction-sepolia.safe.global/api/v1/data-decoder/' \
_10
-H 'accept: application/json' \
_10
-H 'Content-Type: application/json' \
_10
-H 'X-CSRFToken: Gx1aRa8kIJGIAfReLAWwr9Q6dHv22dFt7VprdipLryHcxpfhk9aV0UDAhNz8gGYz' \
_10
-d '{
_10
"data": "0x095ea7b3000000000000000000000000e6fc577e87f7c977c4393300417dcc592d90acf8ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff",
_10
"to": "0x4127839cdf4F73d9fC9a2C2861d8d1799e9DF40C"
_10
}'

Output:


_15
{
_15
"method": "approve",
_15
"parameters": [
_15
{
_15
"name": "spender",
_15
"type": "address",
_15
"value": "0xe6fC577E87F7c977c4393300417dCC592D90acF8"
_15
},
_15
{
_15
"name": "value",
_15
"type": "uint256",
_15
"value": "115792089237316195423570985008687907853269984665640564039457584007913129639935"
_15
}
_15
]
_15
}

This decoded data is also included as dataDecoded in the GET /multisig-transactions, GET /module-transactions and GET /all-transactions endpoints.

Running and Maintenance

In the Safe Transaction Service (opens in a new tab) GitHub repository, under the Setup for production (event indexing) (opens in a new tab) and Setup for production (tracing mode) (opens in a new tab) sections, it is shown how to run the service and retrieve events from any supported network type. Note that the configuration must be adapted to the needs of a specific use case.

In addition, the Service maintenance (opens in a new tab) section shows how to deal with some common errors in the service operation.

Was this page helpful?