Onchain Event Framework for DeFiPy
Project description
Web3Scout: Onchain Event Framework for DeFiPy
🔗 SPDX-Anchor: anchorregistry.ai/AR-2026-5RJKqw5
Web3Scout pulls onchain DeFi data from EVM chains — event retrieval, pool state
reads, and reorg-aware block monitoring — behind a small, stable API (ABILoad,
ConnectW3, RetrieveEvents, FetchToken). As of v1 it stands on its own (no
eth_defi dependency) and is the substrate DeFiPy — and anyone else — builds on.
What it does
- Events — retrieve swaps and liquidity events via
RetrieveEvents: Uniswap V2 / V3 and forks like Sushi (Swap, Mint, Sync, Burn, Transfer, Create), Balancer V2 (VaultSwap/PoolBalanceChanged), and Curve (TokenExchange/AddLiquidity/RemoveLiquidity). - State reads — Uniswap V2 pair reserves and metadata (
FetchPairDetails), plus bundled read ABIs for Balancer (V2Vault/WeightedPool) and Curve (StableSwap) pool state. - Multi-protocol ABIs — Uniswap V2/V3, Sushi, Balancer, Curve, and ERC-20
ABIs resolvable through one
ABILoad(Platform.X, JSONContract.Y)interface. - Reorg-aware monitoring — detect and resolve chain reorganizations with
ReorganizationMonitor/JSONRPCReorganizationMonitor. - Token metadata — fetch ERC-20 details (symbol, decimals, …) with
FetchToken.
Installation
> git clone https://github.com/defipy-devs/web3scout
> pip install .
or
> pip install Web3Scout
Uni V2 Swap Events (Polygon) Example
from web3scout import *
abi = ABILoad(Platform.SUSHI, JSONContract.UniswapV2Pair)
connect = ConnectW3(Net.POLYGON)
connect.apply()
rEvents = RetrieveEvents(connect, abi)
last_block = rEvents.latest_block()
start_block = last_block - 3
dict_events = rEvents.apply(EventType.SWAP, start_block=start_block, end_block=last_block)
swap at block:61,234,918 tx:0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8
swap at block:61,234,918 tx:0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8
swap at block:61,234,918 tx:0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8
swap at block:61,234,918 tx:0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8
.
dict_events
{0: {'chain': 'polygon',
'contract': 'uniswapv2pair',
'type': 'swap',
'platform': 'sushi',
'address': '0x604229c960e5cacf2aaeac8be68ac07ba9df81c3',
'tx_hash': '0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8',
'blk_num': 61234918,
'timestamp': 1725051030,
'details': {'web3_type': web3._utils.datatypes.Swap,
'token0': '0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff',
'token1': '0xbF6f53423F25Df43a057F42A840158D6fDdB45BF',
'amount0In': 19000000000000000000,
'amount1Out': 7889648}},
1: {'chain': 'polygon',
'contract': 'uniswapv2pair',
'type': 'swap',
'platform': 'sushi',
'address': '0x604229c960e5cacf2aaeac8be68ac07ba9df81c3',
'tx_hash': '0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8',
'blk_num': 61234918,
'timestamp': 1725051030,
'details': {'web3_type': web3._utils.datatypes.Swap,
'token0': '0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff',
'token1': '0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff',
'amount0In': 0,
'amount1Out': 0}},
2: {'chain': 'polygon',
'contract': 'uniswapv2pair',
'type': 'swap',
'platform': 'sushi',
'address': '0x3c986748414a812e455dcd5418246b8fded5c369',
'tx_hash': '0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8',
'blk_num': 61234918,
'timestamp': 1725051030,
'details': {'web3_type': web3._utils.datatypes.Swap,
'token0': '0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff',
'token1': '0xbF6f53423F25Df43a057F42A840158D6fDdB45BF',
'amount0In': 21176176598530377323,
'amount1Out': 796785880798504079}},
3: {'chain': 'polygon',
'contract': 'uniswapv2pair',
'type': 'swap',
'platform': 'sushi',
'address': '0x3c986748414a812e455dcd5418246b8fded5c369',
'tx_hash': '0x9f16c76b6a83ac424ea736fb7dd2b1fc735888f222ee04dc1b1f7b933469faf8',
'blk_num': 61234918,
'timestamp': 1725051030,
'details': {'web3_type': web3._utils.datatypes.Swap,
'token0': '0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff',
'token1': '0xa5E0829CaCEd8fFDD4De3c43696c57F7D7A678ff',
'amount0In': 0,
'amount1Out': 0}}}
Uni V3 Swap Events (Polygon) Example
from web3scout import *
abi = ABILoad(Platform.UNIV3, JSONContract.UniswapV3Pool)
connect = ConnectW3(Net.POLYGON)
connect.apply()
rEvents = RetrieveEvents(connect, abi)
last_block = rEvents.latest_block()
start_block = last_block - 15
dict_events = rEvents.apply(EventType.MINT, start_block=start_block, end_block=last_block)
mint at block:61,391,083 tx:0xe499971b5410e766d00bf4467c6b333cda04577f1068bb676debe72331254365
mint at block:61,391,092 tx:0x29d53602b1bbd67734c2e3deba8ad0a55aa84204a6244e720f24ee5160505213
.
dict_events
{0: {'chain': 'polygon',
'contract': 'uniswapv3pool',
'type': 'mint',
'platform': 'uniswap_v3',
'pool_address': '0xb6e57ed85c4c9dbfef2a68711e9d6f36c56e0fcb',
'tx_hash': '0xe499971b5410e766d00bf4467c6b333cda04577f1068bb676debe72331254365',
'blk_num': 61391083,
'timestamp': 1725401207,
'details': {'web3_type': web3._utils.datatypes.Mint,
'owner': '0xC36442b4a4522E871399CD717aBDD847Ab11FE88',
'tick_lower': -286090,
'tick_upper': -284860,
'liquidity_amount': 884887839988325,
'amount0': 39958320744269616249,
'amount1': 17912626}},
1: {'chain': 'polygon',
'contract': 'uniswapv3pool',
'type': 'mint',
'platform': 'uniswap_v3',
'pool_address': '0x960fdfe0de1079459493a7e3aa857f8ce0b34016',
'tx_hash': '0x29d53602b1bbd67734c2e3deba8ad0a55aa84204a6244e720f24ee5160505213',
'blk_num': 61391092,
'timestamp': 1725401227,
'details': {'web3_type': web3._utils.datatypes.Mint,
'owner': '0xC36442b4a4522E871399CD717aBDD847Ab11FE88',
'tick_lower': 22600,
'tick_upper': 40000,
'liquidity_amount': 7675592444129481120,
'amount0': 64052149877205455,
'amount1': 29656680135133456015}}}
Protocol Coverage
Beyond the Uniswap V2/V3 event examples above, Web3Scout bundles minimal, address-based read ABIs for additional protocols, resolvable through the same ABILoad interface:
- Balancer — V2
VaultandWeightedPool:ABILoad(Platform.BALANCER, JSONContract.BalancerVault)andABILoad(Platform.BALANCER, JSONContract.BalancerWeightedPool) - Curve — plain
StableSwap:ABILoad(Platform.CURVE, JSONContract.CurveStableSwap)
These cover onchain state reads (pool tokens, balances, normalized weights, swap fee, amplification coefficient).
Balancer Pool State (Ethereum) Example
The Balancer ABIs are read ABIs for onchain state. ViewContract reads every
zero-input getter on a pool in a single call; parameterized calls (such as the
Vault's getPoolTokens) use the contract proxy from ABILoad(...).apply(w3, address).
from web3scout import *
connect = ConnectW3("https://eth.llamarpc.com") # any Ethereum mainnet RPC
connect.apply()
w3 = connect.get_w3()
# Balancer V2 80/20 BAL-WETH WeightedPool
pool_addr = "0x5c6Ee304399DBdB9C8Ef030aB642B10820DB8F56"
pool_abi = ABILoad(Platform.BALANCER, JSONContract.BalancerWeightedPool)
# Read all zero-input getters at once (verbose=True prints each)
pool_state = ViewContract(connect, pool_abi, verbose=True).apply(pool_addr)
[0] getPoolId() b'\x5c\x6e\xe3\x04...' # 32-byte pool id
[1] getVault() 0xBA12222222228d8Ba445958a75a0704d566BF2C8
[2] getNormalizedWeights() [800000000000000000, 200000000000000000] # 80% / 20% (1e18)
[3] getSwapFeePercentage() 1000000000000000 # 0.1% (1e18)
[4] totalSupply() 24875403726528338391741
Pool token addresses and balances live on the Vault, keyed by the pool id:
vault = ABILoad(Platform.BALANCER, JSONContract.BalancerVault).apply(
w3, "0xBA12222222228d8Ba445958a75a0704d566BF2C8") # canonical Balancer V2 Vault
tokens, balances, last_change_block = vault.functions.getPoolTokens(
pool_state["getPoolId"]).call()
Curve Pool State (Ethereum) Example
ViewContract reads the zero-input getters (A, fee); the per-coin getters
take an index, so those go through the contract proxy.
from web3scout import *
connect = ConnectW3("https://eth.llamarpc.com") # any Ethereum mainnet RPC
connect.apply()
w3 = connect.get_w3()
# Curve 3pool (DAI / USDC / USDT)
pool_addr = "0xbEbc44782C7dB0a1A60Cb6fe97d0b483032FF1C7"
abi = ABILoad(Platform.CURVE, JSONContract.CurveStableSwap)
# Zero-input getters: A (amplification) and fee
state = ViewContract(connect, abi, verbose=True).apply(pool_addr)
# coins(i) / balances(i) take a coin index -> use the proxy
pool = abi.apply(w3, pool_addr)
for i in range(3):
print(pool.functions.coins(i).call(), pool.functions.balances(i).call())
[0] A() 2000
[1] fee() 1000000 # 1e10-scaled
0x6B175474E89094C44Da98b954EedeAC495271d0F 412300000000000000000000000 # DAI
0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48 398700000000 # USDC
0xdAC17F958D2ee523a2206206994597C13D831ec7 376500000000 # USDT
Output values above are illustrative; the addresses (Balancer V2 Vault, Curve 3pool, DAI/USDC/USDT) are the canonical Ethereum mainnet contracts.
Balancer & Curve Events (Ethereum) Example
RetrieveEvents(...).apply(EventType.X, address=...) reads swaps and liquidity
events for Balancer and Curve — the same call used for Uniswap. apply() returns
one generic record per event — {blockNumber, event, address, transactionHash, …, args} — where args holds the decoded event fields.
Curve events are emitted by the pool:
from web3scout import *
connect = ConnectW3("https://eth.llamarpc.com") # any Ethereum mainnet RPC
connect.apply()
pool = "0xbEbc44782C7dB0a1A60Cb6fe97d0b483032FF1C7" # Curve 3pool
rEvents = RetrieveEvents(connect, ABILoad(Platform.CURVE, JSONContract.CurveStableSwap))
last = rEvents.latest_block()
swaps = rEvents.apply(EventType.SWAP, address=pool, start_block=last-50, end_block=last)
adds = rEvents.apply(EventType.ADD_LIQUIDITY, address=pool, start_block=last-500, end_block=last)
rems = rEvents.apply(EventType.REMOVE_LIQUIDITY, address=pool, start_block=last-500, end_block=last)
{0: {'blockNumber': 20850123,
'event': 'TokenExchange',
'address': '0xbEbc44782C7dB0a1A60Cb6fe97d0b483032FF1C7',
'transactionHash': '0x…',
'logIndex': 71,
'args': {'buyer': '0x…', 'sold_id': 1, 'tokens_sold': 250000000000,
'bought_id': 2, 'tokens_bought': 249981044}}}
Balancer events live on the canonical Vault, keyed by poolId — pass the Vault
address (Addr.BALANCER_V2_VAULT) and scope to one pool with argument_filters:
w3 = connect.get_w3()
pool_id = ABILoad(Platform.BALANCER, JSONContract.BalancerWeightedPool) \
.apply(w3, "0x5c6Ee304399DBdB9C8Ef030aB642B10820DB8F56").functions.getPoolId().call()
rEvents = RetrieveEvents(connect, ABILoad(Platform.BALANCER, JSONContract.BalancerVault))
last = rEvents.latest_block()
# one pool's swaps (omit argument_filters to read every pool's swaps)
swaps = rEvents.apply(EventType.SWAP, address=Addr.BALANCER_V2_VAULT,
argument_filters={'poolId': pool_id},
start_block=last-50, end_block=last)
# joins + exits: Balancer emits a single PoolBalanceChanged; the sign of the
# `deltas` array distinguishes add (>0) from remove (<0)
liq = rEvents.apply(EventType.POOL_BALANCE_CHANGED, address=Addr.BALANCER_V2_VAULT,
argument_filters={'poolId': pool_id},
start_block=last-500, end_block=last)
The bundled Curve liquidity ABIs (
AddLiquidity/RemoveLiquidity) are sized for 3-coin pools (e.g. 3pool); swap reads work for any plain pool.
Sushi Uniswap V2: Polygon
- Events (ie, Swap, Mint, Sync, Burn, Transfer): see notebook
Uniswap V3: Polygon
- Events (ie, Swap, Mint, Burn, Create): see notebook
Testing
Run the full test suite from the repo root:
> python -m pytest tests/ -v
Tests cover bug fixes and regression checks across:
- conversion — hex/bytes string conversion
- fetch_token — error handling in token metadata fetches
- deploy — contract deployment class references
- abi_load — import path corrections
- reorg_monitor — chain reorganization monitoring imports
- block_header — BlockHeader dataclass (extracted from eth_defi)
- token — ERC-20 token creation
- base_utils — port scanning utilities
Requirements
> pip install pytest
License
Web3Scout is licensed under the Apache License, Version 2.0.
See LICENSE and NOTICE for details.
Portions of this project may include code from third-party projects under compatible open-source licenses.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file web3scout-1.0.0.tar.gz.
File metadata
- Download URL: web3scout-1.0.0.tar.gz
- Upload date:
- Size: 176.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
105ee13efcf9fbe8f01a43a9cf3412c4edb29f53f655a8e93687059fdf623e40
|
|
| MD5 |
a0235f45206c805920fc397d590c1e66
|
|
| BLAKE2b-256 |
7a288fd882def70eb2cb11872daf7f543ef3785fe23d96cbb3762ea8f9aca2a8
|
File details
Details for the file web3scout-1.0.0-py3-none-any.whl.
File metadata
- Download URL: web3scout-1.0.0-py3-none-any.whl
- Upload date:
- Size: 197.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
cf72432a660d8ee3891bf0150358d94dd6db2693e670fb1b84330c3325c9c7b8
|
|
| MD5 |
96ee8d1d9e9d2885e99efa064524fb9f
|
|
| BLAKE2b-256 |
9f69930e403a4b6f4da55d54ae5ab1cf7f413e067a081d1c118616683b847c34
|