diff --git a/app/ante/ante.go b/app/ante/ante.go index 93d1d8aa13..a64f5ba08f 100644 --- a/app/ante/ante.go +++ b/app/ante/ante.go @@ -36,6 +36,7 @@ func NewAnteHandler( ante.NewSigGasConsumeDecorator(accountKeeper, sigGasConsumer), ante.NewSigVerificationDecorator(accountKeeper, signModeHandler), blobante.NewMinGasPFBDecorator(blobKeeper), + blobante.NewMaxBlobSizeDecorator(blobKeeper), NewGovProposalDecorator(), ante.NewIncrementSequenceDecorator(accountKeeper), ibcante.NewRedundantRelayDecorator(channelKeeper), diff --git a/app/test/check_tx_test.go b/app/test/check_tx_test.go index 38aca0bb4b..df8bfc8de8 100644 --- a/app/test/check_tx_test.go +++ b/app/test/check_tx_test.go @@ -150,10 +150,7 @@ func TestCheckTx(t *testing.T) { tx := blobfactory.RandBlobTxsWithAccounts(encCfg.TxConfig.TxEncoder(), tmrand.NewRand(), kr, nil, 10_000_000, 1, false, testutil.ChainID, accs[8:9])[0] return tx }, - // TODO: consider modifying CheckTx to return an error for this case - // so that consensus nodes do not propagate blobs that are too - // large. - expectedABCICode: abci.CodeTypeOK, + expectedABCICode: blobtypes.ErrTotalBlobSizeTooLarge.ABCICode(), }, } diff --git a/app/test/max_total_blob_size_test.go b/app/test/max_total_blob_size_test.go new file mode 100644 index 0000000000..2197259b6f --- /dev/null +++ b/app/test/max_total_blob_size_test.go @@ -0,0 +1,141 @@ +package app_test + +import ( + "context" + "testing" + + "github.com/celestiaorg/celestia-app/app" + "github.com/celestiaorg/celestia-app/app/encoding" + "github.com/celestiaorg/celestia-app/pkg/appconsts" + "github.com/celestiaorg/celestia-app/pkg/square" + "github.com/celestiaorg/celestia-app/test/util/testfactory" + "github.com/celestiaorg/celestia-app/test/util/testnode" + blobtypes "github.com/celestiaorg/celestia-app/x/blob/types" + sdk_tx "github.com/cosmos/cosmos-sdk/types/tx" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "github.com/stretchr/testify/suite" + abci "github.com/tendermint/tendermint/abci/types" + tmproto "github.com/tendermint/tendermint/proto/tendermint/types" + coretypes "github.com/tendermint/tendermint/types" + "google.golang.org/grpc" +) + +const ( + mebibyte = 1_048_576 // one mebibyte in bytes + squareSize = 64 +) + +func TestMaxTotalBlobSizeSuite(t *testing.T) { + if testing.Short() { + t.Skip("skipping max total blob size suite in short mode.") + } + suite.Run(t, &MaxTotalBlobSizeSuite{}) +} + +type MaxTotalBlobSizeSuite struct { + suite.Suite + + ecfg encoding.Config + accounts []string + cctx testnode.Context +} + +func (s *MaxTotalBlobSizeSuite) SetupSuite() { + t := s.T() + + s.accounts = testfactory.GenerateAccounts(1) + + tmConfig := testnode.DefaultTendermintConfig() + tmConfig.Mempool.MaxTxBytes = 10 * mebibyte + + cParams := testnode.DefaultParams() + cParams.Block.MaxBytes = 10 * mebibyte + + cfg := testnode.DefaultConfig(). + WithAccounts(s.accounts). + WithTendermintConfig(tmConfig). + WithConsensusParams(cParams) + + cctx, _, _ := testnode.NewNetwork(t, cfg) + s.cctx = cctx + s.ecfg = encoding.MakeConfig(app.ModuleEncodingRegisters...) + + require.NoError(t, cctx.WaitForNextBlock()) +} + +// TestSubmitPayForBlob_blobSizes verifies the tx response ABCI code when +// SubmitPayForBlob is invoked with different blob sizes. +func (s *MaxTotalBlobSizeSuite) TestSubmitPayForBlob_blobSizes() { + t := s.T() + + type testCase struct { + name string + blob *tmproto.Blob + // want is the expected tx response ABCI code. + want uint32 + } + testCases := []testCase{ + { + name: "1 byte blob", + blob: mustNewBlob(t, 1), + want: abci.CodeTypeOK, + }, + { + name: "1 mebibyte blob", + blob: mustNewBlob(t, mebibyte), + want: abci.CodeTypeOK, + }, + { + name: "2 mebibyte blob", + blob: mustNewBlob(t, 2*mebibyte), + want: blobtypes.ErrTotalBlobSizeTooLarge.ABCICode(), + }, + } + + signer := blobtypes.NewKeyringSigner(s.cctx.Keyring, s.accounts[0], s.cctx.ChainID) + + for _, tc := range testCases { + s.Run(tc.name, func() { + blobTx := newBlobTx(t, signer, s.cctx.GRPCClient, tc.blob) + res, err := blobtypes.BroadcastTx(context.TODO(), s.cctx.GRPCClient, sdk_tx.BroadcastMode_BROADCAST_MODE_BLOCK, blobTx) + require.NoError(t, err) + require.NotNil(t, res) + require.Equal(t, tc.want, res.TxResponse.Code, res.TxResponse.Logs) + + sq, err := square.Construct([][]byte{blobTx}, appconsts.LatestVersion, squareSize) + if tc.want == abci.CodeTypeOK { + // verify that if the tx was accepted, the blob can fit in a square + assert.NoError(t, err) + assert.False(t, sq.IsEmpty()) + } else { + // verify that if the tx was rejected, the blob can not fit in a square + assert.Error(t, err) + } + }) + } +} + +func newBlobTx(t *testing.T, signer *blobtypes.KeyringSigner, conn *grpc.ClientConn, blob *tmproto.Blob) coretypes.Tx { + addr, err := signer.GetSignerInfo().GetAddress() + require.NoError(t, err) + + msg, err := blobtypes.NewMsgPayForBlobs(addr.String(), blob) + require.NoError(t, err) + + err = signer.QueryAccountNumber(context.TODO(), conn) + require.NoError(t, err) + + options := []blobtypes.TxBuilderOption{blobtypes.SetGasLimit(1e9)} // set gas limit to 1 billion to avoid gas exhaustion + builder := signer.NewTxBuilder(options...) + stx, err := signer.BuildSignedTx(builder, msg) + require.NoError(t, err) + + rawTx, err := signer.EncodeTx(stx) + require.NoError(t, err) + + blobTx, err := coretypes.MarshalBlobTx(rawTx, blob) + require.NoError(t, err) + + return blobTx +} diff --git a/x/blob/ante/ante.go b/x/blob/ante/ante.go index e6cd0b77e3..68912bb38d 100644 --- a/x/blob/ante/ante.go +++ b/x/blob/ante/ante.go @@ -48,4 +48,5 @@ func (d MinGasPFBDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simulate bool type BlobKeeper interface { GasPerBlobByte(ctx sdk.Context) uint32 + GovMaxSquareSize(ctx sdk.Context) uint64 } diff --git a/x/blob/ante/ante_test.go b/x/blob/ante/ante_test.go index dde8bd933a..80acc877f2 100644 --- a/x/blob/ante/ante_test.go +++ b/x/blob/ante/ante_test.go @@ -13,7 +13,10 @@ import ( "github.com/stretchr/testify/require" ) -const testGasPerBlobByte = 10 +const ( + testGasPerBlobByte = 10 + testGovMaxSquareSize = 64 +) func TestPFBAnteHandler(t *testing.T) { txConfig := encoding.MakeConfig(app.ModuleEncodingRegisters...).TxConfig @@ -106,3 +109,7 @@ type mockBlobKeeper struct{} func (mockBlobKeeper) GasPerBlobByte(_ sdk.Context) uint32 { return testGasPerBlobByte } + +func (mockBlobKeeper) GovMaxSquareSize(_ sdk.Context) uint64 { + return testGovMaxSquareSize +} diff --git a/x/blob/ante/max_total_blob_size_ante.go b/x/blob/ante/max_total_blob_size_ante.go new file mode 100644 index 0000000000..a6fd279c65 --- /dev/null +++ b/x/blob/ante/max_total_blob_size_ante.go @@ -0,0 +1,88 @@ +package ante + +import ( + "github.com/celestiaorg/celestia-app/pkg/appconsts" + "github.com/celestiaorg/celestia-app/pkg/shares" + blobtypes "github.com/celestiaorg/celestia-app/x/blob/types" + + "cosmossdk.io/errors" + sdk "github.com/cosmos/cosmos-sdk/types" +) + +// MaxTotalBlobSizeDecorator helps to prevent a PFB from being included in a block +// but not fitting in a data square. +type MaxTotalBlobSizeDecorator struct { + k BlobKeeper +} + +func NewMaxBlobSizeDecorator(k BlobKeeper) MaxTotalBlobSizeDecorator { + return MaxTotalBlobSizeDecorator{k} +} + +// AnteHandle implements the Cosmos SDK AnteHandler function signature. It +// returns an error if tx contains a MsgPayForBlobs where the total blob size is +// greater than the max total blob size. +func (d MaxTotalBlobSizeDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simulate bool, next sdk.AnteHandler) (sdk.Context, error) { + if !ctx.IsCheckTx() { + return next(ctx, tx, simulate) + } + + max := d.maxTotalBlobSize(ctx) + for _, m := range tx.GetMsgs() { + if pfb, ok := m.(*blobtypes.MsgPayForBlobs); ok { + if total := getTotal(pfb.BlobSizes); total > max { + return ctx, errors.Wrapf(blobtypes.ErrTotalBlobSizeTooLarge, "total blob size %d exceeds max %d", total, max) + } + } + } + + return next(ctx, tx, simulate) +} + +// maxTotalBlobSize returns the max the number of bytes available for blobs in a +// data square based on the max square size. Note it is possible that txs with a +// total blob size less than this max still fail to be included in a block due +// to overhead from the PFB tx and/or padding shares. +func (d MaxTotalBlobSizeDecorator) maxTotalBlobSize(ctx sdk.Context) int { + squareSize := d.getMaxSquareSize(ctx) + totalShares := squareSize * squareSize + // The PFB tx share must occupy at least one share so the # of blob shares + // is at least one less than totalShares. + blobShares := totalShares - 1 + return shares.AvailableBytesFromSparseShares(blobShares) +} + +// getMaxSquareSize returns the maximum square size based on the current values +// for the relevant governance parameter and the versioned constant. +func (d MaxTotalBlobSizeDecorator) getMaxSquareSize(ctx sdk.Context) int { + // TODO: fix hack that forces the max square size for the first height to + // 64. This is due to our fork of the sdk not initializing state before + // BeginBlock of the first block. This is remedied in versions of the sdk + // and comet that have full support of PreparePropsoal, although + // celestia-app does not currently use those. see this PR for more details + // https://github.com/cosmos/cosmos-sdk/pull/14505 + if ctx.BlockHeader().Height <= 1 { + return int(appconsts.DefaultGovMaxSquareSize) + } + + upperBound := appconsts.SquareSizeUpperBound(ctx.ConsensusParams().Version.AppVersion) + govParam := d.k.GovMaxSquareSize(ctx) + return min(upperBound, int(govParam)) +} + +// getTotal returns the sum of the given sizes. +func getTotal(sizes []uint32) (sum int) { + for _, size := range sizes { + sum += int(size) + } + return sum +} + +// min returns the minimum of two ints. This function can be removed once we +// upgrade to Go 1.21. +func min(a, b int) int { + if a < b { + return a + } + return b +} diff --git a/x/blob/ante/max_total_blob_size_ante_test.go b/x/blob/ante/max_total_blob_size_ante_test.go new file mode 100644 index 0000000000..8e76ca455d --- /dev/null +++ b/x/blob/ante/max_total_blob_size_ante_test.go @@ -0,0 +1,141 @@ +package ante_test + +import ( + "testing" + + "github.com/celestiaorg/celestia-app/app" + "github.com/celestiaorg/celestia-app/app/encoding" + "github.com/celestiaorg/celestia-app/pkg/shares" + ante "github.com/celestiaorg/celestia-app/x/blob/ante" + blob "github.com/celestiaorg/celestia-app/x/blob/types" + sdk "github.com/cosmos/cosmos-sdk/types" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" +) + +const ( + mebibyte = 1_048_576 // 1 MiB + squareSize = 64 +) + +func TestMaxTotalBlobSizeAnteHandler(t *testing.T) { + type testCase struct { + name string + pfb *blob.MsgPayForBlobs + wantErr bool + } + + testCases := []testCase{ + // tests based on bytes + { + name: "PFB with 1 blob that is 1 byte", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{1}, + }, + wantErr: false, + }, + { + name: "PFB with 1 blob that is 1 MiB", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{mebibyte}, + }, + wantErr: false, + }, + { + name: "PFB with 1 blob that is 2 MiB", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{2 * mebibyte}, + }, + // This test case should return an error because a square size of 64 + // has exactly 2 MiB of total capacity so the total blob capacity + // will be slightly smaller than 2 MiB. + wantErr: true, + }, + { + name: "PFB with 2 blobs that are 1 byte each", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{1, 1}, + }, + wantErr: false, + }, + { + name: "PFB with 2 blobs that are 1 MiB each", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{mebibyte, mebibyte}, + }, + // This test case should return an error for the same reason a + // single blob that is 2 MiB returns an error. + wantErr: true, + }, + // tests based on shares + { + name: "PFB with 1 blob that is 1 share", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{uint32(shares.AvailableBytesFromSparseShares(1))}, + }, + wantErr: false, + }, + { + name: "PFB with 1 blob that occupies total square - 1", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{uint32(shares.AvailableBytesFromSparseShares((squareSize * squareSize) - 1))}, + }, + wantErr: false, + }, + { + name: "PFB with 1 blob that occupies total square", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{uint32(shares.AvailableBytesFromSparseShares(squareSize * squareSize))}, + }, + // This test case should return an error because if the blob + // occupies the total square, there is no space for the PFB tx + // share. + wantErr: true, + }, + { + name: "PFB with 2 blobs that are 1 share each", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{ + uint32(shares.AvailableBytesFromSparseShares(1)), + uint32(shares.AvailableBytesFromSparseShares(1)), + }, + }, + wantErr: false, + }, + { + name: "PFB with 2 blobs that occupy half the square each", + pfb: &blob.MsgPayForBlobs{ + BlobSizes: []uint32{ + uint32(shares.AvailableBytesFromSparseShares(squareSize * squareSize / 2)), + uint32(shares.AvailableBytesFromSparseShares(squareSize * squareSize / 2)), + }, + }, + wantErr: true, + }, + } + + txConfig := encoding.MakeConfig(app.ModuleEncodingRegisters...).TxConfig + + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + ctx := sdk.Context{}.WithIsCheckTx(true) + + txBuilder := txConfig.NewTxBuilder() + require.NoError(t, txBuilder.SetMsgs(tc.pfb)) + tx := txBuilder.GetTx() + + mbsd := ante.NewMaxBlobSizeDecorator(mockBlobKeeper{}) + _, err := mbsd.AnteHandle(ctx, tx, false, mockNext) + + if tc.wantErr { + assert.Error(t, err) + } else { + assert.NoError(t, err) + } + }) + } +} + +func mockNext(ctx sdk.Context, _ sdk.Tx, _ bool) (sdk.Context, error) { + return ctx, nil +} diff --git a/x/blob/types/errors.go b/x/blob/types/errors.go index 0b93876550..574eb75712 100644 --- a/x/blob/types/errors.go +++ b/x/blob/types/errors.go @@ -33,4 +33,5 @@ var ( ErrNoShareCommitments = errors.Register(ModuleName, 11135, "no share commitments provided") ErrInvalidNamespace = errors.Register(ModuleName, 11136, "invalid namespace") ErrInvalidNamespaceVersion = errors.Register(ModuleName, 11137, "invalid namespace version") + ErrTotalBlobSizeTooLarge = errors.Register(ModuleName, 11138, "total blob size too large") )