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

refactor(tests): Port auth integration tests to server v2 #22554

Merged
merged 33 commits into from
Nov 27, 2024
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
432fbbf
add auth app test
sontrinh16 Nov 7, 2024
a69a301
add run msg
sontrinh16 Nov 8, 2024
2942559
add auth integration tests
sontrinh16 Nov 19, 2024
940ce7f
minor
sontrinh16 Nov 19, 2024
8409748
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 19, 2024
bd13abb
add router service factory
sontrinh16 Nov 20, 2024
f105e67
fix conflict
sontrinh16 Nov 20, 2024
ce64f48
more resolve conflict
sontrinh16 Nov 20, 2024
4941f18
lint
sontrinh16 Nov 20, 2024
cdcfede
fix tests
sontrinh16 Nov 20, 2024
71affa1
go mod tidy
sontrinh16 Nov 21, 2024
aecd1f1
lint more
sontrinh16 Nov 21, 2024
b9b1a2c
lint
sontrinh16 Nov 21, 2024
2b94d76
lint
sontrinh16 Nov 21, 2024
78767ec
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 21, 2024
400e3e1
clean up and fix tests
sontrinh16 Nov 21, 2024
14c8813
Merge branch 'son/port_auth_integration_tests_to_server_v2' of https:…
sontrinh16 Nov 21, 2024
c25f6cf
fix tests
sontrinh16 Nov 21, 2024
037d482
clean up
sontrinh16 Nov 21, 2024
709ea17
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 21, 2024
422cca9
move router builder to runtime
sontrinh16 Nov 22, 2024
8c8f0e4
Merge branch 'son/port_auth_integration_tests_to_server_v2' of https:…
sontrinh16 Nov 22, 2024
2c1f3a4
godoc
sontrinh16 Nov 22, 2024
eb6f21b
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 22, 2024
f2b6fff
godoc
sontrinh16 Nov 22, 2024
2bfe906
Merge branch 'son/port_auth_integration_tests_to_server_v2' of https:…
sontrinh16 Nov 22, 2024
c60f7db
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 22, 2024
48efcfe
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 25, 2024
5e2e7ae
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 25, 2024
8d95e16
move service custom to start config
sontrinh16 Nov 26, 2024
53c884b
Merge branch 'son/port_auth_integration_tests_to_server_v2' of https:…
sontrinh16 Nov 26, 2024
5ad8840
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 26, 2024
455858c
Merge branch 'main' into son/port_auth_integration_tests_to_server_v2
sontrinh16 Nov 26, 2024
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
54 changes: 51 additions & 3 deletions runtime/v2/module.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,12 @@ import (
autocliv1 "cosmossdk.io/api/cosmos/autocli/v1"
reflectionv1 "cosmossdk.io/api/cosmos/reflection/v1"
appmodulev2 "cosmossdk.io/core/appmodule/v2"
"cosmossdk.io/core/branch"
"cosmossdk.io/core/comet"
"cosmossdk.io/core/event"
"cosmossdk.io/core/header"
"cosmossdk.io/core/registry"
"cosmossdk.io/core/router"
"cosmossdk.io/core/store"
"cosmossdk.io/core/transaction"
"cosmossdk.io/depinject"
Expand Down Expand Up @@ -213,27 +215,66 @@ func ProvideEnvironment(
memKvService store.MemoryStoreService,
headerService header.Service,
eventService event.Service,
branchService branch.Service,
routerBuilder RouterServiceBuilder,
) appmodulev2.Environment {
return appmodulev2.Environment{
Logger: logger,
BranchService: stf.BranchService{},
BranchService: branchService,
EventService: eventService,
GasService: stf.NewGasMeterService(),
HeaderService: headerService,
QueryRouterService: stf.NewQueryRouterService(),
MsgRouterService: stf.NewMsgRouterService([]byte(key.Name())),
QueryRouterService: routerBuilder.BuildQueryRouter(),
MsgRouterService: routerBuilder.BuildMsgRouter([]byte(key.Name())),
TransactionService: services.NewContextAwareTransactionService(),
KVStoreService: kvService,
MemStoreService: memKvService,
}
}

// RouterServiceBuilder builds the msg router and query router service during app initialization.
// this is mainly use for testing to override message router service in the environment and not in stf.
type RouterServiceBuilder interface {
sontrinh16 marked this conversation as resolved.
Show resolved Hide resolved
// BuildMsgRouter return a msg router service.
// - actor is the module store key.
BuildMsgRouter(actor []byte) router.Service
BuildQueryRouter() router.Service
}

type RouterServiceFactory func([]byte) router.Service

// routerBuilder implements RouterServiceBuilder
type routerBuilder struct {
msgRouterServiceFactory RouterServiceFactory
queryRouter router.Service
}

func NewRouterBuilder(
Copy link
Member

Choose a reason for hiding this comment

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

Let's kill this as you aren't supposed to use construct that manually

msgRouterServiceFactory RouterServiceFactory,
queryRouter router.Service,
) RouterServiceBuilder {
return routerBuilder{
msgRouterServiceFactory: msgRouterServiceFactory,
queryRouter: queryRouter,
}
}

func (b routerBuilder) BuildMsgRouter(actor []byte) router.Service {
return b.msgRouterServiceFactory(actor)
}

func (b routerBuilder) BuildQueryRouter() router.Service {
return b.queryRouter
}

// DefaultServiceBindings provides default services for the following service interfaces:
// - store.KVStoreServiceFactory
// - header.Service
// - comet.Service
// - event.Service
// - store/v2/root.Builder
// - branch.Service
// - RouterServiceBuilder
//
// They are all required. For most use cases these default services bindings should be sufficient.
// Power users (or tests) may wish to provide their own services bindings, in which case they must
Expand All @@ -246,16 +287,23 @@ func DefaultServiceBindings() depinject.Config {
stf.NewKVStoreService(actor),
)
}
routerBuilder RouterServiceBuilder = routerBuilder{
msgRouterServiceFactory: stf.NewMsgRouterService,
queryRouter: stf.NewQueryRouterService(),
}
cometService comet.Service = &services.ContextAwareCometInfoService{}
headerService = services.NewGenesisHeaderService(stf.HeaderService{})
eventService = services.NewGenesisEventService(stf.NewEventService())
storeBuilder = root.NewBuilder()
branchService = stf.BranchService{}
)
return depinject.Supply(
kvServiceFactory,
routerBuilder,
headerService,
cometService,
eventService,
storeBuilder,
branchService,
)
}
57 changes: 57 additions & 0 deletions tests/integration/v2/app.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import (
cmttypes "github.com/cometbft/cometbft/types"
"github.com/stretchr/testify/require"

corebranch "cosmossdk.io/core/branch"
"cosmossdk.io/core/comet"
corecontext "cosmossdk.io/core/context"
"cosmossdk.io/core/server"
Expand Down Expand Up @@ -53,6 +54,8 @@ const (

type stateMachineTx = transaction.Tx

type handler = func(ctx context.Context) (transaction.Msg, error)

// DefaultConsensusParams defines the default CometBFT consensus params used in
// SimApp testing.
var DefaultConsensusParams = &cmtproto.ConsensusParams{
Expand Down Expand Up @@ -116,12 +119,30 @@ func DefaultStartUpConfig(t *testing.T) StartupConfig {
}
}

// RunMsgConfig defines the run message configuration.
type RunMsgConfig struct {
Commit bool
}

// Option is a function that can be used to configure the integration app.
type Option func(*RunMsgConfig)

// WithAutomaticCommit enables automatic commit.
// This means that the integration app will automatically commit the state after each msg.
func WithAutomaticCommit() Option {
return func(cfg *RunMsgConfig) {
cfg.Commit = true
}
}

// NewApp initializes a new runtime.App. A Nop logger is set in runtime.App.
// appConfig defines the application configuration (f.e. app_config.go).
// extraOutputs defines the extra outputs to be assigned by the dependency injector (depinject).
func NewApp(
appConfig depinject.Config,
startupConfig StartupConfig,
branchService corebranch.Service,
sontrinh16 marked this conversation as resolved.
Show resolved Hide resolved
routerServiceBuilder runtime.RouterServiceBuilder,
extraOutputs ...interface{},
) (*App, error) {
// create the app with depinject
Expand All @@ -139,6 +160,18 @@ func NewApp(
err error
)

// set default router service builder if not provided
if routerServiceBuilder == nil {
routerServiceBuilder = runtime.NewRouterBuilder(
stf.NewMsgRouterService, stf.NewQueryRouterService(),
)
}

// set default branch service if not provided
if branchService == nil {
branchService = stf.BranchService{}
}

if err := depinject.Inject(
depinject.Configs(
appConfig,
Expand All @@ -159,6 +192,8 @@ func NewApp(
kvFactory,
&eventService{},
storeBuilder,
branchService,
routerServiceBuilder,
),
depinject.Invoke(
std.RegisterInterfaces,
Expand Down Expand Up @@ -397,6 +432,28 @@ func (a *App) SignCheckDeliver(
return txResult
}

func (app *App) RunMsg(t *testing.T, ctx context.Context, handler handler, option ...Option) (resp transaction.Msg, err error) {
sontrinh16 marked this conversation as resolved.
Show resolved Hide resolved
// set options
cfg := &RunMsgConfig{}
for _, opt := range option {
opt(cfg)
}

// need to have integration context
integrationCtx, ok := ctx.Value(contextKey).(*integrationContext)
require.True(t, ok)

resp, err = handler(ctx)

if cfg.Commit {
app.lastHeight++
_, err := app.Commit(integrationCtx.state)
require.NoError(t, err)
}

return resp, err
}
Comment on lines +433 to +453
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Improve error handling and input validation

The current implementation has several areas for improvement:

  1. Using require.True in a function that returns error is inconsistent
  2. Missing error wrapping for better error context
  3. No validation of nil handler

Consider this improved implementation:

-func (app *App) RunMsg(t *testing.T, ctx context.Context, handler handler, option ...Option) (resp transaction.Msg, err error) {
+func (app *App) RunMsg(t *testing.T, ctx context.Context, handler handler, option ...Option) (transaction.Msg, error) {
+    if handler == nil {
+        return nil, fmt.Errorf("handler cannot be nil")
+    }
+
     // set options
     cfg := &RunMsgConfig{}
     for _, opt := range option {
         opt(cfg)
     }

     // need to have integration context
     integrationCtx, ok := ctx.Value(contextKey).(*integrationContext)
-    require.True(t, ok)
+    if !ok {
+        return nil, fmt.Errorf("context must be an integration context")
+    }

-    resp, err = handler(ctx)
+    resp, err := handler(ctx)
+    if err != nil {
+        return nil, fmt.Errorf("handler execution failed: %w", err)
+    }

     if cfg.Commit {
         app.lastHeight++
         _, err := app.Commit(integrationCtx.state)
-        require.NoError(t, err)
+        if err != nil {
+            return nil, fmt.Errorf("failed to commit state: %w", err)
+        }
     }

-    return resp, err
+    return resp, nil
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
func (app *App) RunMsg(t *testing.T, ctx context.Context, handler handler, option ...Option) (resp transaction.Msg, err error) {
// set options
cfg := &RunMsgConfig{}
for _, opt := range option {
opt(cfg)
}
// need to have integration context
integrationCtx, ok := ctx.Value(contextKey).(*integrationContext)
require.True(t, ok)
resp, err = handler(ctx)
if cfg.Commit {
app.lastHeight++
_, err := app.Commit(integrationCtx.state)
require.NoError(t, err)
}
return resp, err
}
func (app *App) RunMsg(t *testing.T, ctx context.Context, handler handler, option ...Option) (transaction.Msg, error) {
if handler == nil {
return nil, fmt.Errorf("handler cannot be nil")
}
// set options
cfg := &RunMsgConfig{}
for _, opt := range option {
opt(cfg)
}
// need to have integration context
integrationCtx, ok := ctx.Value(contextKey).(*integrationContext)
if !ok {
return nil, fmt.Errorf("context must be an integration context")
}
resp, err := handler(ctx)
if err != nil {
return nil, fmt.Errorf("handler execution failed: %w", err)
}
if cfg.Commit {
app.lastHeight++
_, err := app.Commit(integrationCtx.state)
if err != nil {
return nil, fmt.Errorf("failed to commit state: %w", err)
}
}
return resp, nil
}


// CheckBalance checks the balance of the given address.
func (a *App) CheckBalance(
t *testing.T, ctx context.Context, addr sdk.AccAddress, expected sdk.Coins, keeper bankkeeper.Keeper,
Expand Down
185 changes: 185 additions & 0 deletions tests/integration/v2/auth/accounts_retro_compatibility_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
package auth

import (
"context"
"errors"
"testing"

gogotypes "github.com/cosmos/gogoproto/types"
"github.com/stretchr/testify/require"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"

"cosmossdk.io/x/accounts/accountstd"
basev1 "cosmossdk.io/x/accounts/defaults/base/v1"

codectypes "github.com/cosmos/cosmos-sdk/codec/types"
"github.com/cosmos/cosmos-sdk/crypto/keys/secp256k1"
authkeeper "github.com/cosmos/cosmos-sdk/x/auth/keeper"
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
)

var _ accountstd.Interface = mockRetroCompatAccount{}

type mockRetroCompatAccount struct {
retroCompat *authtypes.QueryLegacyAccountResponse
address []byte
}

var (
valid = &mockRetroCompatAccount{
retroCompat: &authtypes.QueryLegacyAccountResponse{
Account: &codectypes.Any{},
Base: &authtypes.BaseAccount{
Address: "test",
PubKey: nil,
AccountNumber: 10,
Sequence: 20,
},
},
}

noInfo = &mockRetroCompatAccount{
retroCompat: &authtypes.QueryLegacyAccountResponse{
Account: &codectypes.Any{},
},
}
noImplement = &mockRetroCompatAccount{
retroCompat: nil,
}
)

func newMockRetroCompatAccount(name string, acc accountstd.Interface) accountstd.AccountCreatorFunc {
return func(_ accountstd.Dependencies) (string, accountstd.Interface, error) {
_, ok := acc.(*mockRetroCompatAccount)
if !ok {
return name, nil, errors.New("invalid account type")
}
return name, acc, nil
}
}

func ProvideMockRetroCompatAccountValid() accountstd.DepinjectAccount {
return accountstd.DepinjectAccount{MakeAccount: newMockRetroCompatAccount("valid", valid)}
}

func ProvideMockRetroCompatAccountNoInfo() accountstd.DepinjectAccount {
return accountstd.DepinjectAccount{MakeAccount: newMockRetroCompatAccount("no_info", noInfo)}
}

func ProvideMockRetroCompatAccountNoImplement() accountstd.DepinjectAccount {
return accountstd.DepinjectAccount{MakeAccount: newMockRetroCompatAccount("no_implement", noImplement)}
}

func (m mockRetroCompatAccount) RegisterInitHandler(builder *accountstd.InitBuilder) {
accountstd.RegisterInitHandler(builder, func(ctx context.Context, req *gogotypes.Empty) (*gogotypes.Empty, error) {
return &gogotypes.Empty{}, nil
})
}

func (m mockRetroCompatAccount) RegisterExecuteHandlers(_ *accountstd.ExecuteBuilder) {}

func (m mockRetroCompatAccount) RegisterQueryHandlers(builder *accountstd.QueryBuilder) {
if m.retroCompat == nil {
return
}
accountstd.RegisterQueryHandler(builder, func(ctx context.Context, req *authtypes.QueryLegacyAccount) (*authtypes.QueryLegacyAccountResponse, error) {
return m.retroCompat, nil
})
}

func TestAuthToAccountsGRPCCompat(t *testing.T) {
accs := map[string]accountstd.Interface{
"valid": valid,
"no_info": noInfo,
"no_implement": noImplement,
}

f := createTestSuite(t)

// init three accounts
for n, a := range accs {
_, addr, err := f.accountsKeeper.Init(f.ctx, n, []byte("me"), &gogotypes.Empty{}, nil)
require.NoError(t, err)
a.(*mockRetroCompatAccount).address = addr
}

qs := authkeeper.NewQueryServer(f.authKeeper)

t.Run("account supports info and account query", func(t *testing.T) {
infoResp, err := qs.AccountInfo(f.ctx, &authtypes.QueryAccountInfoRequest{
Address: f.mustAddr(valid.address),
})
require.NoError(t, err)
require.Equal(t, infoResp.Info, valid.retroCompat.Base)

accountResp, err := qs.Account(f.ctx, &authtypes.QueryAccountRequest{
Address: f.mustAddr(noInfo.address),
})
require.NoError(t, err)
require.Equal(t, accountResp.Account, valid.retroCompat.Account)
})

t.Run("account only supports account query, not info", func(t *testing.T) {
_, err := qs.AccountInfo(f.ctx, &authtypes.QueryAccountInfoRequest{
Address: f.mustAddr(noInfo.address),
})
require.Error(t, err)
require.Equal(t, status.Code(err), codes.NotFound)

resp, err := qs.Account(f.ctx, &authtypes.QueryAccountRequest{
Address: f.mustAddr(noInfo.address),
})
require.NoError(t, err)
require.Equal(t, resp.Account, valid.retroCompat.Account)
})

t.Run("account does not support any retro compat", func(t *testing.T) {
_, err := qs.AccountInfo(f.ctx, &authtypes.QueryAccountInfoRequest{
Address: f.mustAddr(noImplement.address),
})
require.Error(t, err)
require.Equal(t, status.Code(err), codes.NotFound)

_, err = qs.Account(f.ctx, &authtypes.QueryAccountRequest{
Address: f.mustAddr(noImplement.address),
})

require.Error(t, err)
require.Equal(t, status.Code(err), codes.NotFound)
})
}

func TestAccountsBaseAccountRetroCompat(t *testing.T) {
f := createTestSuite(t)
// init a base acc
anyPk, err := codectypes.NewAnyWithValue(secp256k1.GenPrivKey().PubKey())
require.NoError(t, err)

// we init two accounts. Account number should start with 4
// since the first three accounts are fee_collector, bonded_tokens_pool, not_bonded_tokens_pool
// generated by init genesis plus one more genesis account, which make the current account number 4.
_, _, err = f.accountsKeeper.Init(f.ctx, "base", []byte("me"), &basev1.MsgInit{PubKey: anyPk}, nil)
Comment on lines +159 to +162
Copy link
Contributor

Choose a reason for hiding this comment

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

💡 Codebase verification

Based on the search results and the previous output, I can now generate the final response:

Account number assumption is incorrect

The comment assumes account number 4 based on fee_collector, bonded_tokens_pool, and not_bonded_tokens_pool being the first three accounts. However, from the genesis files we can see that:

  • fee_collector has account_number 2
  • distribution has account_number 3
  • bonded_tokens_pool has account_number 4
  • not_bonded_tokens_pool has account_number 5

The comment should be updated to reflect the correct account sequence and number.

🔗 Analysis chain

Verify account number assumption

The comment makes assumptions about account numbers that should be verified programmatically.


Based on the search results, I can verify the account number assumption. The genesis files and test files show that the following module accounts are initialized in this order:

  1. fee_collector (account_number: 2)
  2. distribution (account_number: 3)
  3. bonded_tokens_pool (account_number: 4)
  4. not_bonded_tokens_pool (account_number: 5)

Let me verify the genesis account initialization with one more search:


Let me try another search to find the genesis account initialization:

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the account number assumption by checking genesis accounts

# Search for genesis account initialization
rg -A 5 "init genesis.*account" 

# Search for fee collector and pool accounts initialization
rg -A 5 "fee_collector|bonded_tokens_pool|not_bonded_tokens_pool"

Length of output: 57957


Script:

#!/bin/bash
# Search for genesis account initialization in the test file
rg -B 5 -A 5 "Init.*genesis" tests/integration/v2/auth/accounts_retro_compatibility_test.go

Length of output: 93


Script:

#!/bin/bash
# Search for genesis account setup in the test file
rg -B 10 -A 2 "genesis account" tests/integration/v2/auth/accounts_retro_compatibility_test.go

# Also search for account initialization in the test setup
rg -B 5 -A 5 "SetupTest|Init.*account" tests/integration/v2/auth/accounts_retro_compatibility_test.go

Length of output: 1222

require.NoError(t, err)

_, addr, err := f.accountsKeeper.Init(f.ctx, "base", []byte("me"), &basev1.MsgInit{PubKey: anyPk}, nil)
require.NoError(t, err)

// try to query it via auth
qs := authkeeper.NewQueryServer(f.authKeeper)

r, err := qs.Account(f.ctx, &authtypes.QueryAccountRequest{
Address: f.mustAddr(addr),
})
require.NoError(t, err)
require.NotNil(t, r.Account)

info, err := qs.AccountInfo(f.ctx, &authtypes.QueryAccountInfoRequest{
Address: f.mustAddr(addr),
})
require.NoError(t, err)
require.NotNil(t, info.Info)
require.Equal(t, info.Info.PubKey, anyPk)
// Account number should be 5
require.Equal(t, info.Info.AccountNumber, uint64(5))
}
Loading
Loading