Files
qfixdpl/quickfix/logon_state_test.go
2026-03-09 15:35:32 -03:00

433 lines
11 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 (
"bytes"
"testing"
"time"
"github.com/stretchr/testify/suite"
"quantex.com/qfixdpl/quickfix/internal"
)
type LogonStateTestSuite struct {
SessionSuiteRig
}
func TestLogonStateTestSuite(t *testing.T) {
suite.Run(t, new(LogonStateTestSuite))
}
func (s *LogonStateTestSuite) SetupTest() {
s.Init()
s.session.stateMachine.State = logonState{}
}
func (s *LogonStateTestSuite) TestPreliminary() {
s.False(s.session.IsLoggedOn())
s.True(s.session.IsConnected())
s.True(s.session.IsSessionTime())
}
func (s *LogonStateTestSuite) TestTimeoutLogonTimeout() {
s.Timeout(s.session, internal.LogonTimeout)
s.State(latentState{})
}
func (s *LogonStateTestSuite) TestTimeoutLogonTimeoutInitiatedLogon() {
s.session.InitiateLogon = true
s.MockApp.On("OnLogout")
s.Timeout(s.session, internal.LogonTimeout)
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
}
func (s *LogonStateTestSuite) TestTimeoutNotLogonTimeout() {
tests := []internal.Event{internal.PeerTimeout, internal.NeedHeartbeat, internal.LogoutTimeout}
for _, test := range tests {
s.Timeout(s.session, test)
s.State(logonState{})
}
}
func (s *LogonStateTestSuite) TestDisconnected() {
s.session.Disconnected(s.session)
s.State(latentState{})
}
func (s *LogonStateTestSuite) TestFixMsgInNotLogon() {
s.fixMsgIn(s.session, s.NewOrderSingle())
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
s.NextTargetMsgSeqNum(1)
}
func (s *LogonStateTestSuite) TestFixMsgInLogon() {
s.IncrNextSenderMsgSeqNum()
s.MessageFactory.seqNum = 1
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.Zero(s.session.HeartBtInt)
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.Equal(32*time.Second, s.session.HeartBtInt) // Should be written from logon message.
s.False(s.session.HeartBtIntOverride)
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogon), s.MockApp.lastToAdmin)
s.FieldEquals(tagHeartBtInt, 32, s.MockApp.lastToAdmin.Body)
s.NextTargetMsgSeqNum(3)
s.NextSenderMsgSeqNum(3)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonHeartBtIntOverride() {
s.IncrNextSenderMsgSeqNum()
s.MessageFactory.seqNum = 1
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.session.HeartBtIntOverride = true
s.session.HeartBtInt = time.Second
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.Equal(time.Second, s.session.HeartBtInt) // Should not have changed.
s.True(s.session.HeartBtIntOverride)
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogon), s.MockApp.lastToAdmin)
s.FieldEquals(tagHeartBtInt, 1, s.MockApp.lastToAdmin.Body)
s.NextTargetMsgSeqNum(3)
s.NextSenderMsgSeqNum(3)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonEnableLastMsgSeqNumProcessed() {
s.session.EnableLastMsgSeqNumProcessed = true
s.MessageFactory.SetNextSeqNum(2)
s.IncrNextSenderMsgSeqNum()
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogon), s.MockApp.lastToAdmin)
s.FieldEquals(tagLastMsgSeqNumProcessed, 2, s.MockApp.lastToAdmin.Header)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonResetSeqNum() {
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
logon.Body.SetField(tagResetSeqNumFlag, FIXBoolean(true))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.Equal(32*time.Second, s.session.HeartBtInt)
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogon), s.MockApp.lastToAdmin)
s.FieldEquals(tagHeartBtInt, 32, s.MockApp.lastToAdmin.Body)
s.FieldEquals(tagResetSeqNumFlag, true, s.MockApp.lastToAdmin.Body)
s.NextTargetMsgSeqNum(2)
s.NextSenderMsgSeqNum(2)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonInitiateLogon() {
s.session.InitiateLogon = true
s.IncrNextSenderMsgSeqNum()
s.MessageFactory.seqNum = 1
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.NextTargetMsgSeqNum(3)
s.NextSenderMsgSeqNum(2)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonInitiateLogonExpectResetSeqNum() {
s.session.InitiateLogon = true
s.session.sentReset = true
s.Require().Nil(s.store.IncrNextSenderMsgSeqNum())
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
logon.Body.SetField(tagResetSeqNumFlag, FIXBoolean(true))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.NextTargetMsgSeqNum(2)
s.NextSenderMsgSeqNum(2)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonInitiateLogonRejectedSeqNumNotReset() {
s.session.InitiateLogon = true
s.session.sentReset = true
s.Require().Nil(s.store.IncrNextSenderMsgSeqNum())
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
logon.Body.SetField(tagResetSeqNumFlag, FIXBoolean(true))
s.MockApp.On("FromAdmin").Return(RejectLogon{"reject message"})
s.MockApp.On("OnLogout")
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
s.NextTargetMsgSeqNum(2)
s.NextSenderMsgSeqNum(3)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonInitiateLogonUnExpectedResetSeqNum() {
s.session.InitiateLogon = true
s.session.sentReset = false
s.IncrNextTargetMsgSeqNum()
s.IncrNextSenderMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
logon.Body.SetField(tagResetSeqNumFlag, FIXBoolean(true))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.NextTargetMsgSeqNum(2)
s.NextSenderMsgSeqNum(1)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonRefreshOnLogon() {
var tests = []bool{true, false}
for _, doRefresh := range tests {
s.SetupTest()
s.session.RefreshOnLogon = doRefresh
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
if doRefresh {
s.MockStore.On("Refresh").Return(nil)
}
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.MockStore.AssertExpectations(s.T())
}
}
func (s *LogonStateTestSuite) TestStop() {
var tests = []bool{true, false}
for _, doInitiateLogon := range tests {
s.SetupTest()
s.session.InitiateLogon = doInitiateLogon
if doInitiateLogon {
s.MockApp.On("OnLogout")
}
s.session.Stop(s.session)
s.MockApp.AssertExpectations(s.T())
s.Disconnected()
s.Stopped()
}
}
func (s *LogonStateTestSuite) TestFixMsgInLogonRejectLogon() {
s.IncrNextSenderMsgSeqNum()
s.MessageFactory.seqNum = 1
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
s.MockApp.On("FromAdmin").Return(RejectLogon{"reject message"})
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(latentState{})
s.LastToAdminMessageSent()
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.FieldEquals(tagText, "reject message", s.MockApp.lastToAdmin.Body)
s.NextTargetMsgSeqNum(3)
s.NextSenderMsgSeqNum(3)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonSeqNumTooHigh() {
s.MessageFactory.SetNextSeqNum(6)
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.State(resendState{})
s.NextTargetMsgSeqNum(1)
// Session should send logon, and then queues resend request for send.
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 2)
msgBytesSent, ok := s.Receiver.LastMessage()
s.Require().True(ok)
sentMessage := NewMessage()
err := ParseMessage(sentMessage, bytes.NewBuffer(msgBytesSent))
s.Require().Nil(err)
s.MessageType(string(msgTypeLogon), sentMessage)
s.session.sendQueued(true)
s.MessageType(string(msgTypeResendRequest), s.MockApp.lastToAdmin)
s.FieldEquals(tagBeginSeqNo, 1, s.MockApp.lastToAdmin.Body)
s.MockApp.On("FromAdmin").Return(nil)
s.MessageFactory.SetNextSeqNum(1)
s.fixMsgIn(s.session, s.SequenceReset(3))
s.State(resendState{})
s.NextTargetMsgSeqNum(3)
s.MessageFactory.SetNextSeqNum(3)
s.MockApp.On("FromAdmin").Return(nil)
s.fixMsgIn(s.session, s.SequenceReset(7))
s.State(inSession{})
s.NextTargetMsgSeqNum(7)
}
func (s *LogonStateTestSuite) TestFixMsgInLogonSeqNumTooLow() {
s.IncrNextSenderMsgSeqNum()
s.IncrNextTargetMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagHeartBtInt, FIXInt(32))
logon.Header.SetInt(tagMsgSeqNum, 1)
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("ToAdmin")
s.NextTargetMsgSeqNum(2)
s.fixMsgIn(s.session, logon)
s.State(latentState{})
s.NextTargetMsgSeqNum(2)
s.MockApp.AssertNumberOfCalls(s.T(), "ToAdmin", 1)
msgBytesSent, ok := s.Receiver.LastMessage()
s.Require().True(ok)
sentMessage := NewMessage()
err := ParseMessage(sentMessage, bytes.NewBuffer(msgBytesSent))
s.Require().Nil(err)
s.MessageType(string(msgTypeLogout), sentMessage)
s.session.sendQueued(true)
s.MessageType(string(msgTypeLogout), s.MockApp.lastToAdmin)
s.FieldEquals(tagText, "MsgSeqNum too low, expecting 2 but received 1", s.MockApp.lastToAdmin.Body)
}
func (s *LogonStateTestSuite) TestStayLoggedInOnReset() {
s.IncrNextTargetMsgSeqNum()
s.IncrNextSenderMsgSeqNum()
logon := s.Logon()
logon.Body.SetField(tagResetSeqNumFlag, FIXBoolean(true))
s.MockApp.On("FromAdmin").Return(nil)
s.MockApp.On("OnLogon")
s.MockApp.On("ToAdmin")
s.fixMsgIn(s.session, logon)
s.MockApp.AssertExpectations(s.T())
s.State(inSession{})
s.IncrNextTargetMsgSeqNum()
s.IncrNextSenderMsgSeqNum()
s.NextTargetMsgSeqNum(3)
s.NextSenderMsgSeqNum(3)
s.fixMsgIn(s.session, logon)
s.True(s.session.IsConnected())
s.True(s.session.IsLoggedOn())
s.NextTargetMsgSeqNum(2)
s.NextSenderMsgSeqNum(2)
}