FIX协议介绍

1,230次阅读

共计 15892 个字符,预计需要花费 40 分钟才能阅读完成。

FIX 协议介绍

来源:宜信技术学院 college.creditease.cn 作者:姜永念

定义


FIX 协议是由国际 FIX 协会组织提供的一个开放式协议,目的是推动国际贸易电子化的进程,在各类参与者之间,包括投资经理、经纪人,买方、卖方建立起实时的电子化通讯协议。FIX 协议的目标是把各类证券金融业务需求流程格式化,使之成为一个个可用计算机语言描述的功能流程,并在每个业务功能接口上统一交换格式,方便各个功能模块的连接。

协议工作原理

通信模型及基本概念

通信模型

  • Initiator:发起者,建立通信连路,通过发送初始 Logon 消息发起会话的参与方。
  • Acceptor:接收方 FIX 会话的接收方。负责执行第一层次的认证和通过传输 Logon 消息的确认正式声明连接请求被接受。
  • 原则:先发起者为 Initiator,接受者为 Acceptor。
  • 标准模式以网关为 Acceptor,客户端为 Initiator 做为常用模式。

Fix connection

  • FIX 连接 由 3 部分组成:logon 登录,message exchange 消息传输,logout 注销。
    • logon 登录

FIX 协议介绍

  • logout 注销

FIX 协议介绍

Fix session

  • FIX 会话由一个或多个 FIX Connection FIX 连接组成。一个 FIX 会话可以有多次登录。

序列号

  • 所有的 FIX 消息都由一个唯一的序列号进行标示。序列号在每一个 FIX 会话开始时被初始化为 1,并在整个会话期间递增。监控序列号可以使会话参与者识别和处理丢失的消息,当在一个 FIX 会话中重新连接时能够快速进行应用程序同步。
  • 每个会话将建立一组互不依赖的接受和发送序列。会话参与者将维护一个赋予发送消息的序列和一个监控接受消息的消息块间隙序列号。

心跳

  • 在消息交互期间,FIX 应用程序将周期性产生 Heartbeat 心跳消息。该心跳消息可以监控通信链路状态及识别接收序列号间隙。发送 Heartbeat 的周期间隔由会话发起者使用在 Logon 消息中 HeartBtInt 域进行定义。
  • Heartbeat 心跳消息的时间间隔应当在每一个消息发送后复位,即发送一个消息后,在间隔给定的时间内无其它消息发送则发送一个 Heartbeat 心跳消息。HeartBtInt 的值应当被会话双方认同,由会话发起方定义并由会话接收者通过 Logon 消息进行确认。同一个 HeartBtInt 被会话双方——登录的发起者和登录的接受者共同使用。

数据完整校验

  • 消息数据内容的完整性可以参用两种方式来验证:消息长度和效验码检查。
  • 程序通过计算 BodyLength 域到 CheckSum 标记(“10=”)分界符的字符数,域 BodyLength 标示的消息长度进行比较来完成完整性效验。
  • ChekSum 完整性检查,通过计算从域“8=”中“8”开始,包括紧跟在 CheckSum 标记域的分界符每个字符的 2 进制和同 CheckSum 进行比较得到。
  • 一个 FIX 消息校验和通过计算到 ChechSum 域(但不包括)的消息的每个字节和得到。然后,校验和被转换为模 256 的数字用于传送和比较。校验和在所有加密操作之后被计算。
  • 校验代码:
样例:8=FIX.4.29=7335=A34=149=CLIENT52=20181119-10:42:48.76856=SERVER98=0108=30141=Y10=208
1、消息长度:9=73
35=A34=149=CLIENT52=20181119-10:42:48.76856=SERVER98=0108=30141=Y(这段长度)2、效验码检查
char *GenerateCheckSum(char *buf, long bufLen) {static char tmpBuf[ 4]; long idx;
unsigned int cks;
for(idx = 0L, cks = 0; idx < bufLen; cks += (unsigned int)buf[idx++] ); sprintf(tmpBuf,“%03d”, (unsigned int)(cks % 256) );
return(tmpBuf);
}

消息确认

  • FIX 协议不支持单个消息的确认。采用的是监控消息时隙的方法来进行消息恢复和验证。
  • 普通的数据传送(无单个消息确认)通过消息序列间隙进行错误识别。每个消息由一个唯一的序列号进行标示。接收端应用程序负责监控接收消息序列号以识别消息间隙并产生重传请求。
  • 每个 FIX 参与方必须为 FIX 会话维护两个序列号,一个是接收序列号,一个是发送序列号,两者都在建立 FIX 会话开始时初始化为 1。每个消息被赋予一个唯一的序列号值,并在消息发送后递增。此外,每个收到的消息都有一个唯一的序列号,接收序列号计数器在收到每个消息后将会被递增。
  • 当接收序列号与所希望得到的的正确序列号不必配时,必须采取纠错处理。

加密

  • 加密算法由连接双方共同协商。
  • 一个消息的任何一个域可以被加密并放在 SecureData 域中。然而,一些显示的标志域必须采用明文进行传输。为确保完整性,明文域可以在 SecureData 域中重复。
  • 当使用加密时,建议但不是必须,所有的消息体都进行加密。如果一个消息中的重复组数据中的部分数据要加密,这个重复组必须全部进行加密。
  • 预先协商好的加密算法在 Logon 消息中进行声明。

自定义域

  • FIX 为给用户提供最大的灵活性,FIX 协议允许用户自定义域。这些域在认同的参与者之间实现、应用,并且应注意避免冲突。
  • Tag 数在 5000 到 9999 保留用于用户自定义域。这些 tag 值用于企业联盟的信息交换。可以通过 FIX 网站进行注册。
  • 10000 以上保留用于单一企业内部使用。不用注册。

消息格式

数据类型

整数 int, 浮点数 float, 单个字符 char, 布尔 Boolean, 字符串 String, 数据 data

常见域

Tag(标记) FieldName(域名) 备注
8 BeginString 起始串,FIX 协议版本
9 BodyLength 消息长度
35 MsgType 消息类型:例如 F =Order Cancel Request,取消订单
11 ClOrdID 客户端订单 ID
37 OrderID 服务端订单 ID
41 OrigClOrdID 原始客户端订单 ID
54 Side 买卖类型。例如:1 = Buy,2 = Sell
55 Symbol 股票代码。例如:YRD
10 CheckSum 校验码

域语法

  • 开始部分应是消息头,随后是正文,最后是消息尾;
  • 消息头的前 3 个域的次序不能改变: 起始串(Tag =8)、消息体长度(Tag =9)、消息类型(Tag =35);
  • 消息尾的最后一个域应是校验和域(Tag=10);
  • 重复组中,域出现的顺序应遵循该重复组在消息或组件中定义时的次序;
  • 在一条消息中,除重复组域外任何其他域不能重复出现。

安全与加密

  • 由于消息有可能在公网或不安全的网络上传输交换,因此需要对相关的敏感数据加密处理。
  • 具体加密的方法由连接双方达成的协议而定。
  • 消息内除某些需要公开识别的域以明文传输外其他任何域都可以加密放置密文数据域 (SecureData)内。当然,这些被加密的域也可以同时保留明文的表示方式。
  • 当决定使用加密方案时,可以对消息正文内所有的域加密。如果消息的重复组内有部分需要加密的,那么要求对整个重复组加密。
  • 本协议还提供的一些域用以支持数字签名、密钥交换和正文加密等安全技术。

消息

消息头

每一个会话或应用消息有一个消息头,该消息头指明消息类型、消息体长度、发送目的地、消息序号、发送起始点和发送时间。

Tag 域名 必需 说明
8 BeginString Y 起始串,取值:FIX.4.2(不可加密,消息的第一个域)
9 BodyLength Y 消息体长度(不可加密,消息的第二个域)
35 MsgType Y 消息类型(不可加密,消息的第三个域)
49 SenderCompID Y 发送方代码(不可加密,发送方标识符)
56 TargetCompID Y 接收方代码(不可加密,接收方标识符)
115 OnBehalfOfCompID N 最初发送方标识符(可加密),用于经第三方发送。
128 DeliverToCompID N 最终接收方标识符(可加密),用于经第三方发送。
90 SecureDataLen N 密文数据长度
91 SecureData N 密文数据(紧跟密文数据长度域)
34 MsgSeqNum Y 消息序号(可加密),如果交易双方不采用 FIX 会话 机制,可将该 tag 置为一个固定的值,例如 0。
50 SenderSubID N 发送方子标识符(可加密)
142 SenderLocationID N 发送方方位标识符(可加密)
57 TargetSubID N 接收方子标识符(可加密)
143 TargetLocationID N 接收方方位标识符(可加密)
116 OnBehalfOfSubID N 最初发送方子标识符(可加密)
144 OnBehalfOfLocationID N 最初发送方方位标识符(可加密)
129 DeliverToSubID N 最终接收方子标识符(可加密)
145 DeliverToLocationID N 最终接收方方位标识符(可加密)
43 PossDupFlag N 可能重复标志,重复发送时,作此标记。(可加密)
97 PossResend N 可能重发标志。(可加密)
52 SendingTime Y 发送时间(可加密)
122 OrigSendingTime N 原始发送时间(可加密)
347 MessageEncoding N 消息中 Encoded 域的字符编码类型(非 ASCII 码)
369 LastMsgSeqNumProcesse d N 最后处理消息序号(可加密)
370 OnBehalfOfSendingTime N 最初发送时间(用 UTC 表示时间)

消息尾

每一个消息 (会话或应用消息) 有一个消息尾,并以此终止。消息尾可用于分隔多个消息,包含有 3 位数的校验和值。

Tag 域名 必需 说明
93 SignatureLength N 数字签名长度(不可加密)
89 Signature N 数字签名(不可加密)
10 CheckSum Y 校验和,消息的最末域。(不可加密)

新订单消息(MsgType=D)

对于在消息头中设置了 PossResend 标志的订单消息,应当使用交易客户方订单编号 (ClOrdID) 核 实是否已收到该订单,具体实现时还应检查订单参数 (买卖方向、证券代码、数量等) 进行核实。如果 之前收到该订单,应以执行报告消息回应订单状态。如果之前未收到,则以执行报告消息回应订单确认。

FIX 协议介绍

Tag 域名 必需 说明
标准消息头 Y MsgType=D
11 ClOrdID Y 交易客户方订单编号,在订单有效交易日内必需
109 ClientID Y 客户资金帐号
1 Account Y 客户交易编码
110 MinQty N 最小成交量。
55 Symbol Y 期货合约代码
167 SecurityType N FUT = 期货
200 MaturityMonthYear N 用于指定期货到期的年和月
205 MaturityDay N 用于期货的到期日期,并被与到期年月 (MaturityMonthYear)联合使用
207 SecurityExchange Y 用于指定交易所
77 OpenClose Y 指明开仓,平仓
8009 HedgeFlag Y 投机套保标志
8010 TouchCondition N 触发条件
54 Side Y 买卖方向
38 OrderQty N 委托手数
60 TransactTime Y 订单发起时间
40 OrdType Y 订单类型
44 Price N 价格(限价订单时有效)
423 PriceType N 价格类型
99 StopPx N 停止价
15 Currency N 币种
59 TimeInForce N 新订单生效时间,默认为当日有效
168 EffectiveTime N 用于指定定单有效的时间
432 ExpireDate N 有条件地用于在生效时间 (TimeInForce)= 在某 日前有效(GTD),而没有指定截止时间 (ExpireTime) 的情况之下
126 ExpireTime N 有条件地用于生效时间 (TimeInForce) = 在某 日前有效(GTD) 和到期日没有被指定的情况之 下
8096 MacNetInfo N 委托方的机器网络信息
标准消息尾 Y

执行报告消息(MsgType=8)

  • 订单确认
  • 订单状态变化确认(如撤单确认)
  • 发送订单的成交回报
  • 订单拒绝
Tag 域名 必需 说明
标准消息头 Y MsgType=8
37 OrderID Y 期货公司委托号,同个交易日必需保证唯一
11 ClOrdID N 交易客户方订单编号。如果是强平回报,则该值 取值为以”NONE”开头的当天交易日唯一的字符 串标识
41 OrigClOrdID N 原始交易客户方订单编号,指示被撤消订单的 ClOrdID
17 ExecID Y 期货公司的执行编号,在订单有效交易日内应保 证唯一
150 ExecType Y 执行类型
39 OrdStatus Y 订单状态
103 OrdRejReason N 订单拒绝时需要
109 ClientID Y 客户资金帐号
1 Account Y 客户交易编码
55 Symbol Y 期货合约代码
167 SecurityType N FUT= 期货
200 MaturityMonthYear N 到期年月
205 MaturityDay N 到期日期
207 SecurityExchange Y 用于指定交易所
77 OpenClose N 指明开仓,平仓
54 Side Y 买卖方向
38 OrderQty Y 委托手数
40 OrdType N 订单类型
44 Price N 订单价格
99 StopPx N 停止价
59 TimeInForce N 新订单生效时间,默认为当日有效
15 Currency N 币种
32 LastShares N 上一成交数(最近一笔成交数量)
31 LastPx N 上一成交价(最近一笔成交价格)
30 LastMkt N 上一成交市场
151 LeavesQty Y 订单剩余数量
14 CumQty Y 成交总数
6 AvgPx Y 成交平均价
60 TransactTime N 执行报告时间
381 GrossTradeAmt N 成交总金额
110 MinQty N 最小成交量
8500 OrderEntryTime N 订单申报时间
8093 DeclarationID N 报单号
8094 TradeID N 撮合编号
标准消息尾 Y

订单状态请求消息(MsgType=H)

订单状态请求用于向交易服务方请求某订单的状态,交易服务方通过执行报告消息返回订单状态。

Tag 域名 必需 说明
标准消息头 Y MsgType=H
37 OrderID Y 期货公司委托号,同个交易日必需保证唯一
11 ClOrdID Y 交易客户方订单编号
109 ClientID Y 客户资金帐号
1 Account Y 客户交易编码
55 Symbol Y 期货合约代码
207 SecurityExchange Y 用于指定交易所
167 SecurityType N FUT= 期货
200 MaturityMonthYear N 用于指定期货到期的年和月
205 MaturityDay N 用于期货的到期日期,并被与到期年月 (MaturityMonthYear)联合使用
54 Side Y 买卖方向
标准消息尾 Y

撤单消息(MsgType=F)

撤单消息用以撤消订单的全部订单剩余数量。
撤单消息也被赋予一个 ClOrdID,可视作另外一个订单。如果被拒绝,撤单拒绝消息的 ClOrdID 放 置撤单消息的 ClOrdID,而原始订单的 ClOrdID 则放入 OrigClOrdID 域。ClOrdID 要保证唯一。

Tag 域名 必需 说明
标准消息头 Y MsgType=F
41 OrigClOrdID Y 原始交易客户方订单编号,指示被撤消订单的 ClOrdID
37 OrderID Y 期货公司委托号,同个交易日必需保证唯一
11 ClOrdID Y 交易客户方订单编号
109 ClientID Y 客户资金帐号
1 Account Y 客户交易编码
55 Symbol Y 期货合约代码。
167 SecurityType N 证券代码源
200 MaturityMonthYear N FUT= 期货
205 MaturityDay N 期货到期年月
207 SecurityExchange Y 期货到期日期
54 Side Y 买卖方向
60 TransactTime Y 订单发起时间
40 OrdType Y 订单类型
38 OrderQty Y 委托手数
8093 DeclarationID N 报单号
58 Text N
标准消息尾 Y

撤单拒绝消息(MsgType=9)

本消息用于撤单消息的拒绝。
交易服务方接收到撤单发现无法执行(已成交订单不可更改等),将发送撤单拒绝。
拒绝撤单时,撤单拒绝消息应用 ClOrdID 指示撤单的 ClOrdID,用 OrigClOrdID 指示之前最后接受的订单(除非拒绝原因是“未知订单”)。

Tag 域名 必需 说明
标准消息头 Y MsgType=9
37 OrderID Y 期货公司委托号,同个交易日必需保证唯一
11 ClOrdID Y 交易客户方订单编号
41 OrigClOrdID Y 原始交易客户方订单编号,指示被撤消订单的 ClOrdID
39 OrdStatus Y 订单状态
109 ClientID Y 客户资金帐号
1 Account Y 客户交易编码
60 TransactTime N 订单发起时间
434 CxlRejResponseTo N 撤单拒绝回应类型
102 CxlRejReason N 撤单拒绝原因
58 Text N
标准消息尾 Y

FIX 配置

  • 会话配置(SESSION)
配置 描述 有效值 默认
BeginString 会话使用的 FIX 版本号(发送和接收消息起始字符串) FIXT.1.1、FIX.4.4、FIX.4.3、FIX.4.2、FIX.4.1、FIX.4.0
SenderCompID 会话当中定义本方的 ID 区分大小写的字符串
SenderSubID 会话相关的本方的子 ID 号 (可选) 区分大小写的字符串
SenderLocationID 会话相关的本方的 locationID 号 (可选) 区分大小写的字符串
TargetCompID 本会话当中的对方 ID 区分大小写的字符串
TargetSubID 本会话当中的对方 SubID (可选) 区分大小写的字符串
TargetLocationID 本会话当中的对方 locationID (可选) 区分大小写的字符串
SessionQualifier 附加的限定词,用于消除歧义,保证会话的唯一性 区分大小写的字符串
DefaultApplVerID 仅 FIXT1.1(或以上版本)需要。忽略早期版本的传输。指定会话的默认应用程序的版本 ID。ApplVerID 的枚举值(请看 ApplVerID 字段详细介绍),或默认 BeginString。 FIX.5.0SP2、FIX.5.0SP1、FIX.5.0、FIX.4.4、FIX.4.3、FIX.4.2、FIX.4.1、FIX.4.0
ConnectionType 定义会话当中本方的角色:acceptor 或者 initiator initiator、acceptor
StartTime 交易日的会话有效开始时间,这时 FIX 会话被激活 UTC 时间,格式:HH:MM:SS
EndTime 交易日的会话失效时间,FIX 会话将被停止 UTC 时间,格式:HH:MM:SS
StartDay 对于为期一周的会话配置,一周会话开始的第一天。与 STARTTIME 结合使用。 使用一周中某天的英语任何缩写都是有效的(比如,mo, mon, mond, monda,Monday 都是有效的)
EndDay 对于为期一周的会话配置,一周会话结束的最后一天。与 EndTime 结合使用。 使用一周中某天的英语任何缩写都是有效的(比如,mo, mon, mond, monda,Monday 都是有效的)
MillisecondsInTimeStamp 时间戳是否加入毫秒。FIX.4.2 和更高版本可用。 Y、N Y
ResetOnLogon 接收登录请求时,序列号是否要复位。只用于 Acceptor Y、N N
ResetOnLogout 正常注销登录时,序列号是否要复位 Y、N N
ResetOnDisconnect 连接异常断开后是否要将序列号重置为 1 Y、N N
RefreshOnLogon 确定是否应当从持久层登录时恢复会话状态。在创建热故障切换会话时有用。 Y、N N
EnableLastMsgSeqNumProcessed 是否在 header 中添加最后一条消息的序列号(可选 tag369)。 Y、N N
MaxMessagesInResendRequest 设置一次重发请求的消息的最大消息数。 任何大于 0 的整数。使用 0 为无穷大(默认)。 0
SendLogoutBeforeDisconnectFromTimeout 指定是否因超时断开连接之前发送 logout 消息 Y、N N
IgnorePossDupResendRequests 当 PossDupFlag(tag 43)设置为 true 时,是否忽略一次重发请求 Y、N N
  • 验证配置
配置 描述 有效值 默认
UseDataDictionary 告诉会话是否使用数据字典,或不希望使用数据字典。如果你要使用 repeating group,你必须使用 DataDictionary。 Y、N Y
DataDictionary 该配置只用于比 FIXT.1.1 还老的版本。详细参考 FIXT.1.1 的 TransportDataDictionary 和 AppDataDictionary 的配置。 FIX44.xml、FIX43.xml、FIX42.xml、FIX41.xml、FIX40.xml
TransportDataDictionary XML 定义文件用于验证传入的管理消息。如果没有提供 DataDictionary,只会做基本消息的验证。该配置只用于 FIXT.1.1(或更高版本)的会话。 FIXT1.1.xml
AppDataDictionary 用于验证应用层消息的 XML 定义文件。仅对 FIXT.1.1(或更高版本)的会话有效。更多信息请参考(FIX.4.0 到 FIX.4.4)的 DataDictionary。该配置可以为每个会话指定一个自定义应用的数据字典。该配置仅用于 FIXT.1.1 或更新的传输协议。使用 FIXT 传输时,该配置可以作为指定多个应用的数据字典的前缀。例如:DefaultApplVerID=FIX.4.2 # For default application version ID AppDataDictionary=FIX42.xml # For nondefault application version ID # Use BeginString suffix for app version AppDataDictionary.FIX.4.4=FIX44.xml 有效的 XML 数据字典文件。QuickFIX/N 配备默认的协议字典数据:FIX50SP2.xml、FIX50SP1.xml、FIX50.xml、FIX44.xml、FIX43.xml、FIX42.xml、FIX41.xml、FIX40.xml
ValidateFieldsOutOfOrder 如果设置为 N,字段放置区域错误(例如,body 字段在 header 区域内,或在 header 字段在 body 区域内)将不会被拒绝。用于连接字段要求不严格的系统。 Y、N Y
ValidateFieldsHaveValues 如果设置为 N,没有值的字段将不会被拒绝。用于连接到系统不当发送空标签。 Y、N Y
ValidateUserDefinedFields 如果设置为 N,用户自定义的字段将不会被拒绝,即使没有在数据字典中定义,或没出现在消息中。 Y、N Y
  • Initiator
配置 描述 有效值 默认
ReconnectInterval 尝试重新连接的时间间隔(秒)。仅用于 initiator。 正整数 30
HeartBtInt 心跳间隔(秒)。仅用于 initiator。 正整数 -
LogonTimeout 登录超时时间间隔(秒) 正整数 10
LogoutTimeout 注销登录超时时间间隔(秒) 正整数 2
SocketConnectPort Socket 服务端口,用于建立会话。仅用于 initiator 正整数 -
SocketConnectHost 连接主机. 仅用于 initiator x.x.x.x 格式 IP 地址或域名 -
SocketConnectPort 一组备用 Socket 端口,用于连接会话的故障转移,n 是正整数。SocketConnectPort1,SocketConnectPort2 ... 必须是连续的,并有一个与之相匹配的数组 SocketConnectHost 正整数 -
SocketConnectHost 一组备用 Socket 服务主机,用于连接会话的故障转移,n 是正整数。SocketConnectHost1, SocketConnectHost2... 必须是连续的,并有一个与之相匹配的数组 SocketConnectPort x.x.x.x 格式 IP 地址或域名 -
SocketNodelay 连接是否禁用 Nagle 算法。在 \[DEFAULT\]配置节点定义。 Y、N Y
ReconnectInterval 尝试重新连接的时间间隔(秒)。仅用于 initiator。 正整数 30
  • Acceptor
配置 描述 有效值 默认
SocketAcceptPort 监听接入连接 Socket 端口。仅用于 acceptor 正整数,有效的、开放的套接字端口 -
SocketAcceptHost 监听接入连接的 Socket 服务的主机。如果不提供,acceptor 将监听所有网络端口(0.0.0.0) 有效的 x.x.x.x 格式 IP 地址 0.0.0.0
SocketNodelay 连接是否禁用 Nagle 算法。在 \[DEFAULT\]配置节点定义。 Y、N Y
  • Storage
配置 描述 有效值 默认
PersistMessages 如果设置为 N,被不会保存消息。这样将迫使 quickfix 总是发送 GapFills,而不是重新发送消息。如果你知道你永远不需要重新发送消息,使用此配置。有用的市场数据流。 Y、N Y
  • File Storage
配置 描述 有效值 默认
FileStorePath 存储序列号和消息的文件目录。 有效的文件存储目录,必须有写入权限。 -
  • Logging
配置 描述 有效值 默认
FileLogPath 存储日志的目录。 有效的文件存储目录,必须有写入权限。 -

FIX 开发

FIX 引擎

DEMO

  • Acceptor 配置文件
[DEFAULT]
FileStorePath=store
FileLogPath=log
ConnectionType=acceptor
ReconnectInterval=60
SenderCompID=SERVER
ResetOnDisconnect=Y
ResetOnLogout=Y
ResetOnLogon=Y

[SESSION]
BeginString=FIX.4.2
TargetCompID=CLIENT
StartTime=00:00:00
EndTime=23:59:59
HeartBtInt=30
SocketAcceptHost=127.0.0.1
SocketAcceptPort=6666
DataDictionary=FIX42.xml
  • Initiator 配置文件
[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
  • FixServer
package com.app.fix;

import quickfix.*;

 * 服务启动主类(线程)
 */
public class FixServer {
    private static ThreadedSocketAcceptor acceptor = null;

     * 指定配置文件启动
     *
     * @param propFile
     * @throws ConfigError
     * @throws FieldConvertError
     */
    public FixServer(String propFile) throws ConfigError, FieldConvertError {SessionSettings settings = new SessionSettings(propFile);

        Application application = new FixServerApplication();

         *
         * quickfix.MessageStore 有 2 种实现。quickfix.JdbcStore,quickfix.FileStore .
         * JdbcStoreFactory 负责创建 JdbcStore,FileStoreFactory 负责创建 FileStorequickfix
         * 默认用文件存储,因为文件存储效率高。*/
        MessageStoreFactory storeFactory = new FileStoreFactory(settings);

        LogFactory logFactory = new FileLogFactory(settings);

        MessageFactory messageFactory = new DefaultMessageFactory();

        acceptor = new ThreadedSocketAcceptor(application, storeFactory, settings, logFactory, messageFactory);

    }

    private void startServer() throws RuntimeError, ConfigError {acceptor.start();
    }

     * 测试本地使用的 main 方法
     *
     * @param args
     * @throws FieldConvertError
     * @throws ConfigError
     */
    public static void main(String[] args) throws ConfigError, FieldConvertError {FixServer fixServer = new FixServer("res/acceptor.config");
        fixServer.startServer();}

}
  • FixServerApplication
package com.app.fix;

import quickfix.Application;
import quickfix.DoNotSend;
import quickfix.FieldNotFound;
import quickfix.IncorrectDataFormat;
import quickfix.IncorrectTagValue;
import quickfix.Message;
import quickfix.MessageCracker;
import quickfix.RejectLogon;
import quickfix.Session;
import quickfix.SessionID;
import quickfix.UnsupportedMessageType;
import quickfix.field.MsgType;

 * 
 */
public class FixServerApplication extends MessageCracker implements Application {
    @Override
    protected void onMessage(Message message, SessionID sessionID) {
        try {String msgType = message.getHeader().getString(35);
            Session session = Session.lookupSession(sessionID);
            switch (msgType) {
                case MsgType.LOGON: 
                    session.logon();
                    session.sentLogon();
                    break;
                case MsgType.HEARTBEAT: 
                    session.generateHeartbeat();
                    break;
            }

        } catch (FieldNotFound e) {e.printStackTrace();
        }

    }

    @Override
    public void onCreate(SessionID sessionId) {System.out.println(" 服务器启动时候调用此方法创建 ");

    }

    @Override
    public void onLogon(SessionID sessionId) {System.out.println(" 客户端登陆成功时候调用此方法 ");

    }

    @Override
    public void onLogout(SessionID sessionId) {System.out.println(" 客户端断开连接时候调用此方法 ");

    }

    @Override
    public void toAdmin(Message message, SessionID sessionId) {System.out.println(" 发送会话消息时候调用此方法 ");

    }

    @Override
    public void toApp(Message message, SessionID sessionId) throws DoNotSend {System.out.println(" 发送业务消息时候调用此方法 ");

    }

    @Override
    public void fromAdmin(Message message, SessionID sessionId)
            throws FieldNotFound, IncorrectDataFormat, IncorrectTagValue, RejectLogon {System.out.println(" 接收会话类型消息时调用此方法 ");
        try {crack(message, sessionId);
        } catch (UnsupportedMessageType | FieldNotFound | IncorrectTagValue e) {e.printStackTrace();
        }

    }

    @Override
    public void fromApp(Message message, SessionID sessionId)
            throws FieldNotFound, IncorrectDataFormat, IncorrectTagValue, UnsupportedMessageType {System.out.println(" 接收业务消息时调用此方法 ");
        crack(message, sessionId);

    }

}
  • FixClient
package com.app.fix;

import quickfix.*;
import quickfix.field.*;
import quickfix.fix42.NewOrderSingle;

import java.io.FileNotFoundException;
import java.util.Date;

public class FixClient implements Application {

    private static volatile SessionID sessionID;

    @Override
    public void onCreate(SessionID sessionID) {System.out.println("OnCreate");
    }

    @Override
    public void onLogon(SessionID sessionID) {System.out.println("OnLogon");
        FixClient.sessionID = sessionID;
    }

    @Override
    public void onLogout(SessionID sessionID) {System.out.println("OnLogout");
        FixClient.sessionID = null;
    }

    @Override
    public void toAdmin(Message message, SessionID sessionID) {System.out.println("ToAdmin");
    }

    @Override
    public void fromAdmin(Message message, SessionID sessionID) throws FieldNotFound, IncorrectDataFormat, IncorrectTagValue, RejectLogon {System.out.println("FromAdmin");
    }

    @Override
    public void toApp(Message message, SessionID sessionID) throws DoNotSend {System.out.println("ToApp: " + message);
    }

    @Override
    public void fromApp(Message message, SessionID sessionID) throws FieldNotFound, IncorrectDataFormat, IncorrectTagValue, UnsupportedMessageType {System.out.println("FromApp");
    }

    public static void main(String[] args) throws ConfigError, FileNotFoundException, InterruptedException, SessionNotFound {SessionSettings settings = new SessionSettings("res/initiator.config");

        Application application = new FixClient();
        MessageStoreFactory messageStoreFactory = new FileStoreFactory(settings);
        LogFactory logFactory = new ScreenLogFactory(true, true, true);
        MessageFactory messageFactory = new DefaultMessageFactory();

        Initiator initiator = new SocketInitiator(application, messageStoreFactory, settings, logFactory, messageFactory);
        initiator.start();

        while (sessionID == null) {Thread.sleep(1000);
        }

        final String orderId = "342";
        NewOrderSingle newOrder = new NewOrderSingle(new ClOrdID(orderId), new HandlInst('1'), new Symbol("YRD"),
                new Side(Side.BUY), new TransactTime(new Date()), new OrdType(OrdType.MARKET));
        Session.sendToTarget(newOrder, sessionID);
        Thread.sleep(5000);
    }
}
正文完
扫码关注公众学习更多内容
post-qrcode
   
jiajianfa007
版权声明:本站原创文章,由 jiajianfa007 2023-07-14发表,共计15892字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。