Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: blob fees #11029

Open
wants to merge 39 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
8484168
fix: bump blob fees on retries
spypsy Jan 3, 2025
eb311d5
PR fixes
spypsy Jan 6, 2025
86b6693
Merge branch 'master' into spy/tx-utils-blobs
spypsy Jan 6, 2025
53b354e
Merge branch 'master' into spy/tx-utils-blobs
spypsy Jan 6, 2025
ce614fe
Try info logging to get blob fees
spypsy Jan 6, 2025
e7b3f71
log existing image in bootstrap
spypsy Jan 6, 2025
aa4e4a9
actually use calculated maxFeePerBlobGas
spypsy Jan 6, 2025
dc303b4
Merge branch 'master' into spy/tx-utils-blobs
spypsy Jan 6, 2025
258a904
min bump percentage is 100 for ALL fees
spypsy Jan 6, 2025
9981f97
Merge branch 'master' into spy/tx-utils-blobs
spypsy Jan 7, 2025
36b6582
max gwei for blob fees
spypsy Jan 7, 2025
6e9e8ac
log maxFeePerBlobGas
spypsy Jan 7, 2025
83c27f9
add metadata on speed-up attempts
spypsy Jan 7, 2025
c286d82
fix conversion
spypsy Jan 7, 2025
39de22e
add some better logging
spypsy Jan 8, 2025
a6a6308
Merge branch 'master' into spy/tx-utils-blobs
spypsy Jan 8, 2025
7b92aaf
fix import
spypsy Jan 8, 2025
d1442f7
Merge remote-tracking branch 'refs/remotes/origin/spy/tx-utils-blobs'…
spypsy Jan 8, 2025
c4783eb
fix logging, add test
spypsy Jan 8, 2025
e553adc
rm unused var
spypsy Jan 8, 2025
db53a66
disable faucet on sepolia
spypsy Jan 8, 2025
5e73f7e
actually use format fn...
spypsy Jan 8, 2025
3bb7df0
allow for fixed priority fee
spypsy Jan 8, 2025
aa0e244
truncate more log fields
spypsy Jan 9, 2025
5737628
don't use fixed blob fees
spypsy Jan 9, 2025
9a4fef2
fix maybe undefined
spypsy Jan 9, 2025
1da6523
values updates
spypsy Jan 9, 2025
11e448c
update typing
spypsy Jan 9, 2025
b04b1b1
fix test
spypsy Jan 9, 2025
379e286
try fixing logging again
spypsy Jan 9, 2025
6e95160
use formatter in sendTransaction
spypsy Jan 9, 2025
9fd9ec2
fix for <1 gwei prio fee
spypsy Jan 9, 2025
cfbe1d0
just format everywhere
spypsy Jan 9, 2025
89d1ec5
Merge branch 'master' into spy/tx-utils-blobs
spypsy Jan 9, 2025
9d1e93d
allow 2 decimals in percentages, set lower gas estimate buffer
spypsy Jan 9, 2025
7140c1e
update test
spypsy Jan 10, 2025
da68f1a
another
spypsy Jan 10, 2025
dec90b6
simplify formatting
spypsy Jan 10, 2025
45ee7f8
maintain shorter hexes
spypsy Jan 10, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions bootstrap.sh
Original file line number Diff line number Diff line change
Expand Up @@ -203,6 +203,7 @@ case "$cmd" in
image=aztecprotocol/aztec:$(git rev-parse HEAD)
docker pull $image &>/dev/null || true
if docker_has_image $image; then
echo "Image $image already exists and has been downloaded." && exit
exit
fi
github_group "image-aztec"
Expand Down
70 changes: 63 additions & 7 deletions yarn-project/ethereum/src/l1_tx_utils.test.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { Blob } from '@aztec/foundation/blob';
import { EthAddress } from '@aztec/foundation/eth-address';
import { createLogger } from '@aztec/foundation/log';
import { sleep } from '@aztec/foundation/sleep';
Expand Down Expand Up @@ -106,8 +107,9 @@ describe('GasUtils', () => {
await cheatCodes.setAutomine(false);
await cheatCodes.setIntervalMining(0);

// Ensure initial base fee is low
await cheatCodes.setNextBlockBaseFeePerGas(initialBaseFee);
spalladino marked this conversation as resolved.
Show resolved Hide resolved
// Add blob data
const blobData = new Uint8Array(131072).fill(1);
const kzg = Blob.getViemKzgInstance();

const request = {
to: '0x1234567890123456789012345678901234567890' as `0x${string}`,
Expand All @@ -119,12 +121,16 @@ describe('GasUtils', () => {

const originalMaxFeePerGas = WEI_CONST * 10n;
const originalMaxPriorityFeePerGas = WEI_CONST;
const originalMaxFeePerBlobGas = WEI_CONST * 10n;

const txHash = await walletClient.sendTransaction({
...request,
gas: estimatedGas,
maxFeePerGas: originalMaxFeePerGas,
maxPriorityFeePerGas: originalMaxPriorityFeePerGas,
blobs: [blobData],
kzg,
maxFeePerBlobGas: originalMaxFeePerBlobGas,
});

const rawTx = await cheatCodes.getRawTransaction(txHash);
Expand All @@ -142,11 +148,12 @@ describe('GasUtils', () => {
params: [rawTx],
});

// keeping auto-mining disabled to simulate a stuck transaction
// The monitor should detect the stall and create a replacement tx

// Monitor should detect stall and replace with higher gas price
const monitorFn = gasUtils.monitorTransaction(request, txHash, { gasLimit: estimatedGas });
const monitorFn = gasUtils.monitorTransaction(request, txHash, { gasLimit: estimatedGas }, undefined, {
blobs: [blobData],
kzg,
maxFeePerBlobGas: WEI_CONST * 20n,
});

await sleep(2000);
// re-enable mining
Expand All @@ -156,11 +163,12 @@ describe('GasUtils', () => {
// Verify that a replacement transaction was created
expect(receipt.transactionHash).not.toBe(txHash);

// Get details of replacement tx to verify higher gas price
// Get details of replacement tx to verify higher gas prices
const replacementTx = await publicClient.getTransaction({ hash: receipt.transactionHash });

expect(replacementTx.maxFeePerGas!).toBeGreaterThan(originalMaxFeePerGas);
expect(replacementTx.maxPriorityFeePerGas!).toBeGreaterThan(originalMaxPriorityFeePerGas);
expect(replacementTx.maxFeePerBlobGas!).toBeGreaterThan(originalMaxFeePerBlobGas);
}, 20_000);

it('respects max gas price limits during spikes', async () => {
Expand Down Expand Up @@ -299,4 +307,52 @@ describe('GasUtils', () => {
const expectedEstimate = baseEstimate + (baseEstimate * 20n) / 100n;
expect(bufferedEstimate).toBe(expectedEstimate);
});

it('correctly handles transactions with blobs', async () => {
// Create a sample blob
const blobData = new Uint8Array(131072).fill(1); // 128KB blob
const kzg = Blob.getViemKzgInstance();

const receipt = await gasUtils.sendAndMonitorTransaction(
{
to: '0x1234567890123456789012345678901234567890',
data: '0x',
value: 0n,
},
undefined,
{
blobs: [blobData],
kzg,
maxFeePerBlobGas: 10000000000n, // 10 gwei
},
);

expect(receipt.status).toBe('success');
expect(receipt.blobGasUsed).toBeDefined();
expect(receipt.blobGasPrice).toBeDefined();
}, 20_000);

it('estimates gas correctly for blob transactions', async () => {
// Create a sample blob
const blobData = new Uint8Array(131072).fill(1); // 128KB blob
const kzg = Blob.getViemKzgInstance();

const request = {
to: '0x1234567890123456789012345678901234567890' as `0x${string}`,
data: '0x' as `0x${string}`,
value: 0n,
};

// Estimate gas without blobs first
const baseEstimate = await gasUtils.estimateGas(walletClient.account!, request);

// Estimate gas with blobs
const blobEstimate = await gasUtils.estimateGas(walletClient.account!, request, undefined, {
blobs: [blobData],
kzg,
maxFeePerBlobGas: 10000000000n,
});
// Blob transactions should require more gas
expect(blobEstimate).toBeGreaterThan(baseEstimate);
Comment on lines +359 to +360
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Curious: why is this? Does eth_estimateGas return both the execution gas and the blob gas mixed up together?

Copy link
Member Author

@spypsy spypsy Jan 6, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it seems in our estimateGas, we have to use prepareTransactionRequest to get a full estimation, which does increase a lot compared to estimateGas so I assumed that includes the blob gas

}, 20_000);
});
74 changes: 62 additions & 12 deletions yarn-project/ethereum/src/l1_tx_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,10 @@ const WEI_CONST = 1_000_000_000n;
// https://github.com/ethereum/go-ethereum/blob/e3d61e6db028c412f74bc4d4c7e117a9e29d0de0/core/txpool/legacypool/list.go#L298
const MIN_REPLACEMENT_BUMP_PERCENTAGE = 10n;

// setting a minimum bump percentage to 100% due to geth's implementation
// https://github.com/ethereum/go-ethereum/blob/e3d61e6db028c412f74bc4d4c7e117a9e29d0de0/core/txpool/blobpool/config.go#L34
const MIN_BLOB_REPLACEMENT_BUMP_PERCENTAGE = 100n;

// Avg ethereum block time is ~12s
const BLOCK_TIME_MS = 12_000;

Expand Down Expand Up @@ -149,6 +153,7 @@ export interface L1BlobInputs {
interface GasPrice {
maxFeePerGas: bigint;
maxPriorityFeePerGas: bigint;
maxFeePerBlobGas?: bigint;
}

export class L1TxUtils {
Expand All @@ -175,7 +180,7 @@ export class L1TxUtils {
public async sendTransaction(
request: L1TxRequest,
_gasConfig?: Partial<L1TxUtilsConfig> & { fixedGas?: bigint },
_blobInputs?: L1BlobInputs,
blobInputs?: L1BlobInputs,
): Promise<{ txHash: Hex; gasLimit: bigint; gasPrice: GasPrice }> {
const gasConfig = { ...this.config, ..._gasConfig };
const account = this.walletClient.account;
Expand All @@ -189,15 +194,24 @@ export class L1TxUtils {

const gasPrice = await this.getGasPrice(gasConfig);

const blobInputs = _blobInputs || {};
const txHash = await this.walletClient.sendTransaction({
...request,
...blobInputs,
gas: gasLimit,
maxFeePerGas: gasPrice.maxFeePerGas,
maxPriorityFeePerGas: gasPrice.maxPriorityFeePerGas,
});

let txHash: Hex;
if (blobInputs) {
txHash = await this.walletClient.sendTransaction({
...request,
...blobInputs,
gas: gasLimit,
maxFeePerGas: gasPrice.maxFeePerGas,
maxPriorityFeePerGas: gasPrice.maxPriorityFeePerGas,
maxFeePerBlobGas: gasPrice.maxFeePerBlobGas!,
});
} else {
txHash = await this.walletClient.sendTransaction({
...request,
gas: gasLimit,
maxFeePerGas: gasPrice.maxFeePerGas,
maxPriorityFeePerGas: gasPrice.maxPriorityFeePerGas,
});
}
this.logger?.verbose(`Sent L1 transaction ${txHash}`, {
gasLimit,
maxFeePerGas: formatGwei(gasPrice.maxFeePerGas),
Expand Down Expand Up @@ -301,7 +315,11 @@ export class L1TxUtils {
gasConfig,
attempts,
tx.maxFeePerGas && tx.maxPriorityFeePerGas
? { maxFeePerGas: tx.maxFeePerGas, maxPriorityFeePerGas: tx.maxPriorityFeePerGas }
? {
maxFeePerGas: tx.maxFeePerGas,
maxPriorityFeePerGas: tx.maxPriorityFeePerGas,
maxFeePerBlobGas: tx.maxFeePerBlobGas,
}
: undefined,
);

Expand Down Expand Up @@ -365,15 +383,29 @@ export class L1TxUtils {
const block = await this.publicClient.getBlock({ blockTag: 'latest' });
const baseFee = block.baseFeePerGas ?? 0n;

// Get blob base fee if available
let blobBaseFee = 0n;
try {
const blobBaseFeeHex = await this.publicClient.request({ method: 'eth_blobBaseFee' });
blobBaseFee = BigInt(blobBaseFeeHex);
Comment on lines +428 to +429
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I understand we should be increasing the blob base fee to 12.5% per stall time in blocks, as we do with the base fee (see here).

this.logger?.debug('Blob base fee:', { blobBaseFee: formatGwei(blobBaseFee) });
} catch {
this.logger?.warn('Failed to get blob base fee', attempt);
}

// Get initial priority fee from the network
let priorityFee = await this.publicClient.estimateMaxPriorityFeePerGas();
let maxFeePerGas = baseFee;

let maxFeePerBlobGas = blobBaseFee;

// Bump base fee so it's valid for next blocks if it stalls
const numBlocks = Math.ceil(gasConfig.stallTimeMs! / BLOCK_TIME_MS);
for (let i = 0; i < numBlocks; i++) {
// each block can go up 12.5% from previous baseFee
maxFeePerGas = (maxFeePerGas * (1_000n + 125n)) / 1_000n;
// same for blob gas fee
maxFeePerBlobGas = (maxFeePerBlobGas * (1_000n + 125n)) / 1_000n;
}

if (attempt > 0) {
Expand Down Expand Up @@ -405,14 +437,32 @@ export class L1TxUtils {
// Ensure priority fee doesn't exceed max fee
const maxPriorityFeePerGas = priorityFee > maxFeePerGas ? maxFeePerGas : priorityFee;

if (attempt > 0 && previousGasPrice?.maxFeePerBlobGas) {
const bumpPercentage =
gasConfig.priorityFeeRetryBumpPercentage! > MIN_BLOB_REPLACEMENT_BUMP_PERCENTAGE
? gasConfig.priorityFeeRetryBumpPercentage!
: MIN_BLOB_REPLACEMENT_BUMP_PERCENTAGE;

// calculate min blob fee based on previous attempt
const minBlobFee = (previousGasPrice.maxFeePerBlobGas * (100n + bumpPercentage)) / 100n;

// use max between current network values and min required values
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry I didn't catch this one earlier, but I believe we're missing the check to not go over gasConfig.maxGwei with the blob fee. Given we're duplicating it on every attempt, let's make sure it's in there.

maxFeePerBlobGas = maxFeePerBlobGas > minBlobFee ? maxFeePerBlobGas : minBlobFee;
}

this.logger?.debug(`Computed gas price`, {
attempt,
baseFee: formatGwei(baseFee),
maxFeePerGas: formatGwei(maxFeePerGas),
maxPriorityFeePerGas: formatGwei(maxPriorityFeePerGas),
...(maxFeePerBlobGas && { maxFeePerBlobGas: formatGwei(maxFeePerBlobGas) }),
});

return { maxFeePerGas, maxPriorityFeePerGas };
return {
maxFeePerGas,
maxPriorityFeePerGas,
...(maxFeePerBlobGas && { maxFeePerBlobGas: maxFeePerBlobGas }),
};
}

/**
Expand Down
Loading