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

feat(accounts): Add codec.BinaryCodec and Gas to dependencies. #19068

Merged
merged 7 commits into from
Jan 17, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
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,141 changes: 1,102 additions & 39 deletions api/cosmos/accounts/testing/counter/v1/counter.pulsar.go

Large diffs are not rendered by default.

5 changes: 4 additions & 1 deletion codec/collections.go
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,10 @@ type protoMessage[T any] interface {
}

// CollValue inits a collections.ValueCodec for a generic gogo protobuf message.
func CollValue[T any, PT protoMessage[T]](cdc BinaryCodec) collcodec.ValueCodec[T] {
func CollValue[T any, PT protoMessage[T]](cdc interface {
Marshal(proto.Message) ([]byte, error)
Unmarshal([]byte, proto.Message) error
}) collcodec.ValueCodec[T] {
testinginprod marked this conversation as resolved.
Show resolved Hide resolved
return &collValue[T, PT]{cdc.(Codec), proto.MessageName(PT(new(T)))}
}

Expand Down
15 changes: 15 additions & 0 deletions proto/cosmos/accounts/testing/counter/v1/counter.proto
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,21 @@ message MsgIncreaseCounterResponse {
uint64 new_amount = 1;
}

// MsgTestDependencies is used to test the dependencies.
message MsgTestDependencies {}

// MsgTestDependenciesResponse is used to test the dependencies.
message MsgTestDependenciesResponse {
// chain_id is used to test that the header service correctly works.
string chain_id = 1;
// address is used to test address codec.
string address = 2;
// before_gas is used to test the gas meter reporting.
uint64 before_gas = 3;
// after_gas is used to test gas meter increasing.
uint64 after_gas = 4;
}

// QueryCounterRequest is used to query the counter value.
message QueryCounterRequest {}

Expand Down
30 changes: 30 additions & 0 deletions runtime/gas.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
package runtime

import (
"context"

"cosmossdk.io/core/gas"

sdk "github.com/cosmos/cosmos-sdk/types"
)

var _ gas.Service = (*GasService)(nil)

type GasService struct{}

func (g GasService) GetGasMeter(ctx context.Context) gas.Meter {
sdkCtx := sdk.UnwrapSDKContext(ctx)
return sdkCtx.GasMeter()
}

func (g GasService) GetBlockGasMeter(ctx context.Context) gas.Meter {
return sdk.UnwrapSDKContext(ctx).BlockGasMeter()
}

func (g GasService) WithGasMeter(ctx context.Context, meter gas.Meter) context.Context {
return sdk.UnwrapSDKContext(ctx).WithGasMeter(meter)
}

func (g GasService) WithBlockGasMeter(ctx context.Context, meter gas.Meter) context.Context {
return sdk.UnwrapSDKContext(ctx).WithBlockGasMeter(meter)
}
2 changes: 2 additions & 0 deletions simapp/app.go
Original file line number Diff line number Diff line change
Expand Up @@ -286,10 +286,12 @@ func NewSimApp(
app.AuthKeeper = authkeeper.NewAccountKeeper(appCodec, runtime.NewKVStoreService(keys[authtypes.StoreKey]), authtypes.ProtoBaseAccount, maccPerms, authcodec.NewBech32Codec(sdk.Bech32MainPrefix), sdk.Bech32MainPrefix, authtypes.NewModuleAddress(govtypes.ModuleName).String())

accountsKeeper, err := accounts.NewKeeper(
appCodec,
runtime.NewKVStoreService(keys[accounts.StoreKey]),
runtime.EventService{},
runtime.HeaderService{},
runtime.BranchService{},
runtime.GasService{},
app.AuthKeeper.AddressCodec(),
appCodec,
app.MsgServiceRouter(),
Expand Down
46 changes: 46 additions & 0 deletions tests/e2e/accounts/wiring_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
//go:build app_v1

package accounts

import (
"testing"

"cosmossdk.io/core/header"
counterv1 "cosmossdk.io/x/accounts/testing/counter/v1"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/stretchr/testify/require"
)

// TestDependencies aims to test wiring between different account components,
// inherited from the runtime, specifically:
// - address codec
// - binary codec
// - header service
// - gas service
func TestDependencies(t *testing.T) {
app := setupApp(t)
ak := app.AccountsKeeper
ctx := sdk.NewContext(app.CommitMultiStore(), false, app.Logger()).WithHeaderInfo(header.Info{ChainID: "chain-id"})

_, counterAddr, err := ak.Init(ctx, "counter", accCreator, &counterv1.MsgInit{
InitialValue: 0,
})
require.NoError(t, err)
// test dependencies
r, err := ak.Execute(ctx, counterAddr, []byte("test"), &counterv1.MsgTestDependencies{})
require.NoError(t, err)
res := r.(*counterv1.MsgTestDependenciesResponse)

// test gas
require.NotZero(t, res.BeforeGas)
require.NotZero(t, res.AfterGas)
require.Equal(t, uint64(10), res.AfterGas-res.BeforeGas)

// test header service
require.Equal(t, ctx.HeaderInfo().ChainID, res.ChainId)

// test address codec
wantAddr, err := app.AuthKeeper.AddressCodec().BytesToString(counterAddr)
require.NoError(t, err)
require.Equal(t, wantAddr, res.Address)
}
36 changes: 33 additions & 3 deletions x/accounts/internal/implementation/context.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import (
"encoding/binary"

"cosmossdk.io/collections"
"cosmossdk.io/core/gas"
"cosmossdk.io/core/header"
"cosmossdk.io/core/store"
"cosmossdk.io/x/accounts/internal/prefixstore"
Expand Down Expand Up @@ -123,9 +124,38 @@ func Whoami(ctx context.Context) []byte {
return ctx.Value(contextKey{}).(contextValue).whoami
}

type headerService struct{ header.Service }
type headerService struct{ hs header.Service }

func (h headerService) GetHeaderInfo(ctx context.Context) header.Info {
originalContext := ctx.Value(contextKey{}).(contextValue).originalContext
return h.Service.GetHeaderInfo(originalContext)
return h.hs.GetHeaderInfo(getOriginalContext(ctx))
}

var _ gas.Service = (*gasService)(nil)

type gasService struct {
gs gas.Service
}

func (g gasService) GetGasMeter(ctx context.Context) gas.Meter {
return g.gs.GetGasMeter(getOriginalContext(ctx))
}

func (g gasService) GetBlockGasMeter(ctx context.Context) gas.Meter {
return g.gs.GetBlockGasMeter(getOriginalContext(ctx))
}

func (g gasService) WithGasMeter(ctx context.Context, meter gas.Meter) context.Context {
v := ctx.Value(contextKey{}).(contextValue)
v.originalContext = g.gs.WithGasMeter(v.originalContext, meter)
testinginprod marked this conversation as resolved.
Show resolved Hide resolved
return context.WithValue(v.originalContext, contextKey{}, v)
}

func (g gasService) WithBlockGasMeter(ctx context.Context, meter gas.Meter) context.Context {
v := ctx.Value(contextKey{}).(contextValue)
v.originalContext = g.gs.WithBlockGasMeter(v.originalContext, meter)
return context.WithValue(v.originalContext, contextKey{}, v)
}

func getOriginalContext(ctx context.Context) context.Context {
return ctx.Value(contextKey{}).(contextValue).originalContext
}
9 changes: 9 additions & 0 deletions x/accounts/internal/implementation/encoding.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,19 @@ import (
"strings"

"github.com/cosmos/gogoproto/proto"
"google.golang.org/protobuf/runtime/protoiface"

codectypes "github.com/cosmos/cosmos-sdk/codec/types"
)

type ProtoMsg = protoiface.MessageV1

// ProtoMsgG is a generic interface for protobuf messages.
type ProtoMsgG[T any] interface {
*T
protoiface.MessageV1
}

type Any = codectypes.Any

func FindMessageByName(name string) (ProtoMsg, error) {
Expand Down
25 changes: 19 additions & 6 deletions x/accounts/internal/implementation/implementation.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,23 @@ import (

"cosmossdk.io/collections"
"cosmossdk.io/core/address"
"cosmossdk.io/core/gas"
"cosmossdk.io/core/header"
gogoproto "github.com/cosmos/gogoproto/proto"

"github.com/cosmos/cosmos-sdk/codec"
)

// Dependencies are passed to the constructor of a smart account.
type Dependencies struct {
SchemaBuilder *collections.SchemaBuilder
AddressCodec address.Codec
HeaderService header.Service
SchemaBuilder *collections.SchemaBuilder
AddressCodec address.Codec
HeaderService header.Service
GasService gas.Service
LegacyStateCodec interface {
tac0turtle marked this conversation as resolved.
Show resolved Hide resolved
Marshal(gogoproto.Message) ([]byte, error)
Unmarshal([]byte, gogoproto.Message) error
}
}

// AccountCreatorFunc is a function that creates an account.
Expand All @@ -22,17 +31,21 @@ type AccountCreatorFunc = func(deps Dependencies) (string, Account, error)
// MakeAccountsMap creates a map of account names to account implementations
// from a list of account creator functions.
func MakeAccountsMap(
cdc codec.BinaryCodec,
addressCodec address.Codec,
hs header.Service,
gs gas.Service,
accounts []AccountCreatorFunc,
) (map[string]Implementation, error) {
accountsMap := make(map[string]Implementation, len(accounts))
for _, makeAccount := range accounts {
stateSchemaBuilder := collections.NewSchemaBuilderFromAccessor(openKVStore)
deps := Dependencies{
SchemaBuilder: stateSchemaBuilder,
AddressCodec: addressCodec,
HeaderService: headerService{hs},
SchemaBuilder: stateSchemaBuilder,
AddressCodec: addressCodec,
HeaderService: headerService{hs},
GasService: gasService{gs},
LegacyStateCodec: cdc,
}
name, accountInterface, err := makeAccount(deps)
if err != nil {
Expand Down
9 changes: 0 additions & 9 deletions x/accounts/internal/implementation/protoaccount.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,17 +5,8 @@ import (
"fmt"

"google.golang.org/protobuf/proto"
"google.golang.org/protobuf/runtime/protoiface"
)

type ProtoMsg = protoiface.MessageV1

// ProtoMsgG is a generic interface for protobuf messages.
type ProtoMsgG[T any] interface {
*T
protoiface.MessageV1
}

// RegisterInitHandler registers an initialisation handler for a smart account that uses protobuf.
func RegisterInitHandler[
Req any, ProtoReq ProtoMsgG[Req], Resp any, ProtoResp ProtoMsgG[Resp],
testinginprod marked this conversation as resolved.
Show resolved Hide resolved
Expand Down
7 changes: 6 additions & 1 deletion x/accounts/keeper.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,13 @@ import (
"cosmossdk.io/core/address"
"cosmossdk.io/core/branch"
"cosmossdk.io/core/event"
"cosmossdk.io/core/gas"
"cosmossdk.io/core/header"
"cosmossdk.io/core/store"
"cosmossdk.io/x/accounts/accountstd"
"cosmossdk.io/x/accounts/internal/implementation"

"github.com/cosmos/cosmos-sdk/codec"
)

var (
Expand Down Expand Up @@ -61,10 +64,12 @@ type InterfaceRegistry interface {
}

func NewKeeper(
cdc codec.BinaryCodec,
ss store.KVStoreService,
es event.Service,
hs header.Service,
bs branch.Service,
gs gas.Service,
addressCodec address.Codec,
signerProvider SignerProvider,
execRouter MsgRouter,
Expand Down Expand Up @@ -92,7 +97,7 @@ func NewKeeper(
return Keeper{}, err
}
keeper.Schema = schema
keeper.accounts, err = implementation.MakeAccountsMap(keeper.addressCodec, hs, accounts)
keeper.accounts, err = implementation.MakeAccountsMap(cdc, keeper.addressCodec, hs, gs, accounts)
if err != nil {
return Keeper{}, err
}
Expand Down
56 changes: 52 additions & 4 deletions x/accounts/testing/counter/counter.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,22 +6,32 @@ import (
"fmt"

"cosmossdk.io/collections"
"cosmossdk.io/core/address"
"cosmossdk.io/core/gas"
"cosmossdk.io/core/header"
"cosmossdk.io/x/accounts/accountstd"
counterv1 "cosmossdk.io/x/accounts/testing/counter/v1"

"github.com/cosmos/cosmos-sdk/codec"
)

var (
OwnerPrefix = collections.NewPrefix(0)
CounterPrefix = collections.NewPrefix(1)
OwnerPrefix = collections.NewPrefix(0)
CounterPrefix = collections.NewPrefix(1)
TestStateCodecPrefix = collections.NewPrefix(2)
)

var _ accountstd.Interface = (*Account)(nil)

// NewAccount creates a new account.
func NewAccount(d accountstd.Dependencies) (Account, error) {
return Account{
Owner: collections.NewItem(d.SchemaBuilder, OwnerPrefix, "owner", collections.BytesValue),
Counter: collections.NewItem(d.SchemaBuilder, CounterPrefix, "counter", collections.Uint64Value),
Owner: collections.NewItem(d.SchemaBuilder, OwnerPrefix, "owner", collections.BytesValue),
Counter: collections.NewItem(d.SchemaBuilder, CounterPrefix, "counter", collections.Uint64Value),
TestStateCodec: collections.NewItem(d.SchemaBuilder, TestStateCodecPrefix, "test_state_codec", codec.CollValue[counterv1.MsgTestDependencies](d.LegacyStateCodec)),
hs: d.HeaderService,
addressCodec: d.AddressCodec,
gs: d.GasService,
}, nil
}

Expand All @@ -32,6 +42,13 @@ type Account struct {
Owner collections.Item[[]byte]
// Counter is the counter value.
Counter collections.Item[uint64]
// TestStateCodec is used to test the binary codec provided by the runtime.
// It simply stores the MsgInit.
TestStateCodec collections.Item[counterv1.MsgTestDependencies]

hs header.Service
addressCodec address.Codec
gs gas.Service
}

func (a Account) Init(ctx context.Context, msg *counterv1.MsgInit) (*counterv1.MsgInitResponse, error) {
Expand Down Expand Up @@ -79,12 +96,43 @@ func (a Account) QueryCounter(ctx context.Context, _ *counterv1.QueryCounterRequ
}, nil
}

func (a Account) TestDependencies(ctx context.Context, _ *counterv1.MsgTestDependencies) (*counterv1.MsgTestDependenciesResponse, error) {
// test binary codec
err := a.TestStateCodec.Set(ctx, counterv1.MsgTestDependencies{})
if err != nil {
return nil, err
}

// test address codec
me := accountstd.Whoami(ctx)
meStr, err := a.addressCodec.BytesToString(me)
if err != nil {
return nil, err
}

// test header service
chainID := a.hs.GetHeaderInfo(ctx).ChainID

// test gas meter
gasBefore := a.gs.GetGasMeter(ctx).GasConsumedToLimit()
a.gs.GetGasMeter(ctx).ConsumeGas(10, "test")
gasAfter := a.gs.GetGasMeter(ctx).GasConsumedToLimit()

return &counterv1.MsgTestDependenciesResponse{
ChainId: chainID,
Address: meStr,
BeforeGas: gasBefore,
AfterGas: gasAfter,
}, nil
}

func (a Account) RegisterInitHandler(builder *accountstd.InitBuilder) {
accountstd.RegisterInitHandler(builder, a.Init)
}

func (a Account) RegisterExecuteHandlers(builder *accountstd.ExecuteBuilder) {
accountstd.RegisterExecuteHandler(builder, a.IncreaseCounter)
accountstd.RegisterExecuteHandler(builder, a.TestDependencies)
}

func (a Account) RegisterQueryHandlers(builder *accountstd.QueryBuilder) {
Expand Down
Loading
Loading