diff --git a/service/message_selector_test.go b/service/message_selector_test.go index 25f0f5d9..c73eb2de 100644 --- a/service/message_selector_test.go +++ b/service/message_selector_test.go @@ -1,3 +1,4 @@ +//stm: #unit package service import ( @@ -8,8 +9,8 @@ import ( "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/big" - "github.com/filecoin-project/venus/pkg/constants" "github.com/filecoin-project/venus/venus-shared/testutil" + shared "github.com/filecoin-project/venus/venus-shared/types" types "github.com/filecoin-project/venus/venus-shared/types/messager" "github.com/stretchr/testify/assert" "go.uber.org/fx/fxtest" @@ -158,6 +159,7 @@ func TestAddrSelectMsgNum(t *testing.T) { } func TestSelectMessage(t *testing.T) { + //stm: @MESSENGER_SELECTOR_SELECT_MESSAGE_001, @MESSENGER_SELECTOR_SELECT_MESSAGE_002 ctx, cancel := context.WithCancel(context.Background()) defer cancel() @@ -187,6 +189,10 @@ func TestSelectMessage(t *testing.T) { assert.NoError(t, err) assert.Equal(t, &MsgSelectResult{}, res) + // If an error occurs retrieving nonce in tipset, return that error + _, err = ms.messageSelector.SelectMessage(ctx, &shared.TipSet{}) + assert.Error(t, err) + totalMsg := len(addrs) * 10 msgs := genMessages(addrs, account, totalMsg) assert.NoError(t, pushMessage(ctx, ms, msgs)) @@ -286,6 +292,7 @@ func TestSelectNum(t *testing.T) { } func TestEstimateMessageGas(t *testing.T) { + //stm: @MESSENGER_SELECTOR_ESTIMATE_MESSAGE_GAS_001 ctx, cancel := context.WithCancel(context.Background()) defer cancel() @@ -544,6 +551,25 @@ func TestSignMessageFailed(t *testing.T) { } } +func TestCapGasFee(t *testing.T) { + //stm: @MESSENGER_SELECTOR_CAP_MESSAGE_GAS_001 + msg := testhelper.NewMessage().Message + maxfee := func(msg *shared.Message) big.Int { + return big.Mul(big.NewInt(msg.GasLimit), msg.GasFeeCap) + } + oldFeeCap := big.NewInt(1000) + oldGasPremium := oldFeeCap + msg.GasLimit = 10000 + msg.GasFeeCap = oldFeeCap + msg.GasPremium = oldGasPremium + oldMaxFee := maxfee(&msg) + descedMaxFee := big.Div(oldMaxFee, big.NewInt(10)) + CapGasFee(&msg, descedMaxFee) + newMaxFee := maxfee(&msg) + assert.Less(t, big.Cmp(msg.GasPremium, oldGasPremium), 0) + assert.Less(t, big.Cmp(newMaxFee, oldMaxFee), 0) +} + type messageServiceHelper struct { fullNode *testhelper.MockFullNode walletProxy *gateway.MockWalletProxy @@ -655,7 +681,7 @@ func waitMsgWithTimeout(ctx context.Context, ms *MessageService, msgID string) ( resChan := make(chan *waitMsgRes) go func() { - res, err := ms.WaitMessage(ctx, msgID, constants.MessageConfidence) + res, err := ms.WaitMessage(ctx, msgID, 1) resChan <- &waitMsgRes{ msg: res, err: err, diff --git a/service/message_service_test.go b/service/message_service_test.go index 343e9df8..ad5cf613 100644 --- a/service/message_service_test.go +++ b/service/message_service_test.go @@ -1,13 +1,18 @@ +//stm: #unit package service import ( "context" "fmt" + "sync" "testing" "time" + "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/big" + "github.com/filecoin-project/venus/pkg/constants" + "github.com/filecoin-project/venus/venus-shared/testutil" shared "github.com/filecoin-project/venus/venus-shared/types" types "github.com/filecoin-project/venus/venus-shared/types/messager" "github.com/stretchr/testify/assert" @@ -48,6 +53,8 @@ func TestVerifyNetworkName(t *testing.T) { } func TestReplaceMessage(t *testing.T) { + //stm: @MESSENGER_SERVICE_REPLACE_MESSAGE_001, @MESSENGER_SERVICE_REPLACE_MESSAGE_002 + //stm: @MESSENGER_SERVICE_REPLACE_MESSAGE_003, @MESSENGER_SERVICE_REPLACE_MESSAGE_004 ctx, cancel := context.WithCancel(context.Background()) defer cancel() @@ -128,6 +135,18 @@ func TestReplaceMessage(t *testing.T) { assert.Equal(t, msg.GasFeeCap, res.GasFeeCap) assert.Equal(t, msg.GasPremium, res.GasPremium) } + + failedMessageReplace := func(*types.ReplacMessageParams) { + _, err = ms.ReplaceMessage(ctx, nil) + assert.Error(t, err) + } + + // param is nil, expect an error + failedMessageReplace(nil) + // message can't find, expect an error + failedMessageReplace(&types.ReplacMessageParams{ID: shared.NewUUID().String(), Auto: true}) + // message is already on chain, expect an error + failedMessageReplace(&types.ReplacMessageParams{ID: replacedMsgs[0].ID, Auto: true}) } func TestReconnectCheck(t *testing.T) { @@ -244,6 +263,7 @@ func TestReconnectCheck(t *testing.T) { } func TestMessageService_ProcessNewHead(t *testing.T) { + //stm: @MESSENGER_SERVICE_LIST_MESSAGE_BY_ADDRESS_001 ctx, cancel := context.WithCancel(context.Background()) defer cancel() @@ -395,6 +415,102 @@ func TestMessageService_ProcessNewHead(t *testing.T) { }) } +func TestMessageService_PushMessage(t *testing.T) { + ctx, cancel := context.WithCancel(context.Background()) + defer cancel() + + cfg := config.DefaultConfig() + cfg.MessageService.WaitingChainHeadStableDuration = time.Second * 2 + blockDelay := cfg.MessageService.WaitingChainHeadStableDuration * 2 + fsRepo := filestore.NewMockFileStore(t.TempDir()) + msh, err := newMessageServiceHelper(ctx, cfg, blockDelay, fsRepo) + assert.NoError(t, err) + + account := defaultLocalToken + addr := testutil.BlsAddressProvider()(t) + assert.NoError(t, msh.fullNode.AddActors([]address.Address{addr})) + assert.NoError(t, msh.walletProxy.AddAddress(account, []address.Address{addr})) + + lc := fxtest.NewLifecycle(t) + _ = StartNodeEvents(lc, msh.fullNode, msh.ms, msh.ms.log) + assert.NoError(t, lc.Start(ctx)) + defer lc.RequireStop() + + var pushedMsg *types.Message + + t.Run("push message:", func(t *testing.T) { + //stm: @MESSENGER_SERVICE_PUSH_MESSAGE_001, @MESSENGER_SERVICE_PUSH_MESSAGE_002, + //stm: @MESSENGER_SERVICE_PUSH_MESSAGE_WITH_ID_001, @MESSENGER_SERVICE_PUSH_MESSAGE_WITH_ID_002 + //stm: @MESSENGER_SERVICE_GET_MESSAGE_BY_UID_001, @MESSENGER_SERVICE_GET_MESSAGE_BY_UID_002 + //stm: @MESSENGER_SERVICE_LIST_MESSAGE_001 + rawMsg := testhelper.NewUnsignedMessage() + rawMsg.From = addr + uidStr, err := msh.ms.PushMessage(ctx, account, &rawMsg, nil) + assert.NoError(t, err) + _, err = shared.ParseUUID(uidStr) + assert.NoError(t, err) + + { + // pushing message would be failed + pushFailedMsg := testhelper.NewUnsignedMessage() + _, err = msh.ms.PushMessage(ctx, "invalid account", &pushFailedMsg, nil) + assert.Error(t, err) + // msg with uuid not exists, expect an error + _, err = msh.ms.GetMessageByUid(ctx, shared.NewUUID().String()) + assert.Error(t, err) + } + + pushedMsg, err = msh.ms.GetMessageByUid(ctx, uidStr) + assert.NoError(t, err) + assert.Equal(t, pushedMsg.ID, uidStr) + + { // list messages + msgs, err := msh.ms.ListMessage(ctx) + assert.NoError(t, err) + assert.Equal(t, len(msgs), 1) + assert.Equal(t, msgs[0].ID, uidStr) + + msgs, err = msh.ms.ListMessageByAddress(ctx, addr) + assert.NoError(t, err) + assert.Equal(t, len(msgs), 1) + assert.Equal(t, msgs[0].ID, uidStr) + } + + }) + + t.Run("wait message:", func(t *testing.T) { + //stm: @MESSENGER_SERVICE_WAIT_MESSAGE_001, @MESSENGER_SERVICE_WAIT_MESSAGE_002 + ctx, cancel := context.WithTimeout(context.TODO(), time.Minute*3) + defer cancel() + _, err := waitMsgWithTimeout(ctx, msh.ms, shared.NewUUID().String()) + assert.Error(t, err) + + wg := sync.WaitGroup{} + + waitOneMsg := func(msgID string, expectErr bool) { + wg.Add(1) + go func() { + defer wg.Done() + res, err := waitMsgWithTimeout(ctx, msh.ms, msgID) + if expectErr { + assert.Error(t, err) + return + } + assert.Equal(t, res.ID, msgID) + msgLookup, err := msh.fullNode.StateSearchMsg(ctx, shared.EmptyTSK, *res.SignedCid, constants.LookbackNoLimit, true) + assert.NoError(t, err) + assert.Equal(t, msgLookup.Height, abi.ChainEpoch(res.Height)) + assert.Equal(t, msgLookup.TipSet, res.TipSetKey) + assert.Equal(t, msgLookup.Receipt, *res.Receipt) + }() + } + + waitOneMsg(pushedMsg.ID, false) + waitOneMsg(shared.NewUUID().String(), true) + wg.Wait() + }) +} + func newMessageService(msh *messageServiceHelper, fsRepo filestore.FSRepo) *MessageService { return &MessageService{ repo: msh.ms.repo, diff --git a/service/message_state_refresh_test.go b/service/message_state_refresh_test.go index 1ca3b3f1..1156b0f6 100644 --- a/service/message_state_refresh_test.go +++ b/service/message_state_refresh_test.go @@ -20,7 +20,7 @@ import ( "github.com/filecoin-project/venus-messager/testhelper" ) -func TestDoRefershMessageState(t *testing.T) { +func TestDoRefreshMessageState(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) defer cancel() diff --git a/service/message_state_test.go b/service/message_state_test.go index c29e0556..0b54010f 100644 --- a/service/message_state_test.go +++ b/service/message_state_test.go @@ -1,3 +1,4 @@ +//stm: #unit package service import ( @@ -14,6 +15,9 @@ import ( ) func TestMessageStateCache(t *testing.T) { + //stm: @MESSENGER_STATE_GET_MESSAGE_001, @MESSENGER_STATE_SET_MESSAGE_ID_001, @MESSENGER_STATE_GET_MESSAGE_STATE_BY_CID_001 + //stm: @MESSENGER_STATE_UPDATE_MESSAGE_BY_CID_001, @MESSENGER_STATE_SET_MESSAGE_ID_001, @MESSENGER_STATE_DELETE_MESSAGE_001 + //stm: @MESSENGER_STATE_MUTATE_MESSAGE_001 fs := filestore.NewMockFileStore(t.TempDir()) db, err := sqlite.OpenSqlite(fs) assert.NoError(t, err) @@ -51,4 +55,11 @@ func TestMessageStateCache(t *testing.T) { state, flag = msgState.GetMessageStateByCid(msgs[1].Cid().String()) assert.True(t, flag) assert.Equal(t, types.OnChainMsg, state) + + msgState.DeleteMessage(msgs[0].ID) + + // Since `msg[0]` has already been removed, `GetMessageStateByCid` should returns `(Unknown, false)` + state, flag = msgState.GetMessageStateByCid(msgs[0].Cid().String()) + assert.Equal(t, state, types.UnKnown) + assert.False(t, flag) }