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 & l1-publisher logging #11029

Merged
merged 45 commits into from
Jan 10, 2025
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
45 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
fa98417
fix test
spypsy Jan 10, 2025
99f4fe6
throw formatted error at the source
spypsy Jan 10, 2025
5cca260
add in 'failed to send transaction'
spypsy Jan 10, 2025
b52c269
remove commented code
spypsy Jan 10, 2025
b0b4f4e
Update yarn-project/ethereum/src/l1_tx_utils.ts
spypsy Jan 10, 2025
00a9b72
Update yarn-project/ethereum/src/utils.ts
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
74 changes: 65 additions & 9 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 @@ -256,7 +264,7 @@ describe('GasUtils', () => {
const initialGasPrice = await gasUtils['getGasPrice']();

// Get retry gas price for 2nd attempt
const retryGasPrice = await gasUtils['getGasPrice'](undefined, 1, initialGasPrice);
const retryGasPrice = await gasUtils['getGasPrice'](undefined, false, 1, initialGasPrice);

// With default config, retry should bump fees by 50%
const expectedPriorityFee = (initialGasPrice.maxPriorityFeePerGas * 150n) / 100n;
Expand All @@ -275,7 +283,7 @@ describe('GasUtils', () => {
const initialGasPrice = await gasUtils['getGasPrice']();

// Get retry gas price with attempt = 1
const retryGasPrice = await gasUtils['getGasPrice'](undefined, 1, initialGasPrice);
const retryGasPrice = await gasUtils['getGasPrice'](undefined, false, 1, initialGasPrice);

// Should use 10% minimum bump even though config specified 5%
const expectedPriorityFee = (initialGasPrice.maxPriorityFeePerGas * 110n) / 100n;
Expand All @@ -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);
});
87 changes: 71 additions & 16 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 @@ -187,17 +192,26 @@ export class L1TxUtils {
gasLimit = await this.estimateGas(account, request);
}

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,
});

const gasPrice = await this.getGasPrice(gasConfig, !!blobInputs);

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 @@ -299,9 +313,14 @@ export class L1TxUtils {
attempts++;
const newGasPrice = await this.getGasPrice(
gasConfig,
!!blobInputs,
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 @@ -358,29 +377,47 @@ export class L1TxUtils {
*/
private async getGasPrice(
_gasConfig?: L1TxUtilsConfig,
isBlobTx: boolean = false,
attempt: number = 0,
previousGasPrice?: typeof attempt extends 0 ? never : GasPrice,
): Promise<GasPrice> {
const gasConfig = { ...this.config, ..._gasConfig };
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);
spalladino marked this conversation as resolved.
Show resolved Hide resolved
this.logger?.debug('Blob base fee:', { blobBaseFee: formatGwei(blobBaseFee) });
} catch {
this.logger?.warn('Failed to get blob base fee', attempt);
}
Comment on lines +428 to +435
Copy link
Collaborator

Choose a reason for hiding this comment

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

Should we just skip all this if isBlobTx is false? And fail loudly if we fail to get the blob base fee, so the tx does not get rejected?


// 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) {
const configBump =
gasConfig.priorityFeeRetryBumpPercentage ?? defaultL1TxUtilsConfig.priorityFeeRetryBumpPercentage!;
Comment on lines 462 to 463
Copy link
Collaborator

Choose a reason for hiding this comment

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

Nit: it'll be a bit confusing that the priorityFeeRetryBumpPercentage ends up being used for bumping not just the priority fee but all fees. Should we rename it? Or use it strictly for the priority fee only? I vote for the latter, since it probably doesn't make sense to bump the base fee too much.

const bumpPercentage =
configBump > MIN_REPLACEMENT_BUMP_PERCENTAGE ? configBump : MIN_REPLACEMENT_BUMP_PERCENTAGE;

// if this is a blob tx, we have to use the blob bump percentage
const minBumpPercentage = isBlobTx ? MIN_BLOB_REPLACEMENT_BUMP_PERCENTAGE : MIN_REPLACEMENT_BUMP_PERCENTAGE;

const bumpPercentage = configBump > minBumpPercentage ? configBump : minBumpPercentage;

// Calculate minimum required fees based on previous attempt
const minPriorityFee = (previousGasPrice!.maxPriorityFeePerGas * (100n + bumpPercentage)) / 100n;
Expand All @@ -405,14 +442,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
spalladino marked this conversation as resolved.
Show resolved Hide resolved
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