[增加App] 账号信息自动入库

This commit is contained in:
msincenselee 2020-01-19 17:20:02 +08:00
parent 65bfdd2241
commit e0d7dff896
5 changed files with 650 additions and 1 deletions

View File

@ -13,6 +13,7 @@
- 提供主力合约/指数合约的信息获取
- 提供期货/股票数据bar 和分笔成交数据下载
- 提供每日增量更新期货数据=> csv文件可配合NFS+Celery实现分布式回测
4. 增加App: tick_recorder, 直接异步写入csv文件
5. 增加App: index_tick_publisher, 订阅通达信指数行情=》rabbit_mq 推送
6. 增强ctp_gateway包括:
@ -20,6 +21,7 @@
- 提供指数行情订阅
- 使用RabbitMQ指数源或tdx单一数据源
- 提供自定义合约功能实时提供其合成后的tick行情
7. 增加App: cta_strategy_pro包括
- 提供cta_line_bar k线组件支持国内文华/交易师/TB等分钟/小时的计算模式,支持任意秒/分钟/小时/天/周等周期支持k线数据实时生成。
@ -34,6 +36,21 @@
- 去除加载k线/tick初始化服务改为策略内部自行实现。
- 提供单独重启某一策略实例功能,可在线更新策略源码后,重启某一策略实例,不影响其他运行实例。
- 支持单策略多合约行情订阅,支持指数合约行情订阅
- 提供组合回测引擎能够直接加载cta_strategy_pro_setting.json文件进行组合回测。
- 增加定时器推动策略on_timer
- 增加定时推送策略持仓event
8、增强主引擎包括
- 支持同一类gateway多个接入配置
- 增加获取当前价格接口
- 增加风控引擎入口 self.rm_engine
9、增加App: account_recorder 包括:
- 异步更新账号资金/委托/成交信息至Mongo数据库
- 异步更新策略持仓数据至Mongo数据库
- 异步查询股票历史委托/历史成交至Mongo数据库
大佳
QQ/Wechat28888502

View File

@ -0,0 +1,32 @@
账号信息 sync=> mongodb
1. 监听 EVENT_ACCOUNT事件数据 => mongodb Account.account_info 和 daily_info
2. 监听 EVENT_ORDER 事件,数据 => mongodb Account.today_orders, 若激活copy_history_orders, 数据=>history_orders
3. 监听 EVENT_TRADE 事件,数据 => mongodb Account.today_trades, 若激活copy_history_trades数据=>history_trades
4. 监听 EVENT_POSITION事件,数据 => mongodb Account.today_position
5. 监听股票接口的 EVENT_HISTORY_TRADE 事件, 数据 => history_orders
6. 监听股票接口的 EVENT_HISTORY_ORDER 事件, 数据 => history_trades
7. 监听股票接口的 EVENT_FUNDS_FLOW 事件, 数据 => funds_flow
8. 监听 EVENT_STRATEGY_POS事件数据 =》 mongodb Account.today_strategy_pos
配置文件 ar_setting.json
{
"mongo_db":
{
"host": "192.168.0.207",
"port": 27017
},
"accounts":
{
"stock":
{ "history_orders": {"begin_day": "2019-07-19"},
"history_trades": {"begin_day": "2019-07-19"}
},
"ctp":
{
"copy_history_trades": true,
"copy_history_orders": true
}
}
}

View File

@ -0,0 +1,17 @@
# encoding: UTF-8
from pathlib import Path
from vnpy.trader.app import BaseApp
from .engine import AccountRecorder, APP_NAME
class AccountRecorderApp(BaseApp):
""""""
app_name = APP_NAME
app_module = __module__
app_path = Path(__file__).parent
display_name = "帐号记录"
engine_class = AccountRecorder
widget_name = None
icon_name = "algo.ico"

View File

@ -0,0 +1,581 @@
# encoding: UTF-8
'''
账号相关数据记录
1. 监听 EVENT_ACCOUNT事件数据 => mongodb Account.account_info daily_info
2. 监听 EVENT_ORDER 事件数据 => mongodb Account.today_orders
3. 监听 EVENT_TRADE 事件数据 => mongodb Account.today_trades
4. 监听 EVENT_POSITION事件,数据 => mongodb Account.today_position
5. 监听股票接口的 EVENT_HISTORY_TRADE 事件 数据 => history_orders
6. 监听股票接口的 EVENT_HISTORY_ORDER 事件 数据 => history_trades
7. 监听股票接口的 EVENT_FUNDS_FLOW 事件 数据 => funds_flow
8. 监听 EVENT_STRATEGY_POS事件数据 = mongodb Account.today_strategy_pos
# 华富资产 李来佳
'''
import json
import os
import sys
import copy
import traceback
import csv
from datetime import datetime, timedelta
from queue import Queue
from threading import Thread
from time import time
from concurrent.futures import ThreadPoolExecutor
from vnpy.event import Event, EventEngine
from vnpy.trader.event import *
from vnpy.trader.constant import Direction
from vnpy.trader.engine import BaseEngine, MainEngine
from vnpy.trader.utility import get_trading_date, load_json, save_json
from vnpy.data.mongo.mongo_data import MongoData
# 入库
ACCOUNT_DB_NAME = 'Account'
ACCOUNT_INFO_COL = 'account_info'
DAILY_INFO_COL = 'daily_info'
TODAY_ORDER_COL = 'today_orders'
TODAY_TRADE_COL = 'today_trades'
TODAY_STRATEGY_POS_COL = 'today_strategy_pos'
TODAY_POSITION_COL = 'today_positions'
HISTORY_ORDER_COL = 'history_orders'
HISTORY_TRADE_COL = 'history_trades'
HISTORY_STRATEGY_POS_COL = 'history_strategy_pos'
FUNDS_FLOW_COL = 'funds_flow'
ALERT_DB_NAME = "Alert"
GW_ERROR_COL_NAME = "gw_error_msg"
WARNING_COL_NAME = "warning_msg"
CRITICAL_COL_NAME = "critical_msg"
APP_NAME = "ACCOUNT_RECORDER"
########################################################################
class AccountRecorder(BaseEngine):
"""账号数据持久化"""
setting_file_name = 'ar_setting.json'
name = u'账户数据记录'
# ----------------------------------------------------------------------
def __init__(self, main_engine: MainEngine, event_engine: EventEngine):
"""Constructor"""
super(AccountRecorder, self).__init__(
main_engine, event_engine, APP_NAME)
# 负责执行数据库插入的单独线程相关
# 是否启动
self.active = False
self.queue = Queue() # 队列
self.thread = Thread(target=self.run) # 线程
# mongo数据库
self.mongo_db = None
# 账号的同步记录
self.account_dict = {} # gateway_name: setting
self.last_qry_dict = {}
self.copy_history_orders = [] # 需要复制至历史成交的gateway名称
self.copy_history_trades = [] # 需要复制至历史成交的gateway名称
self.copy_history_strategypos = [] # 需要复制至历史策略持仓得gateway名称
self.timer_count = 0
self.gw_name_acct_id = {}
self.is_remove_pre_data = False
self.scaning_gw = []
self.last_minute = None
# 最近10条交易接口错误回报
self.last_gw_error_msg = []
# 最近10条系统严重错误信息
self.last_critical_msg = []
# 最近10条系统警告信息
self.last_warning_msg = []
# 加载配置文件
self.load_setting()
# 启动数据库写入线程
self.start()
# 启动事件侦听
self.register_event()
def close(self):
print('AccountRecorder exit')
self.stop()
# ----------------------------------------------------------------------
def load_setting(self):
"""读取配置"""
try:
d = load_json(self.setting_file_name)
# mongo 数据库连接
mongo_seetting = d.get('mongo', {})
self.mongo_db = MongoData(host=mongo_seetting.get('host', 'localhost'),
port=mongo_seetting.get('port', 27017))
# 获取需要处理处理得账号配置
self.account_dict = d.get('accounts', {})
# 识别配置,检查账号是否需要复制委托/成交到历史表
for gateway_name, account_setting in self.account_dict.items():
if account_setting.get('copy_history_orders', False):
self.copy_history_orders.append(gateway_name)
if account_setting.get('copy_history_trades', False):
self.copy_history_trades.append(gateway_name)
if account_setting.get('copy_history_strategypos', False):
self.copy_history_strategypos.append(gateway_name)
except Exception as ex:
self.main_engine.writeCritical(u'读取:{}异常:{}'.format(self.setting_file_name, str(ex)))
def save_setting(self):
try:
save_json(self.setting_file_name, copy.copy(self.account_dict))
except Exception as ex:
self.main_engine.writeCritical(u'写入:{}异常:{}'.format(self.setting_file_name, str(ex)))
# ----------------------------------------------------------------------
def register_event(self):
"""注册事件监听"""
self.event_engine.register(EVENT_TIMER, self.update_timer)
self.event_engine.register(EVENT_ACCOUNT, self.update_account)
self.event_engine.register(EVENT_ORDER, self.update_order)
self.event_engine.register(EVENT_TRADE, self.update_trade)
self.event_engine.register(EVENT_POSITION, self.update_position)
self.event_engine.register(EVENT_HISTORY_TRADE, self.update_history_trade)
self.event_engine.register(EVENT_HISTORY_ORDER, self.update_history_order)
self.event_engine.register(EVENT_FUNDS_FLOW, self.update_funds_flow)
self.event_engine.register(EVENT_STRATEGY_POS, self.update_strategy_pos)
self.event_engine.register(EVENT_ERROR, self.process_gw_error)
self.event_engine.register(EVENT_WARNING, self.process_warning)
self.event_engine.register(EVENT_CRITICAL, self.process_critical)
# ----------------------------------------------------------------------
def update_timer(self, event: Event):
"""更新定时器"""
if len(self.account_dict) == 0 or self.active is False:
return
self.timer_count += 1
if self.timer_count < 10:
return
self.timer_count = 0
dt_now = datetime.now()
# 提交查询历史交易记录/历史委托/资金流水
for data_type in [HISTORY_ORDER_COL, HISTORY_TRADE_COL, FUNDS_FLOW_COL]:
dt = self.last_qry_dict.get(data_type, None)
queryed = False
for gw_name, account_info in self.account_dict.items():
if dt is None or (dt_now - dt).total_seconds() > 60 * 5:
begin_day = self.get_begin_day(gw_name, data_type).replace('-', '')
end_day = dt_now.strftime('%Y%m%d')
gw = self.main_engine.get_gateway(gw_name)
if gw is None:
continue
if hasattr(gw, 'qryHistory'):
self.write_log(u'{}请求{}数据,{}~{}'.format(gw_name, data_type, begin_day, end_day))
gw.qryHistory(data_type, begin_day, end_day)
queryed = True
if queryed:
self.last_qry_dict.update({data_type: dt})
# ----------------------------------------------------------------------
def update_account(self, event: Event):
"""更新账号资金"""
account = event.data
fld = {'accountid': account.accountid,
'gateway_name': account.gateway_name,
'trading_day': account.trading_day,
'currency': account.currency}
if len(account.accountid) == 0:
return
self.gw_name_acct_id.update({account.gateway_name: account.accountid})
data = account.__dict__
# 更新至历史净值数据表
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=DAILY_INFO_COL, fld=copy.copy(fld),
data=copy.copy(data))
fld.pop('trading_day', None)
# 更新至最新净值数据
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=ACCOUNT_INFO_COL, fld=fld, data=data)
self.remove_pre_trading_day_data(account.accountid, account.trading_day)
def update_begin_day(self, gw_name, data_type, begin_day):
"""
更新某一数据的开始日期
:param gw_name: 网关名称
:param data_type: 类型 history_trade, history_order, funds_flow
:param begin_day: '%Y-%m-%d'格式字符串
:return:
"""
try:
account_info = self.account_dict.get(gw_name, {})
date_info = account_info.get(data_type, {})
old_begin_day = date_info.get('begin_day', None)
if old_begin_day is None or datetime.strptime(old_begin_day, '%Y-%m-%d') < datetime.strptime(begin_day,
'%Y-%m-%d'):
self.write_log(u'{}更新:{}:{}=>{}'.format(gw_name, data_type, old_begin_day, begin_day))
date_info.update({'begin_day': begin_day})
account_info.update({data_type: date_info})
self.account_dict.update({gw_name: account_info})
self.save_setting()
except Exception as ex:
self.main_engine.writeError(u'更新数据日期异常:{}'.format(str(ex)))
self.write_log(traceback.format_exc())
def get_begin_day(self, gw_name: str, data_type: str):
"""
获取某一数据的开始日期
:param gw_name: 网关名称
:param data_type: 类型 history_trade, history_order, funds_flow
:return:
"""
account_info = self.account_dict.get(gw_name, {})
date_info = account_info.get(data_type, {})
begin_day = date_info.get('begin_day', None)
if begin_day is None:
begin_day = (datetime.now() - timedelta(days=365)).strftime('%Y-%m-%d')
return begin_day
def remove_pre_trading_day_data(self, accountid: str, trading_day: str):
"""
移除非当前交易日得所有当日交易数据
:param accountid:
:param trading_day:
:return:
"""
if self.is_remove_pre_data:
return
# 移除非当日得交易/持仓
flt = {'accountid': accountid,
'trade_date': {'$ne': trading_day}}
self.main_engine.dbDelete(ACCOUNT_DB_NAME, TODAY_TRADE_COL, flt)
self.main_engine.dbDelete(ACCOUNT_DB_NAME, TODAY_POSITION_COL, flt)
# 移除非当日得委托
flt = {'accountid': accountid,
'order_date': {'$ne': trading_day}}
self.main_engine.dbDelete(ACCOUNT_DB_NAME, TODAY_ORDER_COL, flt)
# 移除非当日得持仓
flt = {'account_id': accountid,
'trading_day': {'$ne': trading_day}}
self.main_engine.dbDelete(ACCOUNT_DB_NAME, TODAY_STRATEGY_POS_COL, flt)
self.is_remove_pre_data = True
def update_order(self, event: Event):
"""更新当日记录"""
order = event.data
self.write_log(u'记录委托日志:{}'.format(order.__dict__))
if len(order.sysOrderID) == 0:
# 未有系统的委托编号,不做持久化
return
order_date = get_trading_date(datetime.now())
# 数据库需要提前建立多重索引
# db.today_orders.createIndex({'accountid':1,'vt_symbol':1,'sys_orderid':1,'order_date':1,'holder_id':1},{'name':'accountid_vtsymbol_sysorderid_order_date_holder_id','unique':true})
# db.today_orders.createIndex({'accountID':1})
fld = {'vt_symbol': order.vt_symbol,
'accountid': order.accountid,
'sys_orderid': order.sys_orderid,
'order_date': order_date,
'holder_id': order.holder_id}
data = copy.copy(order.__dict__)
data.update({'order_date': order_date})
data.update({'exchange': order.exchange.value})
data.update({'direction': order.direction.value})
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=TODAY_ORDER_COL, fld=fld, data=data)
# 数据库需要提前建立多重索引
# db.history_orders.createIndex({'accountid':1,'vt_symbol':1,'sys_orderid':1,'order_date':1,'holder_id':1},{'name':'history_accountid_vtsymbol_sysorderid_order_date_holder_id'})
# db.history_orders.createIndex({'accountid':1})
# 复制委托记录=》历史委托记录
if order.gatewayName in self.copy_history_orders:
history_data = copy.copy(data)
fld2 = {'vt_symbol': order.vt_symbol,
'accountid': order.accountid,
'sys_orderid': order.sys_orderid,
'order_date': order_date,
'holder_id': order.holder_id}
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=HISTORY_ORDER_COL, fld=fld2, data=history_data)
def update_trade(self, event: Event):
"""更新当日成交"""
trade = event.data
trade_date = get_trading_date(datetime.now())
fld = {'vt_symbol': trade.vt_symbol,
'accountid': trade.accountid,
'vt_tradeid': trade.vt_tradeid,
'trade_date': trade_date,
'holder_id': trade.holder_id}
# 提前创建索引
# db.today_trades.createIndex({'accountid':1,'vt_symbol':1,'vt_tradeid':1,'trade_date':1,'holder_id':1},{'name':'accountID_vtSymbol_vtTradeID_trade_date_holder_id','unique':true})
# db.today_trades.createIndex({'accountid':1})
data = copy.copy(trade.__dict__)
data.update({'trade_date': trade_date})
data.update({'exchange': trade.exchange.value})
data.update({'direction': trade.direction.value})
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=TODAY_TRADE_COL, fld=fld, data=data)
# db.history_trades.createIndex({'accountid':1,'vt_symbol':1,'vt_tradeid':1,'trade_date':1,'holder_id':1},{'name':'accountID_vtSymbol_vtTradeID_trade_date_holder_id'})
# db.history_trades.createIndex({'accountid':1})
# 复制成交记录=》历史成交记录
if trade.gateway_name in self.copy_history_trades:
history_trade = copy.copy(data)
fld2 = {'vt_symbol': trade.vt_symbol,
'accountid': trade.accountid,
'vt_tradeid': trade.vt_tradeid,
'trade_date': trade_date,
'holder_id': trade.holder_id}
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=HISTORY_TRADE_COL, fld=fld2, data=history_trade)
def update_position(self, event: Event):
"""更新当日持仓"""
pos = event.data
trade_date = get_trading_date(datetime.now())
# 不处理交易所返回得套利合约
if pos.symbol.startswith('SP') and '&' in pos.symbol and ' ' in pos.symbol:
return
fld = {'vt_symbol': pos.vt_symbol,
'direction': pos.direction.value,
'accountid': pos.accountid,
'trade_date': trade_date,
'holder_id': pos.holder_id}
# db.today_positions.createIndex({'accountid':1,'vt_symbol':1,'direction':1,'trade_date':1,'holder_id':1},{'name':'accountID_vtSymbol_direction_trade_date_holder_id'})
# db.today_positions.createIndex({'accountid':1})
data = pos.__dict__
data.update({'trade_date': trade_date})
data.update({'exchange': pos.exchange.value})
data.update({'direction': pos.direction.value})
# 补充 当前价格
try:
price = self.main_engine.get_price(pos.vt_symbol)
if price:
data.update({'cur_price': price})
except:
pass
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=TODAY_POSITION_COL, fld=fld, data=data)
def update_history_trade(self, event: Event):
"""更新历史查询记录"""
trade = event.data
trade_date = trade.time.split(' ')[0]
fld = {'vt_symbol': trade.vt_symbol,
'accountid': trade.accountid,
'vt_tradeID': trade.vt_tradeid,
'trade_date': trade_date,
'holder_id': trade.holder_id}
data = trade.__dict__
data.update({'trade_date': trade_date})
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=HISTORY_TRADE_COL, fld=fld, data=data)
self.update_begin_day(trade.gatewayName, HISTORY_TRADE_COL, trade_date)
def update_history_order(self, event: Event):
"""更新历史委托"""
order = event.data
order_date = order.time.split(' ')[0]
fld = {'vt_symbol': order.vt_symbol,
'accountid': order.accountid,
'sys_orderid': order.sys_orderid,
'order_date': order_date,
'holder_id': order.holder_id}
data = order.__dict__
data.update({'order_date': order_date})
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=HISTORY_ORDER_COL, fld=fld, data=data)
self.update_begin_day(order.gatewayName, HISTORY_ORDER_COL, order_date)
def update_funds_flow(self, event: Event):
"""更新历史资金流水"""
funds_flow = event.data
data = funds_flow.__dict__
fld = {'accountid': funds_flow.accountid,
'trade_date': funds_flow.trade_date,
'trade_amount': funds_flow.trade_amount,
'fund_remain': funds_flow.fund_remain,
'contract_id': funds_flow.contract_id,
'holder_id': funds_flow.holder_id}
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=FUNDS_FLOW_COL, fld=fld, data=data)
self.update_begin_day(funds_flow.gatewayName, HISTORY_ORDER_COL, funds_flow.trade_date)
def update_strategy_pos(self, event: Event):
data = event.data
dt = data.get('datetime')
pos_trading_day = get_trading_date(dt)
data.update({'trading_day': pos_trading_day})
accountid = data.get('accountid')
fld = {
'accountid': accountid,
'stratregy_group': data.get('straregy_group', '-'),
'strategy_name': data.get('strategy_name'),
'datetime': dt.strftime("%Y-%m-%d %H:%M:%S"),
'inited': True,
'trading': True
}
self.update_data(db_name=ACCOUNT_DB_NAME, col_name=TODAY_STRATEGY_POS_COL, fld=fld, data=data)
def process_gw_error(self, event: Event):
""" 处理gw的回报错误日志"""
try:
data = event.data
if data.errorMsg not in self.last_gw_error_msg:
d = {}
d.update({'gateway_name': data.gateway_name})
d.update({'msg': data.msg})
d.update({'additional_info': data.additional_info})
d.update({'log_time': datetime.now().strftime("%Y-%m-%d %H:%M:%S")})
d.update({'trading_day': get_trading_date()})
account_id = self.gw_name_acct_id.get(data.gateway_name, None)
if account_id:
d.update({'account_id': account_id})
fld = copy.copy(d)
self.update_data(db_name=ALERT_DB_NAME, col_name=GW_ERROR_COL_NAME, fld=fld, data=d)
self.last_gw_error_msg.append(data.errorMsg)
if len(self.last_gw_error_msg) >= 10:
self.last_gw_error_msg.pop(0)
except Exception as ex:
print(u'处理gw告警日志异常:{}'.format(str(ex)), file=sys.stderr)
def process_warning(self, event: Event):
"""处理日志"""
try:
data = event.data
if data.msg not in self.last_warning_msg:
d = {}
d.update({'gateway_name': data.gateway_name})
d.update({'msg': data.msg})
d.update({'additional_info': data.additional_info})
d.update({'log_time': datetime.now().strftime("%Y-%m-%d %H:%M:%S")})
d.update({'trading_day': get_trading_date()})
account_id = self.gw_name_acct_id.get(data.gatewayName, None)
if account_id:
d.update({'account_id': account_id})
fld = copy.copy(d)
self.update_data(db_name=ALERT_DB_NAME, col_name=WARNING_COL_NAME, fld=fld, data=d)
if len(self.last_warning_msg) >= 10:
self.last_warning_msg.pop(0)
except Exception as ex:
print(u'处理vn系统告警日志异常:{}'.format(str(ex)), file=sys.stderr)
def process_critical(self, event: Event):
"""处理日志"""
try:
data = event.data
if data.msg not in self.last_critical_msg:
d = {}
d.update({'gateway_name': data.gateway_name})
d.update({'msg': data.msg})
d.update({'additional_info': data.additional_info})
d.update({'log_time': datetime.now().strftime("%Y-%m-%d %H:%M:%S")})
d.update({'trading_day': get_trading_date()})
account_id = self.gw_name_acct_id.get(data.gateway_name, None)
if account_id:
d.update({'account_id': account_id})
fld = copy.copy(d)
self.update_data(db_name=ALERT_DB_NAME, col_name=CRITICAL_COL_NAME, fld=fld, data=d)
if len(self.last_critical_msg) >= 10:
self.last_critical_msg.pop(0)
except Exception as ex:
print(u'处理vn系统告警日志异常:{}'.format(str(ex)), file=sys.stderr)
# ----------------------------------------------------------------------
def update_data(self, db_name, col_name, fld, data):
"""更新或插入数据到数据库"""
self.queue.put((db_name, col_name, fld, data))
# ----------------------------------------------------------------------
def run(self):
"""运行插入线程"""
while self.active:
try:
db_name, col_name, fld, d = self.queue.get(block=True, timeout=1)
t1 = time()
self.mongo_db.db_update(db_name=db_name,
col_name=col_name,
data_dict=d,
filter_dict=fld,
upsert=True)
t2 = time()
execute_ms = (int(round(t2 * 1000))) - (int(round(t1 * 1000)))
if execute_ms > 200:
self.write_log(u'运行 {}.{} 更新 耗时:{}ms >200ms,数据:{}'
.format(db_name, col_name, execute_ms, d))
except Exception as ex:
pass
# ----------------------------------------------------------------------
def start(self):
"""启动"""
self.active = True
self.thread.start()
# ----------------------------------------------------------------------
def stop(self):
"""退出"""
if self.mongo_db:
self.mongo_db = None
if self.active:
self.active = False
self.thread.join()

View File

@ -883,6 +883,7 @@ class CtpTdApi(TdApi):
symbol=symbol,
exchange=exchange,
orderid=orderid,
sys_orderid=data.get('OrderSysID', ""),
type=ORDERTYPE_CTP2VT[data["OrderPriceType"]],
direction=DIRECTION_CTP2VT[data["Direction"]],
offset=OFFSET_CTP2VT[data["CombOffsetFlag"]],
@ -913,6 +914,7 @@ class CtpTdApi(TdApi):
symbol=symbol,
exchange=exchange,
orderid=orderid,
sys_orderid=data.get("OrderSysID", ""),
tradeid=data["TradeID"],
direction=DIRECTION_CTP2VT[data["Direction"]],
offset=OFFSET_CTP2VT[data["OffsetFlag"]],