QuickFIX是一款C++實現的開源FIX引擎,同時提供Python、Ruby語言實現。
QuickFIX官網:
http://www.quickfixengine.org
Github地址:
https://github.com/quickfix/quickfixios
編譯選項配置:
configure
編譯:
make
檢查:
make check
安裝:
sudo make installgit
FIX應用有initiator和acceptor兩種模式。
initiator是TCP鏈接的發起方,acceptor是TCP鏈接監聽方。標準FIX應用(如CTS FIX網關)能夠同時支持initiator和acceptor兩種模式,既能夠發起鏈接,也能夠接受鏈接請求。
開發FIX應用時,須要先肯定FIX應用模式,而後選擇對應的QuickFIX類對象。
initiator模式的FIX應用可使用SocketInitiator、SSLSocketInitiator、ThreadedSocketInitiator、ThreadedSSLSocketInitiator建立FIX Initiator應用。
acceptor模式的FIX應用可使用SocketAcceptor、SSLSocketAcceptor、ThreadedSocketAcceptor、ThreadedSSLSocketAcceptor建立FIX Acceptor應用。github
FIX應用業務是經過在initiator與acceptor間創建一個FIX Session並交換消息來進行的。FIX業務消息有一系列預約義類型與格式,目前標準的消息類型有五、60個,函蓋了交易、行情、結算等投資管理的各個環節。FIX應用開發基本上是對FIX消息的編程。
Fix Session能夠由一個或多個Fix鏈接組成,即Fix Session能夠屢次登錄。Fix Session一般使用幾個重要標識區分,如Fix版本號字符串(如:FIX.4.2),發送者ID,接收者ID,Fix Session每發送一條消息時,消息頭的BeginString、SenderCompID、TargetCompID都會被賦值。當消息經過FIX鏈接到達對等方時,對等方能夠根據消息頭中的的標識找到本地端的Fix Session去處理消息。
FIX通訊雙方包括Initiator和Acceptor,會各自維護2個遞增的序列號(發送消息序列號–每發送一個消息加1,接收消息序列號–每收到一個消息加1)。
(1)通訊首先由Initiator開始發起TCP鏈接請求, Acceptor接收網絡鏈接請求,創建TCP鏈接。
(2)Initiator發起登陸消息請求。
(3)Acceptor收到登陸請求後,通過一系列消息檢查,合格後,返回登陸確認。
(4)Initiator收到登陸確認後,通過一系列消息檢查,合格後,雙方FIX Session鏈接成功。
(5)Initiator和Acceptor交換消息。
(6)Initiator和Acceptor任何一方發出退出消息。編程
QuickFix中initiator或acceptor會維護多個Fix Session。QuickFix中使用BeginString(Fix版本號)、SenderCompID、TargetCompID的組合標識一個Session,Session標識用於區分其它不一樣的Session。
Session配置文件包含[DEFAULT]和[SESSION]兩種分節,[SESSION]分節表示QuickFix中定義一個Session,[DEFAULT]表示全部Session默認使用的配置項,若是不提供QuickFix所需的配置,QuickFix會拋出ConfigError異常,表示配置缺失或格式不正確。
若是[DEFAULT]和[SESSION]分區中都包含相同的配置項,則[SESSION]分區的配置項會覆蓋[DEFAULT]分區相應的配置項。
QuickFix配置文件sessions.ini以下:網絡
[DEFAULT] ConnectionType=initiator ReconnectInterval=60 FileLogPath=log FileStorePath=store StartTime=00:00:00 EndTime=23:59:59 HeartBtInt=30 ResetOnDisconnect=Y ResetOnLogout=Y ResetOnLogon=Y [SESSION] BeginString=FIX.4.2 SenderCompID=CLIENT TargetCompID=SERVER SocketConnectPort=6666 SocketConnectHost=127.0.0.1 DataDictionary=FIX42.xml
FIX應用須要實現FIX::Application接口:session
class Application { public: virtual ~Application() {}; /// Notification of a session begin created virtual void onCreate( const SessionID& ) = 0; /// Notification of a session successfully logging on virtual void onLogon( const SessionID& ) = 0; /// Notification of a session logging off or disconnecting virtual void onLogout( const SessionID& ) = 0; /// Notification of admin message being sent to target virtual void toAdmin( Message&, const SessionID& ) = 0; /// Notification of app message being sent to target virtual void toApp( Message&, const SessionID& ) EXCEPT ( DoNotSend ) = 0; /// Notification of admin message being received from target virtual void fromAdmin( const Message&, const SessionID& ) EXCEPT ( FieldNotFound, IncorrectDataFormat, IncorrectTagValue, RejectLogon ) = 0; /// Notification of app message being received from target virtual void fromApp( const Message&, const SessionID& ) EXCEPT ( FieldNotFound, IncorrectDataFormat, IncorrectTagValue, UnsupportedMessageType ) = 0; };
onCreate:當一個Fix Session創建時調用。
onLogon:當一個Fix Session登陸成功時調用。
onLogout:當一個Fix Session退出時調用。
fromAdmin:當收到一個消息,通過一系列檢查,合格後,屬於Admin類型時調用。
fromApp:當收到一個消息,通過一系列檢查,合格後,不屬於Admin 類型時候調用。
toAdmin:當發送一個admin類型消息調用。
toApp:當發送一個非admin(業務類型)消息調用。
若是須要使用QuickFIX開發FIX應用,則須要實現FIX::Application接口,並重載不一樣FIX協議版本的MessageCracker::OnMessage接口,如FIX42::MessageCracker。
FIX業務都是異步方式處理的,而業務處理的基本對象是消息。OnMessage是消息接收回調函數,有多個重載版本,開發者只須要使重載一個FIX版本便可。
對於支持交易業務的FIX Initiator應用,一般要重寫4個基本消息,即OnMessage(NewOrderSingle)、OnMessage(CancelRequest)、 OnMessage(ExecutionReport)、 OnMessage(CancelReject),分別用於作委託、撤單、執行回報(包括對委託的拒絕)和對撤單的拒絕等4項業務。app
#include "quickfix/Application.h" #include "quickfix/MessageCracker.h" class FIXApplication: public FIX::Application, public FIX::MessageCracker { public: /************************************************** * reimplementation from Application * ***********************************************/ /// Notification of a session begin created virtual void onCreate( const SessionID& ) { } /// Notification of a session successfully logging on virtual void onLogon( const SessionID& ) { } /// Notification of a session logging off or disconnecting virtual void onLogout( const SessionID& ) { } /// Notification of admin message being sent to target virtual void toAdmin( Message&, const SessionID& ) { } /// Notification of app message being sent to target void fromApp( const FIX::Message& message, const FIX::SessionID& sessionID ) throw( FIX::FieldNotFound&, FIX::IncorrectDataFormat&, FIX::IncorrectTagValue&, FIX::UnsupportedMessageType& ) { crack(message, sessionID); } /// Notification of admin message being received from target virtual void fromAdmin( const Message&, const SessionID& ) throw ( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::RejectLogon ) { } /// Notification of app message being received from target virtual void fromApp( const Message&, const SessionID& ) throw ( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::UnsupportedMessageType ) { } /************************************************** * reimplementation from FIX42::MessageCracker * ***********************************************/ virtual void onMessage( const FIX42::NewOrderSingle& message, const FIX::SessionID& ) { } virtual void onMessage( const FIX42::OrderCancelRequest& message, const FIX::SessionID& ) { } virtual void onMessage( ExecutionReport&, const FIX::SessionID& ) { } virtual void onMessage( OrderCancelReject&, const FIX::SessionID& ) { } };
FIX應用配置使用FIX::SessionSettings讀取FIX Session配置文件並傳遞給QuickFIX框架。一個FIX應用能夠管理多個FIX Session,每一個Session能夠採用相同的FIX協議版本,也能夠採用不一樣的版本。即便採用的是相同的FIX協議版本,不一樣FIX Session間也能夠有FIX協議細節的差別,經過綁定FIX Session與FIX協議字典的方式來實現,即在Session配置文件中[Session]配置項中使用DataDictionary選項指定相應FIX字典文件來實現。FIX::SessionSettings settings("sessionConfig.ini");
框架
QuickFIX提供了存儲消息到文件的類FIX::FileLogFactory。消息文件存儲的路徑在Session配置中指定。FIX::FileStoreFactory storeFactory(settings);
異步
QuickFIX提供了給Fix Session持久化類型(如文件存儲、數據存儲,存儲內容包括狀態、建立時間、消息及其本身維護的發送序列號和接收序列號等)。
若是開發者要自定義持久化方式,能夠本身定義MessageStoreFactory實現,而且自定義一種MessageStore。ide
QuickFIX提供了存儲全部日誌事件到文件的類FIX::FileLogFactory。
QuickFIX中提供了顯示全部消息事件到標準輸出的類ScreenLogFactory。FIX::ScreenLogFactory logFactory(settings);
QuickFIX中定義了不一樣FIX協議版本消息的基類FIX::Message,用於定義FIX消息的通用結構,不一樣的FIX消息版本的消息定義在不一樣的FIX命名空間內定義,如FIX42::Message。FIX::MessageCracker則繼承了全部不一樣FIX協議版本的MessageCracker類,接收消息後生成具體FIX協議Message對象實現對消息進行處理。
不管是FIX Initiator應用仍是FIX Acceptor應用,在Fix Session初始化時,即在FIX::SessionFactory建立Fix Session後都會檢查Fix Session時間範圍。若是Fix Session啓動不是當天有效的Session,則會重置Fix Session的發送序列號和接收序列號。(FIX規定一個Fix Session通常不超過24小時)。
FIX::Session部分接口定義以下:
class Session { public: static bool sendToTarget( Message& message, const std::string& qualifier = "" ) EXCEPT ( SessionNotFound ); static bool sendToTarget( Message& message, const SessionID& sessionID ) EXCEPT ( SessionNotFound ); static bool sendToTarget( Message&, const SenderCompID& senderCompID, const TargetCompID& targetCompID, const std::string& qualifier = "" ) EXCEPT ( SessionNotFound ); static bool sendToTarget( Message& message, const std::string& senderCompID, const std::string& targetCompID, const std::string& qualifier = "" ) EXCEPT ( SessionNotFound ); bool send( Message& ); void next(); void next( const UtcTimeStamp& timeStamp ); void next( const std::string&, const UtcTimeStamp& timeStamp, bool queued = false ); void next( const Message&, const UtcTimeStamp& timeStamp, bool queued = false ); };
next()方法是定時運行的一個方法,主要用於檢測是否須要發心跳消息,是否須要發TEST消息,是否須要斷開鏈接,是否須要產生LOGON(Initiator)。
next( const Message&)方法用於處理Session收到的FIX消息。
FIX::Acceptor用於從Session配置文件讀取信息建立和管理本Acceptor支持的FIX Session,具體FIX Session的TCP鏈接管理、數據讀寫由具體的SocketAcceptor、SSLSocketAcceptor、ThreadedSocketAcceptor、ThreadedSSLSocketAcceptor實現。FIX::Acceptor* acceptor = new FIX::SocketAcceptor(application, storeFactory, settings, logFactory);
FIX::Initiator用於從Session配置文件讀取信息建立和管理本Initiator支持的FIX Session,具體FIX Session的TCP鏈接管理、數據讀寫由具體的SocketInitiator、SSLSocketInitiator、ThreadedSocketInitiator、ThreadedSSLSocketInitiator實現。FIX::Initiator * initiator = new FIX::SocketInitiator(application, storeFactory, settings, logFactory);
FIX Acceptor應用一般用於FIX網關,部署在賣方側。
(1)建立FIX Session配置對象FIX::SessionSettings settings(sessionFile);
(2)建立FIX應用:Application application;
建立日誌工廠:LogFactory logFactory(settings);
建立消息存儲文件工廠:FIX::FileStoreFactory storeFactory(settings);
建立Acceptor服務端:FIX::Acceptor* acceptor = new SocketAcceptor(application, storeFactory, settings, logFactory);
啓動SocketAcceptor:acceptor->start();
Executor示例演示了接收訂單請求並返回成交回執的FIX Acceptor應用。
Application.h文件:
#ifndef EXECUTOR_APPLICATION_H #define EXECUTOR_APPLICATION_H #include "quickfix/Application.h" #include "quickfix/MessageCracker.h" #include "quickfix/Values.h" #include "quickfix/Utility.h" #include "quickfix/Mutex.h" #include "quickfix/fix40/NewOrderSingle.h" #include "quickfix/fix41/NewOrderSingle.h" #include "quickfix/fix42/NewOrderSingle.h" #include "quickfix/fix43/NewOrderSingle.h" #include "quickfix/fix44/NewOrderSingle.h" #include "quickfix/fix50/NewOrderSingle.h" class Application : public FIX::Application, public FIX::MessageCracker { public: Application() : m_orderID(0), m_execID(0) {} // Application overloads void onCreate( const FIX::SessionID& ); void onLogon( const FIX::SessionID& sessionID ); void onLogout( const FIX::SessionID& sessionID ); void toAdmin( FIX::Message&, const FIX::SessionID& ); void toApp( FIX::Message&, const FIX::SessionID& ) EXCEPT( FIX::DoNotSend ); void fromAdmin( const FIX::Message&, const FIX::SessionID& ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::RejectLogon ); void fromApp( const FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::UnsupportedMessageType ); // MessageCracker overloads void onMessage( const FIX40::NewOrderSingle&, const FIX::SessionID& ); void onMessage( const FIX41::NewOrderSingle&, const FIX::SessionID& ); void onMessage( const FIX42::NewOrderSingle&, const FIX::SessionID& ); void onMessage( const FIX43::NewOrderSingle&, const FIX::SessionID& ); void onMessage( const FIX44::NewOrderSingle&, const FIX::SessionID& ); void onMessage( const FIX50::NewOrderSingle&, const FIX::SessionID& ); std::string genOrderID() { std::stringstream stream; stream << ++m_orderID; return stream.str(); } std::string genExecID() { std::stringstream stream; stream << ++m_execID; return stream.str(); } private: int m_orderID, m_execID; }; #endif
Application.cpp文件:
#include "config.h" #include "Application.h" #include "quickfix/Session.h" #include "quickfix/fix40/ExecutionReport.h" #include "quickfix/fix41/ExecutionReport.h" #include "quickfix/fix42/ExecutionReport.h" #include "quickfix/fix43/ExecutionReport.h" #include "quickfix/fix44/ExecutionReport.h" #include "quickfix/fix50/ExecutionReport.h" void Application::onCreate( const FIX::SessionID& sessionID ) {} void Application::onLogon( const FIX::SessionID& sessionID ) {} void Application::onLogout( const FIX::SessionID& sessionID ) {} void Application::toAdmin( FIX::Message& message, const FIX::SessionID& sessionID ) {} void Application::toApp( FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::DoNotSend ) {} void Application::fromAdmin( const FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::RejectLogon ) {} void Application::fromApp( const FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::UnsupportedMessageType ) { crack( message, sessionID ); } void Application::onMessage( const FIX40::NewOrderSingle& message, const FIX::SessionID& sessionID ) { FIX::Symbol symbol; FIX::Side side; FIX::OrdType ordType; FIX::OrderQty orderQty; FIX::Price price; FIX::ClOrdID clOrdID; FIX::Account account; message.get( ordType ); if ( ordType != FIX::OrdType_LIMIT ) throw FIX::IncorrectTagValue( ordType.getField() ); message.get( symbol ); message.get( side ); message.get( orderQty ); message.get( price ); message.get( clOrdID ); FIX40::ExecutionReport executionReport = FIX40::ExecutionReport ( FIX::OrderID( genOrderID() ), FIX::ExecID( genExecID() ), FIX::ExecTransType( FIX::ExecTransType_NEW ), FIX::OrdStatus( FIX::OrdStatus_FILLED ), symbol, side, orderQty, FIX::LastShares( orderQty ), FIX::LastPx( price ), FIX::CumQty( orderQty ), FIX::AvgPx( price ) ); executionReport.set( clOrdID ); if( message.isSet(account) ) executionReport.setField( message.get(account) ); try { FIX::Session::sendToTarget( executionReport, sessionID ); } catch ( FIX::SessionNotFound& ) {} } void Application::onMessage( const FIX41::NewOrderSingle& message, const FIX::SessionID& sessionID ) { FIX::Symbol symbol; FIX::Side side; FIX::OrdType ordType; FIX::OrderQty orderQty; FIX::Price price; FIX::ClOrdID clOrdID; FIX::Account account; message.get( ordType ); if ( ordType != FIX::OrdType_LIMIT ) throw FIX::IncorrectTagValue( ordType.getField() ); message.get( symbol ); message.get( side ); message.get( orderQty ); message.get( price ); message.get( clOrdID ); FIX41::ExecutionReport executionReport = FIX41::ExecutionReport ( FIX::OrderID( genOrderID() ), FIX::ExecID( genExecID() ), FIX::ExecTransType( FIX::ExecTransType_NEW ), FIX::ExecType( FIX::ExecType_FILL ), FIX::OrdStatus( FIX::OrdStatus_FILLED ), symbol, side, orderQty, FIX::LastShares( orderQty ), FIX::LastPx( price ), FIX::LeavesQty( 0 ), FIX::CumQty( orderQty ), FIX::AvgPx( price ) ); executionReport.set( clOrdID ); if( message.isSet(account) ) executionReport.setField( message.get(account) ); try { FIX::Session::sendToTarget( executionReport, sessionID ); } catch ( FIX::SessionNotFound& ) {} } void Application::onMessage( const FIX42::NewOrderSingle& message, const FIX::SessionID& sessionID ) { FIX::Symbol symbol; FIX::Side side; FIX::OrdType ordType; FIX::OrderQty orderQty; FIX::Price price; FIX::ClOrdID clOrdID; FIX::Account account; message.get( ordType ); if ( ordType != FIX::OrdType_LIMIT ) throw FIX::IncorrectTagValue( ordType.getField() ); message.get( symbol ); message.get( side ); message.get( orderQty ); message.get( price ); message.get( clOrdID ); FIX42::ExecutionReport executionReport = FIX42::ExecutionReport ( FIX::OrderID( genOrderID() ), FIX::ExecID( genExecID() ), FIX::ExecTransType( FIX::ExecTransType_NEW ), FIX::ExecType( FIX::ExecType_FILL ), FIX::OrdStatus( FIX::OrdStatus_FILLED ), symbol, side, FIX::LeavesQty( 0 ), FIX::CumQty( orderQty ), FIX::AvgPx( price ) ); executionReport.set( clOrdID ); executionReport.set( orderQty ); executionReport.set( FIX::LastShares( orderQty ) ); executionReport.set( FIX::LastPx( price ) ); if( message.isSet(account) ) executionReport.setField( message.get(account) ); try { FIX::Session::sendToTarget( executionReport, sessionID ); } catch ( FIX::SessionNotFound& ) {} } void Application::onMessage( const FIX43::NewOrderSingle& message, const FIX::SessionID& sessionID ) { FIX::Symbol symbol; FIX::Side side; FIX::OrdType ordType; FIX::OrderQty orderQty; FIX::Price price; FIX::ClOrdID clOrdID; FIX::Account account; message.get( ordType ); if ( ordType != FIX::OrdType_LIMIT ) throw FIX::IncorrectTagValue( ordType.getField() ); message.get( symbol ); message.get( side ); message.get( orderQty ); message.get( price ); message.get( clOrdID ); FIX43::ExecutionReport executionReport = FIX43::ExecutionReport ( FIX::OrderID( genOrderID() ), FIX::ExecID( genExecID() ), FIX::ExecType( FIX::ExecType_FILL ), FIX::OrdStatus( FIX::OrdStatus_FILLED ), side, FIX::LeavesQty( 0 ), FIX::CumQty( orderQty ), FIX::AvgPx( price ) ); executionReport.set( clOrdID ); executionReport.set( symbol ); executionReport.set( orderQty ); executionReport.set( FIX::LastQty( orderQty ) ); executionReport.set( FIX::LastPx( price ) ); if( message.isSet(account) ) executionReport.setField( message.get(account) ); try { FIX::Session::sendToTarget( executionReport, sessionID ); } catch ( FIX::SessionNotFound& ) {} } void Application::onMessage( const FIX44::NewOrderSingle& message, const FIX::SessionID& sessionID ) { FIX::Symbol symbol; FIX::Side side; FIX::OrdType ordType; FIX::OrderQty orderQty; FIX::Price price; FIX::ClOrdID clOrdID; FIX::Account account; message.get( ordType ); if ( ordType != FIX::OrdType_LIMIT ) throw FIX::IncorrectTagValue( ordType.getField() ); message.get( symbol ); message.get( side ); message.get( orderQty ); message.get( price ); message.get( clOrdID ); FIX44::ExecutionReport executionReport = FIX44::ExecutionReport ( FIX::OrderID( genOrderID() ), FIX::ExecID( genExecID() ), FIX::ExecType( FIX::ExecType_TRADE ), FIX::OrdStatus( FIX::OrdStatus_FILLED ), side, FIX::LeavesQty( 0 ), FIX::CumQty( orderQty ), FIX::AvgPx( price ) ); executionReport.set( clOrdID ); executionReport.set( symbol ); executionReport.set( orderQty ); executionReport.set( FIX::LastQty( orderQty ) ); executionReport.set( FIX::LastPx( price ) ); if( message.isSet(account) ) executionReport.setField( message.get(account) ); try { FIX::Session::sendToTarget( executionReport, sessionID ); } catch ( FIX::SessionNotFound& ) {} } void Application::onMessage( const FIX50::NewOrderSingle& message, const FIX::SessionID& sessionID ) { FIX::Symbol symbol; FIX::Side side; FIX::OrdType ordType; FIX::OrderQty orderQty; FIX::Price price; FIX::ClOrdID clOrdID; FIX::Account account; message.get( ordType ); if ( ordType != FIX::OrdType_LIMIT ) throw FIX::IncorrectTagValue( ordType.getField() ); message.get( symbol ); message.get( side ); message.get( orderQty ); message.get( price ); message.get( clOrdID ); FIX50::ExecutionReport executionReport = FIX50::ExecutionReport ( FIX::OrderID( genOrderID() ), FIX::ExecID( genExecID() ), FIX::ExecType( FIX::ExecType_TRADE ), FIX::OrdStatus( FIX::OrdStatus_FILLED ), side, FIX::LeavesQty( 0 ), FIX::CumQty( orderQty ) ); executionReport.set( clOrdID ); executionReport.set( symbol ); executionReport.set( orderQty ); executionReport.set( FIX::LastQty( orderQty ) ); executionReport.set( FIX::LastPx( price ) ); executionReport.set( FIX::AvgPx( price ) ); if( message.isSet(account) ) executionReport.setField( message.get(account) ); try { FIX::Session::sendToTarget( executionReport, sessionID ); } catch ( FIX::SessionNotFound& ) {} }
main.cpp文件:
#include "config.h" #include "quickfix/FileStore.h" #include "quickfix/SocketAcceptor.h" #include "quickfix/Log.h" #include "quickfix/SessionSettings.h" #include "Application.h" #include <string> #include <iostream> #include <fstream> void wait() { std::cout << "Type Ctrl-C to quit" << std::endl; while(true) { FIX::process_sleep(1); } } int main( int argc, char** argv ) { if ( argc < 2 ) { std::cout << "usage: " << argv[ 0 ] << " FILE." << std::endl; return 0; } std::string file = argv[ 1 ]; FIX::Acceptor * acceptor = 0; try { FIX::SessionSettings settings( file ); Application application; FIX::FileStoreFactory storeFactory( settings ); FIX::ScreenLogFactory logFactory( settings ); acceptor = new FIX::SocketAcceptor ( application, storeFactory, settings, logFactory ); acceptor->start(); wait(); acceptor->stop(); delete acceptor; return 0; } catch ( std::exception & e ) { std::cout << e.what() << std::endl; delete acceptor; return 1; } }
FIX Initiator應用一般用於交易客戶端,部署在買方側。
(1)建立FIX Session配置對象FIX::SessionSettings settings(sessionFile);
(2)建立FIX應用:Application application;
建立日誌工廠:LogFactory logFactory(settings);
建立消息存儲文件工廠:FIX::FileStoreFactory storeFactory(settings);
建立Acceptor服務端:FIX::Initiator* client= new SocketInitiator(application, storeFactory,settings, logFactory);
啓動SocketInitiator:client->start();
tradeclient交易客戶端
Application.h文件:
#ifndef TRADECLIENT_APPLICATION_H #define TRADECLIENT_APPLICATION_H #include "quickfix/Application.h" #include "quickfix/MessageCracker.h" #include "quickfix/Values.h" #include "quickfix/Mutex.h" #include "quickfix/fix40/NewOrderSingle.h" #include "quickfix/fix40/ExecutionReport.h" #include "quickfix/fix40/OrderCancelRequest.h" #include "quickfix/fix40/OrderCancelReject.h" #include "quickfix/fix40/OrderCancelReplaceRequest.h" #include "quickfix/fix41/NewOrderSingle.h" #include "quickfix/fix41/ExecutionReport.h" #include "quickfix/fix41/OrderCancelRequest.h" #include "quickfix/fix41/OrderCancelReject.h" #include "quickfix/fix41/OrderCancelReplaceRequest.h" #include "quickfix/fix42/NewOrderSingle.h" #include "quickfix/fix42/ExecutionReport.h" #include "quickfix/fix42/OrderCancelRequest.h" #include "quickfix/fix42/OrderCancelReject.h" #include "quickfix/fix42/OrderCancelReplaceRequest.h" #include "quickfix/fix43/NewOrderSingle.h" #include "quickfix/fix43/ExecutionReport.h" #include "quickfix/fix43/OrderCancelRequest.h" #include "quickfix/fix43/OrderCancelReject.h" #include "quickfix/fix43/OrderCancelReplaceRequest.h" #include "quickfix/fix43/MarketDataRequest.h" #include "quickfix/fix44/NewOrderSingle.h" #include "quickfix/fix44/ExecutionReport.h" #include "quickfix/fix44/OrderCancelRequest.h" #include "quickfix/fix44/OrderCancelReject.h" #include "quickfix/fix44/OrderCancelReplaceRequest.h" #include "quickfix/fix44/MarketDataRequest.h" #include "quickfix/fix50/NewOrderSingle.h" #include "quickfix/fix50/ExecutionReport.h" #include "quickfix/fix50/OrderCancelRequest.h" #include "quickfix/fix50/OrderCancelReject.h" #include "quickfix/fix50/OrderCancelReplaceRequest.h" #include "quickfix/fix50/MarketDataRequest.h" #include <queue> class Application : public FIX::Application, public FIX::MessageCracker { public: void run(); private: void onCreate( const FIX::SessionID& ) {} void onLogon( const FIX::SessionID& sessionID ); void onLogout( const FIX::SessionID& sessionID ); void toAdmin( FIX::Message&, const FIX::SessionID& ) {} void toApp( FIX::Message&, const FIX::SessionID& ) EXCEPT( FIX::DoNotSend ); void fromAdmin( const FIX::Message&, const FIX::SessionID& ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::RejectLogon ) {} void fromApp( const FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::UnsupportedMessageType ); void onMessage( const FIX42::ExecutionReport&, const FIX::SessionID& ); void onMessage( const FIX42::OrderCancelReject&, const FIX::SessionID& ); void queryEnterOrder(); void queryCancelOrder(); void queryReplaceOrder(); FIX42::NewOrderSingle queryNewOrderSingle42(); FIX42::OrderCancelRequest queryOrderCancelRequest42(); FIX42::OrderCancelReplaceRequest queryCancelReplaceRequest42(); void queryHeader( FIX::Header& header ); char queryAction(); int queryVersion(); bool queryConfirm( const std::string& query ); FIX::SenderCompID querySenderCompID(); FIX::TargetCompID queryTargetCompID(); FIX::TargetSubID queryTargetSubID(); FIX::ClOrdID queryClOrdID(); FIX::OrigClOrdID queryOrigClOrdID(); FIX::Symbol querySymbol(); FIX::Side querySide(); FIX::OrderQty queryOrderQty(); FIX::OrdType queryOrdType(); FIX::Price queryPrice(); FIX::StopPx queryStopPx(); FIX::TimeInForce queryTimeInForce(); }; #endif
Application.cpp文件:
#include "config.h" #include "Application.h" #include "quickfix/Session.h" #include <iostream> void Application::onLogon( const FIX::SessionID& sessionID ) { std::cout << std::endl << "Logon - " << sessionID << std::endl; } void Application::onLogout( const FIX::SessionID& sessionID ) { std::cout << std::endl << "Logout - " << sessionID << std::endl; } void Application::fromApp( const FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::FieldNotFound, FIX::IncorrectDataFormat, FIX::IncorrectTagValue, FIX::UnsupportedMessageType ) { crack( message, sessionID ); std::cout << std::endl << "IN: " << message << std::endl; } void Application::toApp( FIX::Message& message, const FIX::SessionID& sessionID ) EXCEPT( FIX::DoNotSend ) { try { FIX::PossDupFlag possDupFlag; message.getHeader().getField( possDupFlag ); if ( possDupFlag ) throw FIX::DoNotSend(); } catch ( FIX::FieldNotFound& ) {} std::cout << std::endl << "OUT: " << message << std::endl; } void Application::onMessage ( const FIX42::ExecutionReport&, const FIX::SessionID& ) {} void Application::onMessage ( const FIX42::OrderCancelReject&, const FIX::SessionID& ) {} void Application::run() { while ( true ) { try { char action = queryAction(); if ( action == '1' ) queryEnterOrder(); else if ( action == '2' ) queryCancelOrder(); else if ( action == '3' ) queryReplaceOrder(); else if ( action == '5' ) break; } catch ( std::exception & e ) { std::cout << "Message Not Sent: " << e.what(); } } } void Application::queryEnterOrder() { int version = queryVersion(); std::cout << "\nNewOrderSingle\n"; FIX::Message order; switch ( version ) { case 42: order = queryNewOrderSingle42(); break; default: std::cerr << "No test for version " << version << std::endl; break; } if ( queryConfirm( "Send order" ) ) FIX::Session::sendToTarget( order ); } void Application::queryCancelOrder() { int version = queryVersion(); std::cout << "\nOrderCancelRequest\n"; FIX::Message cancel; switch ( version ) { case 42: cancel = queryOrderCancelRequest42(); break; default: std::cerr << "No test for version " << version << std::endl; break; } if ( queryConfirm( "Send cancel" ) ) FIX::Session::sendToTarget( cancel ); } void Application::queryReplaceOrder() { int version = queryVersion(); std::cout << "\nCancelReplaceRequest\n"; FIX::Message replace; switch ( version ) { case 42: replace = queryCancelReplaceRequest42(); break; default: std::cerr << "No test for version " << version << std::endl; break; } if ( queryConfirm( "Send replace" ) ) FIX::Session::sendToTarget( replace ); } FIX42::NewOrderSingle Application::queryNewOrderSingle42() { FIX::OrdType ordType; FIX42::NewOrderSingle newOrderSingle( queryClOrdID(), FIX::HandlInst( '1' ), querySymbol(), querySide(), FIX::TransactTime(), ordType = queryOrdType() ); newOrderSingle.set( queryOrderQty() ); newOrderSingle.set( queryTimeInForce() ); if ( ordType == FIX::OrdType_LIMIT || ordType == FIX::OrdType_STOP_LIMIT ) newOrderSingle.set( queryPrice() ); if ( ordType == FIX::OrdType_STOP || ordType == FIX::OrdType_STOP_LIMIT ) newOrderSingle.set( queryStopPx() ); queryHeader( newOrderSingle.getHeader() ); return newOrderSingle; } FIX42::OrderCancelRequest Application::queryOrderCancelRequest42() { FIX42::OrderCancelRequest orderCancelRequest( queryOrigClOrdID(), queryClOrdID(), querySymbol(), querySide(), FIX::TransactTime() ); orderCancelRequest.set( queryOrderQty() ); queryHeader( orderCancelRequest.getHeader() ); return orderCancelRequest; } FIX42::OrderCancelReplaceRequest Application::queryCancelReplaceRequest42() { FIX42::OrderCancelReplaceRequest cancelReplaceRequest( queryOrigClOrdID(), queryClOrdID(), FIX::HandlInst( '1' ), querySymbol(), querySide(), FIX::TransactTime(), queryOrdType() ); if ( queryConfirm( "New price" ) ) cancelReplaceRequest.set( queryPrice() ); if ( queryConfirm( "New quantity" ) ) cancelReplaceRequest.set( queryOrderQty() ); queryHeader( cancelReplaceRequest.getHeader() ); return cancelReplaceRequest; } void Application::queryHeader( FIX::Header& header ) { header.setField( querySenderCompID() ); header.setField( queryTargetCompID() ); if ( queryConfirm( "Use a TargetSubID" ) ) header.setField( queryTargetSubID() ); } char Application::queryAction() { char value; std::cout << std::endl << "1) Enter Order" << std::endl << "2) Cancel Order" << std::endl << "3) Replace Order" << std::endl << "4) Market data test" << std::endl << "5) Quit" << std::endl << "Action: "; std::cin >> value; switch ( value ) { case '1': case '2': case '3': case '4': case '5': break; default: throw std::exception(); } return value; } int Application::queryVersion() { char value; std::cout << std::endl << "1) FIX.4.0" << std::endl << "2) FIX.4.1" << std::endl << "3) FIX.4.2" << std::endl << "4) FIX.4.3" << std::endl << "5) FIX.4.4" << std::endl << "6) FIXT.1.1 (FIX.5.0)" << std::endl << "BeginString: "; std::cin >> value; switch ( value ) { case '1': return 40; case '2': return 41; case '3': return 42; case '4': return 43; case '5': return 44; case '6': return 50; default: throw std::exception(); } } bool Application::queryConfirm( const std::string& query ) { std::string value; std::cout << std::endl << query << "?: "; std::cin >> value; return toupper( *value.c_str() ) == 'Y'; } FIX::SenderCompID Application::querySenderCompID() { std::string value; std::cout << std::endl << "SenderCompID: "; std::cin >> value; return FIX::SenderCompID( value ); } FIX::TargetCompID Application::queryTargetCompID() { std::string value; std::cout << std::endl << "TargetCompID: "; std::cin >> value; return FIX::TargetCompID( value ); } FIX::TargetSubID Application::queryTargetSubID() { std::string value; std::cout << std::endl << "TargetSubID: "; std::cin >> value; return FIX::TargetSubID( value ); } FIX::ClOrdID Application::queryClOrdID() { std::string value; std::cout << std::endl << "ClOrdID: "; std::cin >> value; return FIX::ClOrdID( value ); } FIX::OrigClOrdID Application::queryOrigClOrdID() { std::string value; std::cout << std::endl << "OrigClOrdID: "; std::cin >> value; return FIX::OrigClOrdID( value ); } FIX::Symbol Application::querySymbol() { std::string value; std::cout << std::endl << "Symbol: "; std::cin >> value; return FIX::Symbol( value ); } FIX::Side Application::querySide() { char value; std::cout << std::endl << "1) Buy" << std::endl << "2) Sell" << std::endl << "3) Sell Short" << std::endl << "4) Sell Short Exempt" << std::endl << "5) Cross" << std::endl << "6) Cross Short" << std::endl << "7) Cross Short Exempt" << std::endl << "Side: "; std::cin >> value; switch ( value ) { case '1': return FIX::Side( FIX::Side_BUY ); case '2': return FIX::Side( FIX::Side_SELL ); case '3': return FIX::Side( FIX::Side_SELL_SHORT ); case '4': return FIX::Side( FIX::Side_SELL_SHORT_EXEMPT ); case '5': return FIX::Side( FIX::Side_CROSS ); case '6': return FIX::Side( FIX::Side_CROSS_SHORT ); case '7': return FIX::Side( 'A' ); default: throw std::exception(); } } FIX::OrderQty Application::queryOrderQty() { long value; std::cout << std::endl << "OrderQty: "; std::cin >> value; return FIX::OrderQty( value ); } FIX::OrdType Application::queryOrdType() { char value; std::cout << std::endl << "1) Market" << std::endl << "2) Limit" << std::endl << "3) Stop" << std::endl << "4) Stop Limit" << std::endl << "OrdType: "; std::cin >> value; switch ( value ) { case '1': return FIX::OrdType( FIX::OrdType_MARKET ); case '2': return FIX::OrdType( FIX::OrdType_LIMIT ); case '3': return FIX::OrdType( FIX::OrdType_STOP ); case '4': return FIX::OrdType( FIX::OrdType_STOP_LIMIT ); default: throw std::exception(); } } FIX::Price Application::queryPrice() { double value; std::cout << std::endl << "Price: "; std::cin >> value; return FIX::Price( value ); } FIX::StopPx Application::queryStopPx() { double value; std::cout << std::endl << "StopPx: "; std::cin >> value; return FIX::StopPx( value ); } FIX::TimeInForce Application::queryTimeInForce() { char value; std::cout << std::endl << "1) Day" << std::endl << "2) IOC" << std::endl << "3) OPG" << std::endl << "4) GTC" << std::endl << "5) GTX" << std::endl << "TimeInForce: "; std::cin >> value; switch ( value ) { case '1': return FIX::TimeInForce( FIX::TimeInForce_DAY ); case '2': return FIX::TimeInForce( FIX::TimeInForce_IMMEDIATE_OR_CANCEL ); case '3': return FIX::TimeInForce( FIX::TimeInForce_AT_THE_OPENING ); case '4': return FIX::TimeInForce( FIX::TimeInForce_GOOD_TILL_CANCEL ); case '5': return FIX::TimeInForce( FIX::TimeInForce_GOOD_TILL_CROSSING ); default: throw std::exception(); } }
main.cpp文件:
#include "config.h" #include "quickfix/FileStore.h" #include "quickfix/SocketInitiator.h" #include "quickfix/SessionSettings.h" #include "quickfix/Log.h" #include "Application.h" #include <string> #include <iostream> #include <fstream> int main( int argc, char** argv ) { if ( argc < 2 ) { std::cout << "usage: " << argv[ 0 ] << " FILE." << std::endl; return 0; } std::string file = argv[ 1 ]; FIX::Initiator * initiator = 0; try { FIX::SessionSettings settings( file ); Application application; FIX::FileStoreFactory storeFactory( settings ); FIX::ScreenLogFactory logFactory( settings ); initiator = new FIX::SocketInitiator( application, storeFactory, settings, logFactory ); initiator->start(); application.run(); initiator->stop(); delete initiator; return 0; } catch ( std::exception & e ) { std::cout << e.what(); delete initiator; return 1; } }