Files
qfixpt/quickfix/in_session_test.go
2026-03-12 12:14:13 -03:00

431 lines
12 KiB
Go

// Copyright (c) quickfixengine.org All rights reserved.
//
// This file may be distributed under the terms of the quickfixengine.org
// license as defined by quickfixengine.org and appearing in the file
// LICENSE included in the packaging of this file.
//
// This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING
// THE WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A
// PARTICULAR PURPOSE.
//
// See http://www.quickfixengine.org/LICENSE for licensing information.
//
// Contact ask@quickfixengine.org if any conditions of this licensing
// are not clear to you.
package quickfix
import (
"testing"
"time"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/suite"
"quantex.com/qfixpt/quickfix/internal"
)
type InSessionTestSuite struct {
SessionSuiteRig
}
func TestInSessionTestSuite(t *testing.T) {
suite.Run(t, new(InSessionTestSuite))
}
func (s *InSessionTestSuite) SetupTest() {
s.Init()
s.session.State = inSession{}
}
func (s *InSessionTestSuite) TestPreliminary() {
s.True(s.session.IsLoggedOn())
s.True(s.session.IsConnected())
s.True(s.session.IsSessionTime())
}
func (s *InSessionTestSuite) TestLogout() {
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.MockApp.On("OnLogout")
s.session.fixMsgIn(s.session, s.Logout())
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.NextTargetMsgSeqNum(2)
s.NextSenderMsgSeqNum(2)
}
func (s *InSessionTestSuite) TestLogoutEnableLastMsgSeqNumProcessed() {
s.session.EnableLastMsgSeqNumProcessed = true
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.MockApp.On("OnLogout")
s.session.fixMsgIn(s.session, s.Logout())
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.FieldEquals(tagLastMsgSeqNumProcessed, 1, s.MockApp.lastToAdmin.Header)
}
func (s *InSessionTestSuite) TestLogoutResetOnLogout() {
s.session.ResetOnLogout = true
s.MockApp.On("ToApp").Return(nil)
s.Nil(s.queueForSend(s.NewOrderSingle()))
s.MockApp.AssertExpectations(s.T())
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.MockApp.On("OnLogout")
s.session.fixMsgIn(s.session, s.Logout())
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
s.LastToAppMessageSent()
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.NextTargetMsgSeqNum(1)
s.NextSenderMsgSeqNum(1)
s.NoMessageQueued()
}
func (s *InSessionTestSuite) TestLogoutTargetTooHigh() {
s.MessageFactory.seqNum = 5
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.MockApp.On("OnLogout")
s.session.fixMsgIn(s.session, s.Logout())
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.NextTargetMsgSeqNum(1)
s.NextSenderMsgSeqNum(2)
}
func (s *InSessionTestSuite) TestTimeoutNeedHeartbeat() {
s.MockApp.On("ToAdmin").Return(nil)
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeHeartbeat), s.MockApp.lastToAdmin)
s.NextSenderMsgSeqNum(2)
}
func (s *InSessionTestSuite) TestTimeoutPeerTimeout() {
s.MockApp.On("ToAdmin").Return(nil)
s.session.Timeout(s.session, internal.PeerTimeout)
s.MockApp.AssertExpectations(s.T())
s.State(pendingTimeout{inSession{}})
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeTestRequest), s.MockApp.lastToAdmin)
s.NextSenderMsgSeqNum(2)
}
func (s *InSessionTestSuite) TestDisconnected() {
s.MockApp.On("OnLogout").Return(nil)
s.session.Disconnected(s.session)
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
}
func (s *InSessionTestSuite) TestStop() {
s.MockApp.On("ToAdmin")
s.session.Stop(s.session)
s.MockApp.AssertExpectations(s.T())
s.State(logoutState{})
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.MockApp.On("OnLogout")
s.session.Timeout(s.session, <-s.sessionEvent)
s.MockApp.AssertExpectations(s.T())
s.Stopped()
s.Disconnected()
}
func (s *InSessionTestSuite) TestFIXMsgInTargetTooHighEnableLastMsgSeqNumProcessed() {
s.session.EnableLastMsgSeqNumProcessed = true
s.MessageFactory.seqNum = 5
s.MockApp.On("ToAdmin")
msgSeqNumTooHigh := s.NewOrderSingle()
s.fixMsgIn(s.session, msgSeqNumTooHigh)
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeResendRequest), s.MockApp.lastToAdmin)
s.FieldEquals(tagLastMsgSeqNumProcessed, 0, s.MockApp.lastToAdmin.Header)
}
func (s *InSessionTestSuite) TestFIXMsgInTargetTooHigh() {
s.MessageFactory.seqNum = 5
s.MockApp.On("ToAdmin")
msgSeqNumTooHigh := s.NewOrderSingle()
s.fixMsgIn(s.session, msgSeqNumTooHigh)
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeResendRequest), s.MockApp.lastToAdmin)
s.FieldEquals(tagBeginSeqNo, 1, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagEndSeqNo, 0, s.MockApp.lastToAdmin.Body)
resendState, ok := s.session.State.(resendState)
s.True(ok)
s.NextTargetMsgSeqNum(1)
stashedMsg, ok := resendState.messageStash[6]
s.True(ok)
rawMsg := msgSeqNumTooHigh.build()
stashedRawMsg := stashedMsg.build()
s.Equal(string(rawMsg), string(stashedRawMsg))
}
func (s *InSessionTestSuite) TestFIXMsgInTargetTooHighResendRequestChunkSize() {
var tests = []struct {
chunkSize int
expectedEndSeqNo int
}{
{0, 0},
{10, 0},
{5, 0},
{2, 2},
{3, 3},
}
for _, test := range tests {
s.SetupTest()
s.MessageFactory.seqNum = 5
s.session.ResendRequestChunkSize = test.chunkSize
s.MockApp.On("ToAdmin")
msgSeqNumTooHigh := s.NewOrderSingle()
s.fixMsgIn(s.session, msgSeqNumTooHigh)
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeResendRequest), s.MockApp.lastToAdmin)
s.FieldEquals(tagBeginSeqNo, 1, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagEndSeqNo, test.expectedEndSeqNo, s.MockApp.lastToAdmin.Body)
resendState, ok := s.session.State.(resendState)
s.True(ok)
s.NextTargetMsgSeqNum(1)
stashedMsg, ok := resendState.messageStash[6]
s.True(ok)
rawMsg := msgSeqNumTooHigh.build()
stashedRawMsg := stashedMsg.build()
s.Equal(string(rawMsg), string(stashedRawMsg))
}
}
func (s *InSessionTestSuite) TestFIXMsgInResendRequestAllAdminExpectGapFill() {
s.MockApp.On("ToAdmin")
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 3)
s.NextSenderMsgSeqNum(4)
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, s.ResendRequest(1))
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeSequenceReset), s.MockApp.lastToAdmin)
s.FieldEquals(tagMsgSeqNum, 1, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagPossDupFlag, true, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagNewSeqNo, 4, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagGapFillFlag, true, s.MockApp.lastToAdmin.Body)
s.NextSenderMsgSeqNum(4)
s.State(inSession{})
}
func (s *InSessionTestSuite) TestFIXMsgInResendRequestAllAdminThenApp() {
s.MockApp.On("ToAdmin")
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.MockApp.On("ToApp").Return(nil)
s.Require().Nil(s.session.send(s.NewOrderSingle()))
s.LastToAppMessageSent()
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 2)
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 1)
s.NextSenderMsgSeqNum(4)
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.MockApp.On("ToApp").Return(nil)
s.fixMsgIn(s.session, s.ResendRequest(1))
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 3)
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 2)
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeSequenceReset), s.MockApp.lastToAdmin)
s.FieldEquals(tagMsgSeqNum, 1, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagPossDupFlag, true, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagNewSeqNo, 3, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagGapFillFlag, true, s.MockApp.lastToAdmin.Body)
s.LastToAppMessageSent()
s.MessageType("D", s.MockApp.lastToApp)
s.FieldEquals(tagMsgSeqNum, 3, s.MockApp.lastToApp.Header)
s.FieldEquals(tagPossDupFlag, true, s.MockApp.lastToApp.Header)
s.NextSenderMsgSeqNum(4)
s.State(inSession{})
}
func (s *InSessionTestSuite) TestFIXMsgInResendRequestNoMessagePersist() {
s.session.DisableMessagePersist = true
s.MockApp.On("ToApp").Return(nil)
s.Require().Nil(s.session.send(s.NewOrderSingle()))
s.LastToAppMessageSent()
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 1)
s.NextSenderMsgSeqNum(2)
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, s.ResendRequest(1))
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 1)
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 1)
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeSequenceReset), s.MockApp.lastToAdmin)
s.FieldEquals(tagMsgSeqNum, 1, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagPossDupFlag, true, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagNewSeqNo, 2, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagGapFillFlag, true, s.MockApp.lastToAdmin.Body)
s.NextSenderMsgSeqNum(2)
s.State(inSession{})
}
func (s *InSessionTestSuite) TestFIXMsgInResendRequestDoNotSendApp() {
s.MockApp.On("ToAdmin")
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.MockApp.On("ToApp").Return(nil)
s.Require().Nil(s.session.send(s.NewOrderSingle()))
s.LastToAppMessageSent()
s.session.Timeout(s.session, internal.NeedHeartbeat)
s.LastToAdminMessageSent()
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 2)
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 1)
s.NextSenderMsgSeqNum(4)
// NOTE: a cheat here, need to reset mock.
s.MockApp = MockApp{}
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToApp").Return(ErrDoNotSend)
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, s.ResendRequest(1))
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 1)
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 1)
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeSequenceReset), s.MockApp.lastToAdmin)
s.FieldEquals(tagMsgSeqNum, 1, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagPossDupFlag, true, s.MockApp.lastToAdmin.Header)
s.FieldEquals(tagNewSeqNo, 4, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagGapFillFlag, true, s.MockApp.lastToAdmin.Body)
s.NoMessageSent()
s.NextSenderMsgSeqNum(4)
s.State(inSession{})
}
func (s *InSessionTestSuite) TestFIXMsgInResendRequestBlocksSend() {
s.MockApp.On("ToApp").Return(nil)
s.Require().Nil(s.session.send(s.NewOrderSingle()))
s.LastToAppMessageSent()
s.MockApp.AssertNumberOfCalls(s.T(), "ToApp", 1)
s.NextSenderMsgSeqNum(2)
s.MockStore.On("IterateMessages", mock.Anything, mock.Anything, mock.AnythingOfType("func([]byte) error")).
Run(func(_ mock.Arguments) {
s.Require().Nil(s.session.send(s.NewOrderSingle()))
}).
Return(nil)
s.MockApp.On("FromAdmin").Return(nil)
s.fixMsgIn(s.session, s.ResendRequest(1))
s.NextSenderMsgSeqNum(2)
}
func (s *InSessionTestSuite) TestFIXMsgInTargetTooLow() {
s.IncrNextTargetMsgSeqNum()
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, s.NewOrderSingle())
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.FieldEquals(tagText, "MsgSeqNum too low, expecting 2 but received 1", s.MockApp.lastToAdmin.Body)
s.State(logoutState{})
}
func (s *InSessionTestSuite) TestFIXMsgInTargetTooLowPossDup() {
s.IncrNextTargetMsgSeqNum()
s.MockApp.On("ToAdmin")
nos := s.NewOrderSingle()
nos.Header.SetField(tagPossDupFlag, FIXBoolean(true))
s.fixMsgIn(s.session, nos)
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeReject), s.MockApp.lastToAdmin)
s.FieldEquals(tagText, "Required tag missing", s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagRefTagID, int(tagOrigSendingTime), s.MockApp.lastToAdmin.Body)
s.State(inSession{})
nos.Header.SetField(tagOrigSendingTime, FIXUTCTimestamp{Time: time.Now().Add(time.Duration(-1) * time.Minute)})
nos.Header.SetField(tagSendingTime, FIXUTCTimestamp{Time: time.Now()})
s.fixMsgIn(s.session, nos)
s.MockApp.AssertExpectations(s.T())
s.NoMessageSent()
s.State(inSession{})
s.NextTargetMsgSeqNum(2)
}