adding quickfix
This commit is contained in:
432
quickfix/logon_state_test.go
Normal file
432
quickfix/logon_state_test.go
Normal file
@ -0,0 +1,432 @@
|
||||
// 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)
|
||||
}
|
||||
Reference in New Issue
Block a user