WeaveVM Bundler is a data protocol specification and library that introduces the first bundled EVM transactions format. This protocol draws inspiration from Arweave's ANS-102 specification.
Bundler as data protocol and library is still in PoC (Proof of Concept) phase - not recommended for production usage, testing purposes only.
- Reduces transaction overhead fees from multiple fees (
n
) pern
transaction to a single fee per bundle of envelopes (n
transactions) - Enables third-party services to handle bundle settlement on WeaveVM
- Maximizes the TPS capacity of WeaveVM network without requiring additional protocol changes or constraints
- Supports relational data grouping by combining multiple related transactions into a single bundle
- Bundler: Refers to the data protocol specification of the EVM bundled transactions on WeaveVM.
- Envelope: A legacy EVM transaction that serves as the fundamental building block and composition unit of a Bundle.
- Bundle: An EIP-1559 transaction that groups multiple envelopes (
n > 0
), enabling efficient transaction batching and processing. - Superbundle: A transaction that carries multiple bundles.
- Bundler Lib: Refers to the Bundler Rust library that facilitates composing and propagating Bundler's bundles.
A bundle is a group of envelopes organized through the following process:
- Envelopes MUST be grouped in a vector
- The bundle is Borsh serialized according to the
BundleData
type - The resulting serialization vector is compressed using Brotli compression
- The Borsh-Brotli serialized-compressed vector is added as
input
(calldata) to an EIP-1559 transaction - The resulting bundle is broadcasted on WeaveVM with
target
set to0xbabe1
address (0xbabe1d25501157043c7b4ea7CBC877B9B4D8A057
)
pub struct BundleData {
pub envelopes: Vec<TxEnvelopeWrapper>,
}
flowchart TB
env[Envelopes] --> vec[Vector of TxEnvelopeWrapper]
vec --> bs[Borsh Serialization]
subgraph Serialization
bs --> bv[Serialized Vector]
bv --> bc[Brotli Compression]
end
bc --> comp[Compressed Data]
comp --> eip[EIP-1559 Transaction]
eip --> |input/calldata| bundle[Bundle]
bundle --> |broadcast| wvm[WeaveVM Network]
style vec fill:#2d3748,stroke:#1a202c,color:#fff
style bs fill:#1a365d,stroke:#2a4365,color:#fff
style bc fill:#1a365d,stroke:#2a4365,color:#fff
style bundle fill:#22543d,stroke:#276749,color:#fff
style wvm fill:#3c366b,stroke:#4c51bf,color:#fff
style env fill:#2d3748,stroke:#1a202c,color:#fff
style comp fill:#2d3748,stroke:#1a202c,color:#fff
style eip fill:#2d3748,stroke:#1a202c,color:#fff
style Serialization fill:#1a202c,stroke:#2d3748,color:#fff
Bundles versioning is based on the bundles target address:
Bundle Version | Bundler Target Acronym | Bundler Target Address |
---|---|---|
v0.1.0 | 0xbabe1 |
0xbabe1d25501157043c7b4ea7CBC877B9B4D8A057 |
An envelope is a signed Legacy EVM transaction with the following MUSTs and restrictions.
pub struct Tag {
pub name: String,
pub value: String,
}
pub struct EnvelopeSignature {
pub y_parity: bool,
pub r: String,
pub s: String,
}
pub struct TxEnvelopeWrapper {
pub chain_id: u64,
pub nonce: u64,
pub gas_price: u128,
pub gas_limit: u64,
pub to: String,
pub value: String,
pub input: String,
pub hash: String,
pub signature: EnvelopeSignature,
pub tags: Option<Vec<Tag>>,
}
-
Transaction Fields
nonce
: MUST be 0gas_limit
: MUST be 0gas_price
: MUST be 0value
: MUST be 0
-
Size Restrictions
- Total Borsh-Brotli compressed envelopes (Bundle data) MUST be under 9 MB
- Total Tags bytes size must be <= 2048 bytes before compression.
-
Signature Requirements
- each envelope MUST have a valid signature
-
Usage Constraints
- MUST be used strictly for data settling on WeaveVM
- MUST only contain envelope's calldata, with optional
target
setting (default fallback to ZERO address) - CANNOT be used for:
- tWVM transfers
- Contract interactions
- Any purpose other than data settling
The selection of transaction types follows clear efficiency principles. Legacy transactions were chosen for envelopes due to their minimal size (144 bytes), making them the most space-efficient option for data storage. EIP-1559 transactions were adopted for bundles as the widely accepted standard for transaction processing.
%%{init: {'theme': 'dark'}}%%
graph LR
subgraph Transaction Size in Bytes
L["Legacy (144)"]
E1["Eip2930 (160)"]
E2["Eip1559 (176)"]
E3["Eip4844 (208)"]
E4["Eip7702 (192)"]
D["Deposit (192)"]
end
style L fill:#1a365d,stroke:#2a4365,color:#fff,stroke-width:4px
style E1 fill:#1a365d,stroke:#2a4365,color:#fff,stroke-width:4.4px
style E2 fill:#1a365d,stroke:#2a4365,color:#fff,stroke-width:4.8px
style E3 fill:#1a365d,stroke:#2a4365,color:#fff,stroke-width:5.7px
style E4 fill:#1a365d,stroke:#2a4365,color:#fff,stroke-width:5.3px
style D fill:#1a365d,stroke:#2a4365,color:#fff,stroke-width:5.3px
-
Envelopes exist as signed Legacy transactions within bundles but operate under distinct processing rules - they are not individually processed by the WeaveVM network as transactions, despite having the structure of a Legacy transaction (signed data with a Transaction type). Instead, they are bundled together and processed as a single onchain transaction (therefore the advantage of Bundler).
-
Multiple instances of the same envelope within a bundle are permissible and do not invalidate either the bundle or the envelopes themselves. These duplicate instances are treated as copies sharing the same timestamp when found in a single bundle. When appearing across different bundles, they are considered distinct instances with their respective bundle timestamps (valid envelopes and considered as copies of distinct timestamps).
-
Since envelopes are implemented as signed Legacy transactions, they are strictly reserved for data settling purposes. Their use for any other purpose is explicitly prohibited for the envelope's signer security.
bundler = { git = "https://github.com/weaveVM/bundler", branch = "main" }
use bundler::utils::core::envelope::Envelope;
use bundler::utils::core::bundle::Bundle;
use bundler::utils::core::tags::Tag;
// Envelope
let envelope = Envelope::new()
.data(byte_vec)
.target(address)
.tags(tags)
.build()?;
// Bundle
let bundle_tx = Bundle::new()
.private_key(private_key)
.envelopes(envelopes)
.build()
.propagate()
.await?;
async fn send_bundle_without_target() -> eyre::Result<String> {
// will fail until a tWVM funded EOA (pk) is provided
let private_key = String::from("");
let mut envelopes: Vec<Envelope> = vec![];
for _ in 0..10 {
let random_calldata: String = generate_random_calldata(128_000); // 128 KB of random calldata
let envelope_data = serde_json::to_vec(&random_calldata).unwrap();
let envelope = Envelope::new()
.data(Some(envelope_data))
.target(None)
.build()?;
envelopes.push(envelope);
}
let bundle_tx = Bundle::new()
.private_key(private_key)
.envelopes(envelopes)
.build()
.propagate()
.await?;
Ok(bundle_tx)
}
async fn send_envelope_with_tags() -> eyre::Result<String> {
// will fail until a tWVM funded EOA (pk) is provided
let private_key = String::from("");
let mut envelopes: Vec<Envelope> = vec![];
// add your tags to a vector
let tags = vec![Tag::new(
"Content-Type".to_string(),
"text/plain".to_string(),
)];
for _ in 0..1 {
let random_calldata: String = generate_random_calldata(128_000); // 128 KB of random calldata
let envelope_data = serde_json::to_vec(&random_calldata).unwrap();
let envelope = Envelope::new()
.data(Some(envelope_data))
.target(None)
.tags(Some(tags.clone())) // register your tags
.build()
.unwrap();
envelopes.push(envelope);
}
let bundle_tx = Bundle::new()
.private_key(private_key)
.envelopes(envelopes)
.build()
.expect("REASON")
.propagate()
.await
.unwrap();
Ok(bundle_tx)
}
For more examples, check the tests in lib.rs and have a look over types
- Base endpoint: https://bundler.wvm.network/
GET /v1/envelopes/:bundle_txid
GET /v1/envelopes/ids/:bundle_txid
In the comparison below, we tested data settling of 1MB of non-zero bytes. WeaveVM's pricing of non-zero bytes (8 gas) and large transaction data size limit (8MB) allows us to fit the whole MB in a single transaction, paying a single overhead fee.
Chain | File Size (bytes) | Number of Contracts/Tx | Gas Used | Gas Price (Gwei) | Cost in Native | Native Price (USD) | Total (USD) |
---|---|---|---|---|---|---|---|
WeaveVM L1 Calldata | 1,000,000 | 1 | 8,500,000 (8M for calldata & 500k as base gas fee) | 1 Gwei | - | - | ~$0.05 |
Ethereum L1 | 1,000,000 | 41 | 202,835,200 gas | 20 Gwei | 4.056704 | $3641.98 | $14774.43 |
Polygon Sidechain | 1,000,000 | 41 | 202,835,200 gas | 40 Gwei (L1: 20 Gwei) | 8.113408 | $0.52 | $4.21 |
BSC L1 | 1,000,000 | 41 | 202,835,200 gas | 5 Gwei | 1.014176 | $717.59 | $727.76 |
Arbitrum (Optimistic L2) | 1,000,000 | 41 | 202,835,200 gas (+15,000,000 L1 gas) | 0.1 Gwei (L1: 20 Gwei) | 0.020284 (+0.128168 L1 fee) | $3641.98 | $540.66 |
Avalanche L1 | 1,000,000 | 41 | 202,835,200 gas | 25 Gwei | 5.070880 | $43.90 | $222.61 |
Base (Optimistic L2) | 1,000,000 | 41 | 202,835,200 gas (+15,000,000 L1 gas) | 0.001 Gwei (L1: 20 Gwei) | 0.000203 (+0.128168 L1 fee) | $3641.98 | $467.52 |
Optimism (Optimistic L2) | 1,000,000 | 41 | 202,835,200 gas (+15,000,000 L1 gas) | 0.001 Gwei (L1: 20 Gwei) | 0.000203 (+0.128168 L1 fee) | $3641.98 | $467.52 |
Blast (Optimistic L2) | 1,000,000 | 41 | 202,835,200 gas (+15,000,000 L1 gas) | 0.001 Gwei (L1: 20 Gwei) | 0.000203 (+0.128168 L1 fee) | $3641.98 | $467.52 |
Linea (ZK L2) | 1,000,000 | 41 | 202,835,200 gas (+12,000,000 L1 gas) | 0.05 Gwei (L1: 20 Gwei) | 0.010142 (+0.072095 L1 fee) | $3641.98 | $299.50 |
Scroll (ZK L2) | 1,000,000 | 41 | 202,835,200 gas (+12,000,000 L1 gas) | 0.05 Gwei (L1: 20 Gwei) | 0.010142 (+0.072095 L1 fee) | $3641.98 | $299.50 |
Moonbeam (Polkadot) | 1,000,000 | 41 | 202,835,200 gas (+NaN L1 gas) | 100 Gwei | 20.283520 | $0.27 | $5.40 |
Polygon zkEVM (ZK L2) | 1,000,000 | 41 | 202,835,200 gas (+12,000,000 L1 gas) | 0.05 Gwei (L1: 20 Gwei) | 0.010142 (+0.072095 L1 fee) | $3641.98 | $299.50 |
Solana L1 | 1,000,000 | 98 | 490,000 imports | N/A | 0.000495 (0.000005 deposit) | $217.67 | $0.11 |
Now let's take the data even higher, but for simplicity, let's not fit the whole data in a single WeaveVM L1 calldata transaction. Instead, we'll split it into 1MB transactions (creating multiple data settlement overhead fees): 5MB, 5 txs of 1 MB each:
Chain | File Size (bytes) | Number of Contracts/Tx | Gas Used | Gas Price (Gwei) | Cost in Native | Native Price (USD) | Total (USD) |
---|---|---|---|---|---|---|---|
WeaveVM Bundler | 5,000,000 | 1 | 40,500,000 (40M for calldata & 500k as base gas fee) | 1 Gwei | - | - | ~$0.25-$0.27 |
WeaveVM L1 Calldata | 5,000,000 | 5 | 42,500,000 (40M for calldata & 2.5M as base gas fee) | 1 Gwei | - | - | ~$0.22 |
Ethereum L1 | 5,000,000 | 204 | 1,009,228,800 gas | 20 Gwei | 20.184576 | $3650.62 | $73686.22 |
Polygon Sidechain | 5,000,000 | 204 | 1,009,228,800 gas | 40 Gwei (L1: 20 Gwei) | 40.369152 | $0.52 | $20.95 |
BSC L1 | 5,000,000 | 204 | 1,009,228,800 gas | 5 Gwei | 5.046144 | $717.75 | $3621.87 |
Arbitrum (Optimistic L2) | 5,000,000 | 204 | 1,009,228,800 gas (+80,000,000 L1 gas) | 0.1 Gwei (L1: 20 Gwei) | 0.100923 (+0.640836 L1 fee) | $3650.62 | $2707.88 |
Avalanche L1 | 5,000,000 | 204 | 1,009,228,800 gas | 25 Gwei | 25.230720 | $44.01 | $1110.40 |
Base (Optimistic L2) | 5,000,000 | 204 | 1,009,228,800 gas (+80,000,000 L1 gas) | 0.001 Gwei (L1: 20 Gwei) | 0.001009 (+0.640836 L1 fee) | $3650.62 | $2343.13 |
Optimism (Optimistic L2) | 5,000,000 | 204 | 1,009,228,800 gas (+80,000,000 L1 gas) | 0.001 Gwei (L1: 20 Gwei) | 0.001009 (+0.640836 L1 fee) | $3650.62 | $2343.13 |
Blast (Optimistic L2) | 5,000,000 | 204 | 1,009,228,800 gas (+80,000,000 L1 gas) | 0.001 Gwei (L1: 20 Gwei) | 0.001009 (+0.640836 L1 fee) | $3650.62 | $2343.13 |
Linea (ZK L2) | 5,000,000 | 204 | 1,009,228,800 gas (+60,000,000 L1 gas) | 0.05 Gwei (L1: 20 Gwei) | 0.050461 (+0.360470 L1 fee) | $3650.62 | $1500.16 |
Scroll (ZK L2) | 5,000,000 | 204 | 1,009,228,800 gas (+60,000,000 L1 gas) | 0.05 Gwei (L1: 20 Gwei) | 0.050461 (+0.360470 L1 fee) | $3650.62 | $1500.16 |
Moonbeam (Polkadot) | 5,000,000 | 204 | 1,009,228,800 gas (+NaN L1 gas) | 100 Gwei | 100.922880 | $0.27 | $26.94 |
Polygon zkEVM (ZK L2) | 5,000,000 | 204 | 1,009,228,800 gas (+60,000,000 L1 gas) | 0.05 Gwei (L1: 20 Gwei) | 0.050461 (+0.360470 L1 fee) | $3650.62 | $1500.16 |
Solana L1 | 5,000,000 | 489 tx | 2445.00k imports | N/A | 0.002468 (0.000023 deposit) | $218.44 | $0.54 |
Let's compare storing 40 MB of data (40 x 1 MB transactions) using two different methods, considering the 8 MB bundle size limit:
Metric | WeaveVM L1 Calldata | WeaveVM Bundler |
---|---|---|
Total Data Size | 40 MB | 40 MB |
Transaction Format | 40 separate EIP-1559 transactions | 5 bundle transactions (8MB each, 40 * 1MB envelopes) |
Transactions per Bundle | 1 MB each | 8 x 1MB per bundle |
Gas Cost per Tx | 8.5M gas (8M calldata + 500k base) | 64.5M gas (64M + 500k base) per bundle |
Number of Base Fees | 40 | 5 |
Total Gas Used | 340M gas (40 x 8.5M) | 322.5M gas (5 x 64.5M) |
Gas Price | 1 Gwei | 1 Gwei |
Total Cost | ~$1.5-1.7 | ~$1.3 |
Cost Savings | - | ~15% cheaper |
The key advantage of Bundler comes from reducing the number of base fee payments from 40 (in L1 calldata) to just 5 bundles, with each bundle containing 8 transactions, resulting in approximately 15% cost savings.
This project is licensed under the MIT License