2015-05-21 07:14:20 +00:00
|
|
|
|
# encoding: UTF-8
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
from datetime import datetime,timedelta
|
2015-09-16 07:34:00 +00:00
|
|
|
|
print u'StragegyEngine.py import datetime.datetime success'
|
|
|
|
|
|
|
|
|
|
from pymongo import MongoClient as Connection
|
|
|
|
|
print u'demoStrategy.py import pymongo.Connection success'
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
from pymongo.errors import *
|
2015-09-16 07:34:00 +00:00
|
|
|
|
print u'demoStrategy.py import pymongo.errors.* success'
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
from eventEngine import *
|
2015-09-16 07:34:00 +00:00
|
|
|
|
print u'demoStrategy.py import eventEngine.* success'
|
|
|
|
|
|
2015-10-17 16:23:24 +00:00
|
|
|
|
from vtConstant import *
|
|
|
|
|
|
2015-09-28 07:34:32 +00:00
|
|
|
|
import MySQLdb
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
2015-10-23 17:12:08 +00:00
|
|
|
|
import os
|
|
|
|
|
import sys
|
|
|
|
|
import cPickle
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
# 常量定义
|
|
|
|
|
OFFSET_OPEN = '0' # 开仓
|
|
|
|
|
OFFSET_CLOSE = '1' # 平仓
|
|
|
|
|
|
2015-11-04 16:28:41 +00:00
|
|
|
|
DIRECTION_BUY = '0' # 买入 多
|
|
|
|
|
DIRECTION_SELL = '1' # 卖出 空
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
PRICETYPE_LIMIT = '2' # 限价
|
|
|
|
|
|
2015-11-04 16:28:41 +00:00
|
|
|
|
# buy 买入开仓 开多 : DIRECTION_BUY = '0' OFFSET_OPEN = '0'
|
|
|
|
|
# sell 卖出平仓 平多 : DIRECTION_SELL = '1' OFFSET_CLOSE = '1'
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
2015-11-04 16:28:41 +00:00
|
|
|
|
# short 卖出开仓 开空 : DIRECTION_SELL = '1' OFFSET_OPEN = '0'
|
|
|
|
|
# cover 买入平仓 平空 : DIRECTION_BUY = '0' OFFSET_CLOSE = '1'
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
########################################################################
|
|
|
|
|
class Tick:
|
|
|
|
|
"""Tick数据对象"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self, symbol):
|
|
|
|
|
"""Constructor"""
|
|
|
|
|
self.symbol = symbol # 合约代码
|
|
|
|
|
|
|
|
|
|
self.openPrice = 0 # OHLC
|
|
|
|
|
self.highPrice = 0
|
|
|
|
|
self.lowPrice = 0
|
|
|
|
|
self.lastPrice = 0
|
|
|
|
|
|
|
|
|
|
self.volume = 0 # 成交量
|
|
|
|
|
self.openInterest = 0 # 持仓量
|
|
|
|
|
|
|
|
|
|
self.upperLimit = 0 # 涨停价
|
|
|
|
|
self.lowerLimit = 0 # 跌停价
|
|
|
|
|
|
|
|
|
|
self.time = '' # 更新时间和毫秒
|
|
|
|
|
self.ms= 0
|
|
|
|
|
|
|
|
|
|
self.bidPrice1 = 0 # 深度行情
|
|
|
|
|
self.bidPrice2 = 0
|
|
|
|
|
self.bidPrice3 = 0
|
|
|
|
|
self.bidPrice4 = 0
|
|
|
|
|
self.bidPrice5 = 0
|
|
|
|
|
|
|
|
|
|
self.askPrice1 = 0
|
|
|
|
|
self.askPrice2 = 0
|
|
|
|
|
self.askPrice3 = 0
|
|
|
|
|
self.askPrice4 = 0
|
|
|
|
|
self.askPrice5 = 0
|
|
|
|
|
|
|
|
|
|
self.bidVolume1 = 0
|
|
|
|
|
self.bidVolume2 = 0
|
|
|
|
|
self.bidVolume3 = 0
|
|
|
|
|
self.bidVolume4 = 0
|
|
|
|
|
self.bidVolume5 = 0
|
|
|
|
|
|
|
|
|
|
self.askVolume1 = 0
|
|
|
|
|
self.askVolume2 = 0
|
|
|
|
|
self.askVolume3 = 0
|
|
|
|
|
self.askVolume4 = 0
|
|
|
|
|
self.askVolume5 = 0
|
|
|
|
|
|
|
|
|
|
|
2015-10-17 16:23:24 +00:00
|
|
|
|
########################################################################
|
|
|
|
|
class Bar(object):
|
|
|
|
|
"""K线数据"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self):
|
|
|
|
|
"""Constructor"""
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-17 16:23:24 +00:00
|
|
|
|
self.symbol = EMPTY_STRING # 代码
|
|
|
|
|
#self.exchange = EMPTY_STRING # 交易所
|
|
|
|
|
|
|
|
|
|
self.open = EMPTY_FLOAT # OHLC
|
|
|
|
|
self.high = EMPTY_FLOAT
|
|
|
|
|
self.low = EMPTY_FLOAT
|
|
|
|
|
self.close = EMPTY_FLOAT
|
|
|
|
|
|
|
|
|
|
self.date = EMPTY_STRING # bar开始的时间,日期
|
|
|
|
|
self.time = EMPTY_STRING # 时间
|
|
|
|
|
self.datetime = None # python的datetime时间对象
|
|
|
|
|
|
|
|
|
|
self.volume = EMPTY_INT # 成交量
|
|
|
|
|
self.openInterest = EMPTY_INT # 持仓量
|
|
|
|
|
|
|
|
|
|
########################################################################
|
|
|
|
|
class EmaData(object):
|
|
|
|
|
"""数据"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self):
|
|
|
|
|
"""Constructor"""
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
self.symbol = EMPTY_STRING # 代码
|
|
|
|
|
self.fastEMA = EMPTY_FLOAT # 快速EMA的数值
|
|
|
|
|
self.slowEMA = EMPTY_FLOAT # 慢速EMA的数值
|
2015-10-17 16:23:24 +00:00
|
|
|
|
|
|
|
|
|
self.date = EMPTY_STRING # EMA开始的时间,日期
|
|
|
|
|
self.time = EMPTY_STRING # 时间
|
|
|
|
|
self.datetime = None # python的datetime时间对象
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
########################################################################
|
2015-05-28 05:52:59 +00:00
|
|
|
|
class Trade(object):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""成交数据对象"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self, symbol):
|
|
|
|
|
"""Constructor"""
|
|
|
|
|
self.symbol = symbol # 合约代码
|
|
|
|
|
|
|
|
|
|
self.orderRef = '' # 报单号
|
|
|
|
|
self.tradeID = '' # 成交编号
|
|
|
|
|
|
|
|
|
|
self.direction = None # 方向
|
|
|
|
|
self.offset = None # 开平
|
|
|
|
|
self.price = 0 # 成交价
|
|
|
|
|
self.volume = 0 # 成交量
|
2015-10-14 16:41:45 +00:00
|
|
|
|
|
|
|
|
|
self.tradeTime = '' # 成交时间
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
########################################################################
|
2015-05-28 05:52:59 +00:00
|
|
|
|
class Order(object):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""报单数据对象"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self, symbol):
|
|
|
|
|
"""Constructor"""
|
|
|
|
|
self.symbol = symbol # 合约代码
|
|
|
|
|
|
|
|
|
|
self.orderRef = '' # 报单编号
|
|
|
|
|
|
|
|
|
|
self.direction = None # 方向
|
|
|
|
|
self.offset = None # 开平
|
|
|
|
|
self.price = 0 # 委托价
|
|
|
|
|
self.volumeOriginal = 0 # 报单量
|
|
|
|
|
self.volumeTraded = 0 # 已成交数量
|
|
|
|
|
|
|
|
|
|
self.insertTime = '' # 报单时间
|
|
|
|
|
self.cancelTime = '' # 撤单时间
|
|
|
|
|
|
|
|
|
|
self.frontID = 0 # 前置机编号
|
|
|
|
|
self.sessionID = 0 # 会话编号
|
|
|
|
|
|
|
|
|
|
self.status = '' # 报单状态代码
|
|
|
|
|
|
2015-10-23 17:12:08 +00:00
|
|
|
|
self.preTradeID = '' # 上一成交单编号(用于平仓)
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
########################################################################
|
2015-05-28 05:52:59 +00:00
|
|
|
|
class StopOrder(object):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""
|
|
|
|
|
停止单对象
|
|
|
|
|
用于实现价格突破某一水平后自动追入
|
|
|
|
|
即通常的条件单和止损单
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self, symbol, direction, offset, price, volume, strategy):
|
|
|
|
|
"""Constructor"""
|
|
|
|
|
self.symbol = symbol
|
|
|
|
|
self.direction = direction
|
|
|
|
|
self.offset = offset
|
|
|
|
|
self.price = price
|
|
|
|
|
self.volume = volume
|
|
|
|
|
self.strategy = strategy
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
########################################################################
|
|
|
|
|
class StrategyEngine(object):
|
|
|
|
|
"""策略引擎"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-28 05:52:59 +00:00
|
|
|
|
def __init__(self, eventEngine, mainEngine, backtesting=False):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""Constructor"""
|
2015-10-14 16:41:45 +00:00
|
|
|
|
self.__eventEngine = eventEngine # 引用事件引擎
|
2015-10-17 16:23:24 +00:00
|
|
|
|
self.mainEngine = mainEngine # 主引擎,在回测中,为backtestingEngine,在交易中,为MainEngine
|
2015-05-28 05:52:59 +00:00
|
|
|
|
self.backtesting = backtesting # 是否在进行回测
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
# 获取代表今日的datetime
|
|
|
|
|
t = datetime.today()
|
|
|
|
|
self.today = t.replace(hour=0, minute=0, second=0, microsecond=0)
|
|
|
|
|
|
|
|
|
|
# 保存所有报单数据的字典
|
|
|
|
|
self.__dictOrder = {}
|
|
|
|
|
|
|
|
|
|
# 保存策略对象的字典
|
|
|
|
|
# key为策略名称
|
|
|
|
|
# value为策略对象
|
|
|
|
|
self.dictStrategy = {}
|
|
|
|
|
|
|
|
|
|
# 保存合约代码和策略对象映射关系的字典
|
|
|
|
|
# key为合约代码
|
|
|
|
|
# value为交易该合约的策略列表
|
|
|
|
|
self.__dictSymbolStrategy = {}
|
|
|
|
|
|
|
|
|
|
# 保存报单编号和策略对象映射关系的字典
|
|
|
|
|
# key为报单编号
|
|
|
|
|
# value为策略对象
|
|
|
|
|
self.__dictOrderRefStrategy = {}
|
|
|
|
|
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 保存合约代码和相关停止单(止损单)的字典
|
2015-05-21 07:14:20 +00:00
|
|
|
|
# key为合约代码
|
|
|
|
|
# value为该合约相关的停止单列表
|
|
|
|
|
self.__dictStopOrder = {}
|
|
|
|
|
|
2015-09-28 07:34:32 +00:00
|
|
|
|
# MongoDB/Mysql数据库相关
|
|
|
|
|
#self.__mongoConnected = False
|
|
|
|
|
|
|
|
|
|
self.__mysqlConnected = False
|
|
|
|
|
|
|
|
|
|
#self.__mongoConnection = None
|
|
|
|
|
self.__mysqlConnection = None
|
|
|
|
|
|
|
|
|
|
#self.__mongoTickDB = None
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
# 调用函数
|
2015-09-28 07:34:32 +00:00
|
|
|
|
#self.__connectMongo()
|
|
|
|
|
self.__connectMysql()
|
2015-05-21 07:14:20 +00:00
|
|
|
|
self.__registerEvent()
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def createStrategy(self, strategyName, strategySymbol, strategyClass, strategySetting):
|
2015-10-14 16:41:45 +00:00
|
|
|
|
"""创建策略(实例化)"""
|
2015-05-21 07:14:20 +00:00
|
|
|
|
strategy = strategyClass(strategyName, strategySymbol, self)
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
|
|
|
|
self.writeLog(u"创建策略:{0}".format(strategyName))
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
self.dictStrategy[strategyName] = strategy
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-26 08:05:04 +00:00
|
|
|
|
strategy.loadSetting(strategySetting)
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-26 08:05:04 +00:00
|
|
|
|
# 订阅合约行情,注意这里因为是CTP,所以ExchangeID可以忽略
|
|
|
|
|
self.mainEngine.subscribe(strategySymbol, None)
|
2015-10-14 16:41:45 +00:00
|
|
|
|
|
2015-05-26 08:05:04 +00:00
|
|
|
|
# 注册策略监听
|
|
|
|
|
self.registerStrategy(strategySymbol, strategy)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
2015-09-28 07:34:32 +00:00
|
|
|
|
#def __connectMongo(self):
|
|
|
|
|
# """连接MongoDB数据库"""
|
|
|
|
|
# try:
|
|
|
|
|
# self.__mongoConnection = Connection()
|
|
|
|
|
# self.__mongoConnected = True
|
|
|
|
|
# self.__mongoTickDB = self.__mongoConnection['TickDB']
|
|
|
|
|
# self.writeLog(u'策略引擎连接MongoDB成功')
|
|
|
|
|
# except ConnectionFailure:
|
|
|
|
|
# self.writeLog(u'策略引擎连接MongoDB失败')
|
|
|
|
|
|
|
|
|
|
#-------------------------------------------#---------------------------
|
|
|
|
|
# def __recordTickToMongo(self, data):
|
|
|
|
|
# """将Tick数据插入到MongoDB中"""
|
|
|
|
|
# if self.__mongoConnected:
|
|
|
|
|
# symbol = data['InstrumentID']
|
|
|
|
|
# data['date'] = self.today
|
|
|
|
|
# self.__mongoTickDB[symbol].insert(data)
|
2015-10-14 16:41:45 +00:00
|
|
|
|
#
|
|
|
|
|
|
2015-09-28 07:34:32 +00:00
|
|
|
|
# #----------------------------------------------------------------------
|
|
|
|
|
# def loadTickFromMongo(self, symbol, startDate, endDate=None):
|
|
|
|
|
# """从MongoDB中读取Tick数据"""
|
|
|
|
|
# if self.__mongoConnected:
|
|
|
|
|
# collection = self.__mongoTickDB[symbol]
|
|
|
|
|
#
|
|
|
|
|
# # 如果输入了读取TICK的最后日期
|
|
|
|
|
# if endDate:
|
|
|
|
|
# cx = collection.find({'date':{'$gte':startDate, '$lte':endDate}})
|
|
|
|
|
# else:
|
|
|
|
|
# cx = collection.find({'date':{'$gte':startDate}})
|
|
|
|
|
# return cx
|
|
|
|
|
# else:
|
|
|
|
|
# return None
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
2015-09-28 07:34:32 +00:00
|
|
|
|
def __connectMysql(self):
|
|
|
|
|
"""连接MysqlDB"""
|
|
|
|
|
try:
|
|
|
|
|
self.__mysqlConnection = MySQLdb.connect(host='vnpy.cloudapp.net', user='stockcn', passwd='7uhb*IJN', db='stockcn', port=3306)
|
|
|
|
|
self.__mysqlConnected = True
|
|
|
|
|
self.writeLog(u'策略引擎连接MysqlDB成功')
|
|
|
|
|
except ConnectionFailure:
|
|
|
|
|
self.writeLog(u'策略引擎连接MysqlDB失败')
|
2015-10-22 02:40:59 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-09-28 07:34:32 +00:00
|
|
|
|
def __recordTickToMysql(self, data):
|
|
|
|
|
"""将Tick数据插入到MysqlDB中"""
|
|
|
|
|
#if self.__mongoConnected:
|
|
|
|
|
# symbol = data['InstrumentID']
|
|
|
|
|
# data['date'] = self.today
|
|
|
|
|
# self.__mongoTickDB[symbol].insert(data)
|
2015-10-14 16:41:45 +00:00
|
|
|
|
pass
|
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __executeMysql(self, sql):
|
|
|
|
|
"""执行mysql语句"""
|
|
|
|
|
if not self.__mysqlConnected:
|
|
|
|
|
self.__connectMysql()
|
|
|
|
|
|
|
|
|
|
cur = self.__mysqlConnection.cursor(MySQLdb.cursors.DictCursor)
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
cur.execute(sql)
|
|
|
|
|
self.__mysqlConnection.commit()
|
|
|
|
|
|
|
|
|
|
except Exception, e:
|
|
|
|
|
print e
|
|
|
|
|
print sql
|
|
|
|
|
|
|
|
|
|
self.__connectMysql()
|
|
|
|
|
cur = self.__mysqlConnection.cursor(MySQLdb.cursors.DictCursor)
|
|
|
|
|
cur.execute(sql)
|
|
|
|
|
self.__mysqlConnection.commit()
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-09-28 07:34:32 +00:00
|
|
|
|
def loadTickFromMysql(self, symbol, startDate, endDate=None):
|
|
|
|
|
"""从MysqlDB中读取Tick数据"""
|
|
|
|
|
if self.__mysqlConnected:
|
|
|
|
|
|
|
|
|
|
#获取指针
|
|
|
|
|
cur = self.__mysqlConnection.cursor(MySQLdb.cursors.DictCursor)
|
|
|
|
|
|
2015-05-28 05:52:59 +00:00
|
|
|
|
if endDate:
|
2015-09-28 07:34:32 +00:00
|
|
|
|
#指定开始与结束日期
|
|
|
|
|
sqlstring = ' select \'{0}\' as InstrumentID, str_to_date(concat(ndate,\' \', ntime),' \
|
|
|
|
|
'\'%Y-%m-%d %H:%i:%s\') as UpdateTime,price as LastPrice,vol as Volume,' \
|
|
|
|
|
'position_vol as OpenInterest,bid1_price as BidPrice1,bid1_vol as BidVolume1, ' \
|
|
|
|
|
'sell1_price as AskPrice1, sell1_vol as AskVolume1 from TB_{0}MI ' \
|
|
|
|
|
'where ndate between cast(\'{1}\' as date) and cast(\'{2}\' as date)'.format(symbol, startDate, endDate)
|
|
|
|
|
|
|
|
|
|
elif startDate:
|
|
|
|
|
#指定开始日期
|
|
|
|
|
sqlstring = ' select \'{0}\' as InstrumentID,str_to_date(concat(ndate,\' \', ntime),' \
|
|
|
|
|
'\'%Y-%m-%d %H:%i:%s\') as UpdateTime,price as LastPrice,vol as Volume,' \
|
|
|
|
|
'position_vol as OpenInterest,bid1_price as BidPrice1,bid1_vol as BidVolume1, ' \
|
|
|
|
|
'sell1_price as AskPrice1, sell1_vol as AskVolume1 from TB__{0}MI ' \
|
|
|
|
|
'where ndate > cast(\'{1}\' as date)'.format( symbol, startDate)
|
|
|
|
|
|
2015-05-28 05:52:59 +00:00
|
|
|
|
else:
|
2015-09-28 07:34:32 +00:00
|
|
|
|
#没有指定,所有日期数据
|
|
|
|
|
sqlstring =' select \'{0}\' as InstrumentID,str_to_date(concat(ndate,\' \', ntime),' \
|
|
|
|
|
'\'%Y-%m-%d %H:%i:%s\') as UpdateTime,price as LastPrice,vol as Volume,' \
|
|
|
|
|
'position_vol as OpenInterest,bid1_price as BidPrice1,bid1_vol as BidVolume1, ' \
|
|
|
|
|
'sell1_price as AskPrice1, sell1_vol as AskVolume1 from TB__{0}MI '.format(symbol)
|
|
|
|
|
|
2015-10-17 16:23:24 +00:00
|
|
|
|
print sqlstring
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
|
|
|
|
count = cur.execute(sqlstring)
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
# cx = cur.fetchall()
|
|
|
|
|
fetch_counts = 0
|
|
|
|
|
|
2015-10-20 14:54:57 +00:00
|
|
|
|
cx = None
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
fetch_size = 1000
|
|
|
|
|
|
|
|
|
|
while True:
|
|
|
|
|
results = cur.fetchmany(fetch_size)
|
|
|
|
|
|
|
|
|
|
if not results:
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
if fetch_counts == 0:
|
|
|
|
|
cx = results
|
|
|
|
|
else:
|
|
|
|
|
cx = cx + results
|
|
|
|
|
|
|
|
|
|
fetch_counts = fetch_counts+fetch_size
|
|
|
|
|
|
|
|
|
|
print u'历史TICK数据载入{0}条'.format(fetch_counts)
|
|
|
|
|
|
|
|
|
|
self.writeLog(u'历史TICK数据载入完成,{1}~{2},共{0}条'.format(count,startDate,endDate))
|
|
|
|
|
|
|
|
|
|
print u'策略引擎:历史TICK数据载入完成,{1}~{2},共{0}条'.format(count,startDate,endDate)
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return cx
|
|
|
|
|
else:
|
2015-09-28 07:34:32 +00:00
|
|
|
|
return None
|
|
|
|
|
|
2015-11-04 16:28:41 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def loadBarFromMysql(self, symbol, startDate, endDate, barType='M5'):
|
|
|
|
|
"""从MysqlDB中读取Bar数据
|
|
|
|
|
startDate,(包含)开始日期
|
|
|
|
|
endDate, (包含)结束日期
|
|
|
|
|
barType ='M1' K线类型,1分钟线
|
|
|
|
|
barType ='M5' K线类型,5分钟线
|
|
|
|
|
barType = 'D1' K线类型,日线
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
if self.__mysqlConnected:
|
|
|
|
|
|
|
|
|
|
#获取指针
|
|
|
|
|
cur = self.__mysqlConnection.cursor(MySQLdb.cursors.DictCursor)
|
|
|
|
|
|
|
|
|
|
if endDate:
|
|
|
|
|
# 指定开始与结束日期
|
|
|
|
|
sqlstring = 'select open,high,low,close,volume,date,time,datetime from TB_{0}{1} ' \
|
|
|
|
|
'where date between cast(\'{2}\' as date) and cast(\'{3}\' as date) ' \
|
|
|
|
|
'order by datetime'.format(symbol, barType, startDate, endDate)
|
|
|
|
|
|
|
|
|
|
elif startDate:
|
|
|
|
|
# 指定开始日期
|
|
|
|
|
sqlstring = 'select open,high,low,close,volume,date,time,datetime from TB_{0}{1} ' \
|
|
|
|
|
'where date > cast(\'{2}\' as date) order by datetime'.format(symbol,
|
|
|
|
|
barType, startDate)
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
# 没有指定,所有日期数据
|
|
|
|
|
sqlstring = 'select open,high,low,close,volume,date,time,datetime from TB_{0}{1} ' \
|
|
|
|
|
' order by datetime'.format(symbol, barType)
|
|
|
|
|
|
|
|
|
|
print sqlstring
|
|
|
|
|
|
|
|
|
|
count = cur.execute(sqlstring)
|
|
|
|
|
|
|
|
|
|
# cx = cur.fetchall()
|
|
|
|
|
fetch_counts = 0
|
|
|
|
|
|
|
|
|
|
cx = None
|
|
|
|
|
|
|
|
|
|
fetch_size = 1000
|
|
|
|
|
|
|
|
|
|
while True:
|
|
|
|
|
results = cur.fetchmany(fetch_size)
|
|
|
|
|
|
|
|
|
|
if not results:
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
if fetch_counts == 0:
|
|
|
|
|
cx = results
|
|
|
|
|
else:
|
|
|
|
|
cx = cx + results
|
|
|
|
|
|
|
|
|
|
fetch_counts = fetch_counts+fetch_size
|
|
|
|
|
|
|
|
|
|
print u'历史{0}Bar数据载入{1}条'.format(barType, fetch_counts)
|
|
|
|
|
|
|
|
|
|
self.writeLog(u'历史{0}Bar数据载入完成,{1}~{2},共{3}条'.format(barType,startDate, endDate, count))
|
|
|
|
|
|
|
|
|
|
print u'策略引擎:历史{0}Bar数据载入完成,{1}~{2},共{3}条'.format(barType,startDate, endDate, count)
|
|
|
|
|
|
|
|
|
|
return cx
|
|
|
|
|
else:
|
|
|
|
|
return None
|
|
|
|
|
|
2015-09-28 07:34:32 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def getMysqlDeltaDate(self,symbol, startDate, decreaseDays):
|
2015-10-14 16:41:45 +00:00
|
|
|
|
"""从mysql获取交易日天数差"""
|
2015-09-28 07:34:32 +00:00
|
|
|
|
try:
|
|
|
|
|
if self.__mysqlConnected:
|
|
|
|
|
|
|
|
|
|
#获取指针
|
|
|
|
|
cur = self.__mysqlConnection.cursor()
|
|
|
|
|
|
|
|
|
|
sqlstring='select distinct ndate from TB_{0}MI where ndate < ' \
|
|
|
|
|
'cast(\'{1}\' as date) order by ndate desc limit {2},1'.format(symbol, startDate, decreaseDays-1)
|
|
|
|
|
|
|
|
|
|
self.writeLog(sqlstring)
|
|
|
|
|
|
|
|
|
|
count = cur.execute(sqlstring)
|
|
|
|
|
|
|
|
|
|
if count > 0:
|
|
|
|
|
|
|
|
|
|
result = cur.fetchone()
|
|
|
|
|
|
|
|
|
|
return result[0]
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
self.writeLog(u'MysqlDB没有查询结果,请检查日期')
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
self.writeLog(u'MysqlDB未连接,请检查')
|
|
|
|
|
|
|
|
|
|
except MySQLdb.Error, e:
|
2015-10-14 16:41:45 +00:00
|
|
|
|
self.writeLog(u'MysqlDB载入数据失败,请检查.Error %s'.format(e))
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
|
|
|
|
td = timedelta(days=3)
|
|
|
|
|
|
|
|
|
|
return startDate-td;
|
2015-10-19 08:42:17 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def saveBarToMysql(self,id, barList):
|
|
|
|
|
"""
|
|
|
|
|
保存K线数据到数据库
|
|
|
|
|
id, 回测ID
|
|
|
|
|
barList, 对象为Bar的列表
|
|
|
|
|
"""
|
2015-10-23 17:12:08 +00:00
|
|
|
|
|
|
|
|
|
# 保存本地pickle文件
|
2015-10-24 16:56:43 +00:00
|
|
|
|
resultPath=os.getcwd()+'/result'
|
2015-10-23 17:12:08 +00:00
|
|
|
|
|
|
|
|
|
if not os.path.isdir(resultPath):
|
|
|
|
|
os.mkdir(resultPath)
|
|
|
|
|
|
2015-10-24 16:56:43 +00:00
|
|
|
|
resultFile = u'{0}/{1}_Bar.pickle'.format(resultPath,id)
|
2015-10-23 17:12:08 +00:00
|
|
|
|
|
|
|
|
|
cache= open(resultFile, mode='w')
|
|
|
|
|
|
|
|
|
|
cPickle.dump(barList,cache)
|
|
|
|
|
|
|
|
|
|
cache.close()
|
|
|
|
|
|
|
|
|
|
# 保存数据库
|
|
|
|
|
|
|
|
|
|
self.__connectMysql()
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
if self.__mysqlConnected:
|
2015-10-24 00:01:02 +00:00
|
|
|
|
sql = 'insert into BackTest.TB_Bar ' \
|
|
|
|
|
'(Id, symbol, open,high, low,close,date,time,datetime, volume, openInterest) ' \
|
|
|
|
|
'values '
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
values = ''
|
|
|
|
|
|
|
|
|
|
print u'共{0}条Bar记录.'.format(len(barList))
|
|
|
|
|
|
2015-10-20 14:54:57 +00:00
|
|
|
|
if len(barList) == 0:
|
|
|
|
|
return
|
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
counts = 0
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
|
|
|
|
for bar in barList:
|
|
|
|
|
|
|
|
|
|
if len(values) > 0:
|
|
|
|
|
values = values + ','
|
|
|
|
|
|
|
|
|
|
values = values + '(\'{0}\',\'{1}\',{2},{3},{4},{5},\'{6}\',\'{7}\',\'{8}\',{9},{10})'.format(
|
|
|
|
|
id,
|
|
|
|
|
bar.symbol,
|
|
|
|
|
bar.open,
|
|
|
|
|
bar.high,
|
|
|
|
|
bar.low,
|
|
|
|
|
bar.close,
|
|
|
|
|
bar.date,
|
|
|
|
|
bar.time,
|
|
|
|
|
bar.datetime.strftime('%Y-%m-%d %H:%M:%S'),
|
|
|
|
|
bar.volume,
|
|
|
|
|
bar.openInterest)
|
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
if counts >= 3600:
|
|
|
|
|
|
|
|
|
|
self.__executeMysql(sql+values)
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
print u'写入{0}条Bar记录'.format(counts)
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
counts = 0
|
|
|
|
|
values = ''
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
|
|
|
|
else:
|
2015-10-22 02:40:59 +00:00
|
|
|
|
counts = counts + 1
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
if counts > 0:
|
|
|
|
|
|
|
|
|
|
self.__executeMysql(sql+values)
|
|
|
|
|
print u'写入{0}条Bar记录'.format(counts)
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def saveEmaToMysql(self, id, emaList):
|
|
|
|
|
"""
|
|
|
|
|
保存EMA到数据库
|
|
|
|
|
id,回测的编号
|
|
|
|
|
"""
|
2015-10-23 17:12:08 +00:00
|
|
|
|
|
|
|
|
|
# 保存本地pickle文件
|
2015-10-24 16:56:43 +00:00
|
|
|
|
resultPath=os.getcwd()+'/result'
|
2015-10-23 17:12:08 +00:00
|
|
|
|
|
|
|
|
|
if not os.path.isdir(resultPath):
|
|
|
|
|
os.mkdir(resultPath)
|
|
|
|
|
|
2015-10-24 16:56:43 +00:00
|
|
|
|
resultFile = u'{0}/{1}_Ema.pickle'.format(resultPath, id)
|
2015-10-23 17:12:08 +00:00
|
|
|
|
cache= open(resultFile, mode='w')
|
|
|
|
|
cPickle.dump(emaList,cache)
|
|
|
|
|
cache.close()
|
|
|
|
|
|
|
|
|
|
self.__connectMysql()
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
if self.__mysqlConnected:
|
2015-10-24 00:01:02 +00:00
|
|
|
|
sql = 'insert into BackTest.TB_Ema ' \
|
|
|
|
|
'(Id, symbol ,fastEMA,slowEMA ,date ,time ,datetime) ' \
|
|
|
|
|
'values '
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
values = ''
|
|
|
|
|
|
|
|
|
|
print u'共{0}条EMA记录.'.format(len(emaList))
|
|
|
|
|
|
2015-10-20 14:54:57 +00:00
|
|
|
|
if len(emaList) == 0:
|
|
|
|
|
return
|
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
counts = 0
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
|
|
|
|
for ema in emaList:
|
|
|
|
|
|
|
|
|
|
if len(values) > 0:
|
|
|
|
|
values = values + ','
|
|
|
|
|
|
|
|
|
|
values = values + '(\'{0}\',\'{1}\',{2},{3},\'{4}\',\'{5}\',\'{6}\')'.format(
|
|
|
|
|
id,
|
|
|
|
|
ema.symbol,
|
|
|
|
|
ema.fastEMA,
|
|
|
|
|
ema.slowEMA,
|
|
|
|
|
ema.date,
|
|
|
|
|
ema.time,
|
|
|
|
|
ema.datetime.strftime('%Y-%m-%d %H:%M:%S'))
|
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
if counts >= 3600:
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
self.__executeMysql(sql+values)
|
|
|
|
|
print u'写入{0}条EMA记录'.format(counts)
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
counts = 0
|
|
|
|
|
values = ''
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
|
|
|
|
else:
|
2015-10-22 02:40:59 +00:00
|
|
|
|
counts = counts + 1
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
if counts > 0:
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-10-22 02:40:59 +00:00
|
|
|
|
self.__executeMysql(sql+values)
|
|
|
|
|
print u'写入{0}条EMA记录'.format(counts)
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-28 05:52:59 +00:00
|
|
|
|
def updateMarketData(self, event):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""行情更新"""
|
|
|
|
|
data = event.dict_['data']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
symbol = data['InstrumentID']
|
|
|
|
|
|
|
|
|
|
# 检查是否存在交易该合约的策略
|
|
|
|
|
if symbol in self.__dictSymbolStrategy:
|
|
|
|
|
# 创建TICK数据对象并更新数据
|
|
|
|
|
tick = Tick(symbol)
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# tick.openPrice = data['OpenPrice']
|
|
|
|
|
# tick.highPrice = data['HighestPrice']
|
|
|
|
|
# tick.lowPrice = data['LowestPrice']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
tick.lastPrice = float(data['LastPrice'])
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
tick.volume = data['Volume']
|
|
|
|
|
tick.openInterest = data['OpenInterest']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# tick.upperLimit = data['UpperLimitPrice']
|
|
|
|
|
# tick.lowerLimit = data['LowerLimitPrice']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
tick.time = data['UpdateTime']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
|
|
|
|
tick.bidPrice1 = float(data['BidPrice1'])
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# tick.bidPrice2 = data['BidPrice2']
|
|
|
|
|
# tick.bidPrice3 = data['BidPrice3']
|
|
|
|
|
# tick.bidPrice4 = data['BidPrice4']
|
|
|
|
|
# tick.bidPrice5 = data['BidPrice5']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
|
|
|
|
tick.askPrice1 = float(data['AskPrice1'])
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# tick.askPrice2 = data['AskPrice2']
|
|
|
|
|
# tick.askPrice3 = data['AskPrice3']
|
|
|
|
|
# tick.askPrice4 = data['AskPrice4']
|
|
|
|
|
# tick.askPrice5 = data['AskPrice5']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
tick.bidVolume1 = data['BidVolume1']
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# tick.bidVolume2 = data['BidVolume2']
|
|
|
|
|
# tick.bidVolume3 = data['BidVolume3']
|
|
|
|
|
# tick.bidVolume4 = data['BidVolume4']
|
|
|
|
|
# tick.bidVolume5 = data['BidVolume5']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
tick.askVolume1 = data['AskVolume1']
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# tick.askVolume2 = data['AskVolume2']
|
|
|
|
|
# tick.askVolume3 = data['AskVolume3']
|
|
|
|
|
# tick.askVolume4 = data['AskVolume4']
|
|
|
|
|
# tick.askVolume5 = data['AskVolume5']
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
# 首先检查停止单是否需要发出
|
|
|
|
|
self.__processStopOrder(tick)
|
|
|
|
|
|
|
|
|
|
# 将该TICK数据推送给每个策略
|
|
|
|
|
for strategy in self.__dictSymbolStrategy[symbol]:
|
|
|
|
|
strategy.onTick(tick)
|
|
|
|
|
|
2015-09-28 07:34:32 +00:00
|
|
|
|
# 将数据插入MongoDB/Mysql数据库,实盘建议另开程序记录TICK数据
|
2015-05-28 05:52:59 +00:00
|
|
|
|
if not self.backtesting:
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# self.__recordTickToMongo(data)
|
2015-09-28 07:34:32 +00:00
|
|
|
|
self.__recordTickToMysql(data)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __processStopOrder(self, tick):
|
|
|
|
|
"""处理停止单"""
|
|
|
|
|
symbol = tick.symbol
|
|
|
|
|
lastPrice = tick.lastPrice
|
|
|
|
|
upperLimit = tick.upperLimit
|
|
|
|
|
lowerLimit = tick.lowerLimit
|
|
|
|
|
|
|
|
|
|
# 如果当前有该合约上的止损单
|
|
|
|
|
if symbol in self.__dictStopOrder:
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
# 获取止损单列表
|
|
|
|
|
listSO = self.__dictStopOrder[symbol] # SO:stop order
|
|
|
|
|
|
|
|
|
|
# 准备一个空的已发止损单列表
|
|
|
|
|
listSent = []
|
|
|
|
|
|
|
|
|
|
for so in listSO:
|
|
|
|
|
# 如果是买入停止单,且最新成交价大于停止触发价
|
|
|
|
|
if so.direction == DIRECTION_BUY and lastPrice >= so.price:
|
|
|
|
|
# 以当日涨停价发出限价单买入
|
2015-10-10 09:08:13 +00:00
|
|
|
|
print u'sendOrder({0},{1},{2},{3},{4}'.format(symbol,'Direction_Buy',so.offset,upperLimit,so.volume)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
ref = self.sendOrder(symbol, DIRECTION_BUY, so.offset,
|
2015-10-17 16:23:24 +00:00
|
|
|
|
upperLimit, so.volume, tick.time, so.strategy)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
# 触发策略的止损单发出更新
|
|
|
|
|
so.strategy.onStopOrder(ref)
|
|
|
|
|
|
|
|
|
|
# 将该止损单对象保存到已发送列表中
|
|
|
|
|
listSent.append(so)
|
|
|
|
|
|
|
|
|
|
# 如果是卖出停止单,且最新成交价小于停止触发价
|
|
|
|
|
elif so.direction == DIRECTION_SELL and lastPrice <= so.price:
|
2015-10-10 09:08:13 +00:00
|
|
|
|
print u'sendOrder({0},{1},{2},{3},{4}'.format(symbol,'Direction_Sell',so.offset,upperLimit,so.volume)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
ref = self.sendOrder(symbol, DIRECTION_SELL, so.offset,
|
2015-10-17 16:23:24 +00:00
|
|
|
|
lowerLimit, so.volume,tick.time, so.strategy)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
so.strategy.onStopOrder(ref)
|
|
|
|
|
|
|
|
|
|
listSent.append(so)
|
|
|
|
|
|
|
|
|
|
# 从停止单列表中移除已经发单的停止单对象
|
|
|
|
|
if listSent:
|
|
|
|
|
for so in listSent:
|
|
|
|
|
listSO.remove(so)
|
|
|
|
|
|
|
|
|
|
# 检查停止单列表是否为空,若为空,则从停止单字典中移除该合约代码
|
|
|
|
|
if not listSO:
|
|
|
|
|
del self.__dictStopOrder[symbol]
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-28 05:52:59 +00:00
|
|
|
|
def updateOrder(self, event):
|
2015-10-10 09:08:13 +00:00
|
|
|
|
"""事件响应:报单更新"""
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
data = event.dict_['data']
|
|
|
|
|
orderRef = data['OrderRef']
|
|
|
|
|
|
|
|
|
|
# 检查是否存在监听该报单的策略
|
|
|
|
|
if orderRef in self.__dictOrderRefStrategy:
|
|
|
|
|
|
|
|
|
|
# 创建Order数据对象
|
|
|
|
|
order = Order(data['InstrumentID'])
|
|
|
|
|
|
|
|
|
|
order.orderRef = data['OrderRef']
|
|
|
|
|
order.direction = data['Direction']
|
2015-05-26 08:05:04 +00:00
|
|
|
|
order.offset = data['CombOffsetFlag']
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
order.price = data['LimitPrice']
|
|
|
|
|
order.volumeOriginal = data['VolumeTotalOriginal']
|
|
|
|
|
order.volumeTraded = data['VolumeTraded']
|
|
|
|
|
order.insertTime = data['InsertTime']
|
|
|
|
|
order.cancelTime = data['CancelTime']
|
|
|
|
|
order.frontID = data['FrontID']
|
|
|
|
|
order.sessionID = data['SessionID']
|
|
|
|
|
|
|
|
|
|
order.status = data['OrderStatus']
|
|
|
|
|
|
|
|
|
|
# 推送给策略
|
|
|
|
|
strategy = self.__dictOrderRefStrategy[orderRef]
|
|
|
|
|
strategy.onOrder(order)
|
|
|
|
|
|
|
|
|
|
# 记录该Order的数据
|
|
|
|
|
self.__dictOrder[orderRef] = data
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-28 05:52:59 +00:00
|
|
|
|
def updateTrade(self, event):
|
2015-10-10 09:08:13 +00:00
|
|
|
|
"""事件响应:成交更新"""
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
data = event.dict_['data']
|
|
|
|
|
orderRef = data['OrderRef']
|
|
|
|
|
|
|
|
|
|
if orderRef in self.__dictOrderRefStrategy:
|
|
|
|
|
|
|
|
|
|
# 创建Trade数据对象
|
2015-10-17 16:23:24 +00:00
|
|
|
|
trade = Trade(data['InstrumentID']) # 合约代码
|
|
|
|
|
|
|
|
|
|
trade.orderRef = orderRef # 报单号
|
|
|
|
|
trade.tradeID = data['TradeID'] # 成交编号
|
|
|
|
|
trade.direction = data['Direction'] # 方向
|
|
|
|
|
trade.offset = data['OffsetFlag'] # 开平
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
2015-10-17 16:23:24 +00:00
|
|
|
|
trade.price = data['Price'] # 成交价
|
|
|
|
|
trade.volume = data['Volume'] # 成交量
|
|
|
|
|
|
|
|
|
|
trade.tradeTime = data['TradeTime'] # 成交时间
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
# 推送给策略
|
|
|
|
|
strategy = self.__dictOrderRefStrategy[orderRef]
|
2015-10-10 09:08:13 +00:00
|
|
|
|
strategy.onTrade(trade)
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-10-17 16:23:24 +00:00
|
|
|
|
def sendOrder(self, symbol, direction, offset, price, volume, ordertime, strategy):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""
|
|
|
|
|
发单(仅允许限价单)
|
|
|
|
|
symbol:合约代码
|
|
|
|
|
direction:方向,DIRECTION_BUY/DIRECTION_SELL
|
|
|
|
|
offset:开平,OFFSET_OPEN/OFFSET_CLOSE
|
|
|
|
|
price:下单价格
|
|
|
|
|
volume:下单手数
|
2015-10-17 16:23:24 +00:00
|
|
|
|
ordertime:下单时间(回归测试使用)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
strategy:策略对象
|
|
|
|
|
"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
contract = self.mainEngine.selectInstrument(symbol)
|
|
|
|
|
|
|
|
|
|
if contract:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 调用主引擎的发单函数
|
2015-05-21 07:14:20 +00:00
|
|
|
|
ref = self.mainEngine.sendOrder(symbol,
|
|
|
|
|
contract['ExchangeID'],
|
|
|
|
|
price,
|
|
|
|
|
PRICETYPE_LIMIT,
|
|
|
|
|
volume,
|
|
|
|
|
direction,
|
2015-10-14 16:41:45 +00:00
|
|
|
|
offset,
|
2015-10-17 16:23:24 +00:00
|
|
|
|
ordertime)
|
|
|
|
|
|
|
|
|
|
# 添加报单编号及其映射的策略
|
2015-05-21 07:14:20 +00:00
|
|
|
|
self.__dictOrderRefStrategy[ref] = strategy
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return ref
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def cancelOrder(self, orderRef):
|
|
|
|
|
"""
|
|
|
|
|
撤单
|
|
|
|
|
"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
print u'strategyEngine.py cancelOrder() begin.'
|
2015-05-21 07:14:20 +00:00
|
|
|
|
order = self.__dictOrder[orderRef]
|
|
|
|
|
symbol = order['InstrumentID']
|
|
|
|
|
contract = self.mainEngine.selectInstrument(symbol)
|
|
|
|
|
|
|
|
|
|
if contract:
|
2015-10-14 16:41:45 +00:00
|
|
|
|
#调用主引擎的撤单函数
|
2015-05-21 07:14:20 +00:00
|
|
|
|
self.mainEngine.cancelOrder(symbol,
|
|
|
|
|
contract['ExchangeID'],
|
|
|
|
|
orderRef,
|
|
|
|
|
order['FrontID'],
|
|
|
|
|
order['SessionID'])
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
|
|
|
|
print u'strategyEngine.py cancelOrder() end.'
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __registerEvent(self):
|
|
|
|
|
"""注册事件监听"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# 注册 订阅行情数据更新事件
|
2015-05-28 05:52:59 +00:00
|
|
|
|
self.__eventEngine.register(EVENT_MARKETDATA, self.updateMarketData)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# 注册 订阅订单更新事件
|
2015-05-28 05:52:59 +00:00
|
|
|
|
self.__eventEngine.register(EVENT_ORDER, self.updateOrder)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
# 注册 订阅交易响应事件
|
2015-05-28 05:52:59 +00:00
|
|
|
|
self.__eventEngine.register(EVENT_TRADE ,self.updateTrade)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def writeLog(self, log):
|
|
|
|
|
"""写日志"""
|
|
|
|
|
event = Event(type_=EVENT_LOG)
|
|
|
|
|
event.dict_['log'] = log
|
|
|
|
|
self.__eventEngine.put(event)
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def registerStrategy(self, symbol, strategy):
|
|
|
|
|
"""注册策略对合约TICK数据的监听"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
print u'strategyEngine.py registerStrategy() begin.'
|
2015-05-21 07:14:20 +00:00
|
|
|
|
# 尝试获取监听该合约代码的策略的列表,若无则创建
|
|
|
|
|
try:
|
|
|
|
|
listStrategy = self.__dictSymbolStrategy[symbol]
|
|
|
|
|
except KeyError:
|
|
|
|
|
listStrategy = []
|
|
|
|
|
self.__dictSymbolStrategy[symbol] = listStrategy
|
|
|
|
|
|
|
|
|
|
# 防止重复注册
|
|
|
|
|
if strategy not in listStrategy:
|
|
|
|
|
listStrategy.append(strategy)
|
|
|
|
|
|
2015-10-10 09:08:13 +00:00
|
|
|
|
print u'strategyEngine.py registerStrategy() end.'
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def placeStopOrder(self, symbol, direction, offset, price, volume, strategy):
|
|
|
|
|
"""
|
2015-10-14 16:41:45 +00:00
|
|
|
|
下停止单(止损单,运行于本地引擎中)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
注意这里的price是停止单的触发价
|
|
|
|
|
"""
|
|
|
|
|
# 创建止损单对象
|
2015-10-24 00:01:02 +00:00
|
|
|
|
print u'strategyEngine.py placeStopOrder() symbol:{0}, direction:{1}, offset:{2}, price:{3}, volume:{4}.'\
|
|
|
|
|
.format(symbol, direction, offset, price, volume)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
so = StopOrder(symbol, direction, offset, price, volume, strategy)
|
|
|
|
|
|
|
|
|
|
# 获取该合约相关的止损单列表
|
|
|
|
|
try:
|
|
|
|
|
listSO = self.__dictStopOrder[symbol]
|
|
|
|
|
except KeyError:
|
|
|
|
|
listSO = []
|
|
|
|
|
self.__dictStopOrder[symbol] = listSO
|
|
|
|
|
|
|
|
|
|
# 将该止损单插入列表中
|
|
|
|
|
listSO.append(so)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return so
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def cancelStopOrder(self, so):
|
|
|
|
|
"""撤销停止单"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
symbol = so.symbol
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
listSO = self.__dictStopOrder[symbol]
|
|
|
|
|
|
|
|
|
|
if so in listSO:
|
|
|
|
|
listSO.remove(so)
|
|
|
|
|
|
|
|
|
|
if not listSO:
|
|
|
|
|
del self.__dictStopOrder[symbol]
|
|
|
|
|
except KeyError:
|
|
|
|
|
pass
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def startAll(self):
|
|
|
|
|
"""启动所有策略"""
|
2015-09-28 07:34:32 +00:00
|
|
|
|
print( u'启动所有策略')
|
2015-05-21 07:14:20 +00:00
|
|
|
|
for strategy in self.dictStrategy.values():
|
2015-09-28 07:34:32 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
strategy.start()
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def stopAll(self):
|
|
|
|
|
"""停止所有策略"""
|
2015-09-28 07:34:32 +00:00
|
|
|
|
print(u'停止所有策略')
|
2015-05-21 07:14:20 +00:00
|
|
|
|
for strategy in self.dictStrategy.values():
|
|
|
|
|
strategy.stop()
|
|
|
|
|
|
2015-10-19 08:42:17 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def saveData(self,id):
|
|
|
|
|
"""保存所有策略的过程数据"""
|
|
|
|
|
print(u'保存所有策略的过程数据')
|
|
|
|
|
for strategy in self.dictStrategy.values():
|
|
|
|
|
strategy.saveData(id)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
########################################################################
|
|
|
|
|
class StrategyTemplate(object):
|
|
|
|
|
"""策略模板"""
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def __init__(self, name, symbol, engine):
|
|
|
|
|
"""Constructor"""
|
|
|
|
|
self.name = name # 策略名称(注意唯一性)
|
|
|
|
|
self.symbol = symbol # 策略交易的合约
|
2015-05-26 08:05:04 +00:00
|
|
|
|
self.engine = engine # 策略引擎对象
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
self.trading = False # 策略是否启动交易
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def onTick(self, tick):
|
|
|
|
|
"""行情更新"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def onTrade(self, trade):
|
|
|
|
|
"""交易更新"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def onOrder(self, order):
|
|
|
|
|
"""报单更新"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def onStopOrder(self, orderRef):
|
|
|
|
|
"""停止单更新"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def onBar(self, o, h, l, c, volume, time):
|
|
|
|
|
"""K线数据更新"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def start(self):
|
|
|
|
|
"""
|
|
|
|
|
启动交易
|
|
|
|
|
这里是最简单的改变self.trading
|
|
|
|
|
有需要可以重新实现更复杂的操作
|
|
|
|
|
"""
|
|
|
|
|
self.trading = True
|
2015-05-26 08:05:04 +00:00
|
|
|
|
self.engine.writeLog(self.name + u'开始运行')
|
2015-10-19 08:42:17 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def saveData(self,Id):
|
|
|
|
|
"""保存数据"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
def stop(self):
|
|
|
|
|
"""
|
|
|
|
|
停止交易
|
|
|
|
|
同上
|
|
|
|
|
"""
|
|
|
|
|
self.trading = False
|
2015-05-26 08:05:04 +00:00
|
|
|
|
self.engine.writeLog(self.name + u'停止运行')
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-26 08:05:04 +00:00
|
|
|
|
def loadSetting(self, setting):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""
|
2015-05-26 08:05:04 +00:00
|
|
|
|
载入设置
|
2015-05-21 07:14:20 +00:00
|
|
|
|
setting通常是一个包含了参数设置的字典
|
|
|
|
|
"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
2015-10-17 16:23:24 +00:00
|
|
|
|
def buy(self, price, volume, orderTime, stopOrder=False ):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""买入开仓"""
|
2015-10-17 16:23:24 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
print u'strategyEngine.py StrategyTemplate({3}) buy(symbol:{0}, price:{1},volume:{2},time:{4})'.\
|
|
|
|
|
format(self.symbol, price, volume, self.name,orderTime)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
if self.trading:
|
|
|
|
|
if stopOrder:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 止损单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
so = self.engine.placeStopOrder(self.symbol, DIRECTION_BUY,
|
2015-05-21 07:14:20 +00:00
|
|
|
|
OFFSET_OPEN, price, volume, self)
|
|
|
|
|
return so
|
|
|
|
|
else:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 委托单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
ref = self.engine.sendOrder(self.symbol, DIRECTION_BUY,
|
2015-10-14 16:41:45 +00:00
|
|
|
|
OFFSET_OPEN, price, volume, orderTime, self)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return ref
|
|
|
|
|
else:
|
|
|
|
|
return None
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
|
|
|
|
#print (u'strategyEngine.py buy() end.')
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-10-17 16:23:24 +00:00
|
|
|
|
def cover(self, price, volume,orderTime, stopOrder=False):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""买入平仓"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
print u'strategyEngine.py StrategyTemplate({3}) cover(symbol:{0}, price:{1},volume:{2},time:{4})'.\
|
|
|
|
|
format(self.symbol, price, volume, self.name, orderTime)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
if self.trading:
|
|
|
|
|
if stopOrder:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
|
|
|
|
|
# 止损单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
so = self.engine.placeStopOrder(self.symbol, DIRECTION_BUY,
|
2015-05-21 07:14:20 +00:00
|
|
|
|
OFFSET_CLOSE, price, volume, self)
|
|
|
|
|
return so
|
|
|
|
|
else:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 委托单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
ref = self.engine.sendOrder(self.symbol, DIRECTION_BUY,
|
2015-10-14 16:41:45 +00:00
|
|
|
|
OFFSET_CLOSE, price, volume, orderTime, self)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return ref
|
|
|
|
|
else:
|
|
|
|
|
return None
|
2015-11-04 16:40:18 +00:00
|
|
|
|
|
2015-10-10 09:08:13 +00:00
|
|
|
|
print (u'strategyEngine.py cover() end.')
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-10-17 16:23:24 +00:00
|
|
|
|
def sell(self, price, volume, orderTime, stopOrder=False):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""卖出平仓"""
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-10-24 00:01:02 +00:00
|
|
|
|
print u'strategyEngine.py StrategyTemplate({3}) sell(symbol:{0}, price:{1},volume:{2},time:{4})'.\
|
|
|
|
|
format(self.symbol, price, volume, self.name, orderTime)
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
if self.trading:
|
|
|
|
|
if stopOrder:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 止损单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
so = self.engine.placeStopOrder(self.symbol, DIRECTION_SELL,
|
2015-05-21 07:14:20 +00:00
|
|
|
|
OFFSET_CLOSE, price, volume, self)
|
|
|
|
|
return so
|
|
|
|
|
else:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 委托单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
ref = self.engine.sendOrder(self.symbol, DIRECTION_SELL,
|
2015-10-14 16:41:45 +00:00
|
|
|
|
OFFSET_CLOSE, price, volume, orderTime, self)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return ref
|
|
|
|
|
else:
|
|
|
|
|
return None
|
2015-10-10 09:08:13 +00:00
|
|
|
|
#print u'strategyEngine.py sell() end.'
|
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-10-17 16:23:24 +00:00
|
|
|
|
def short(self, price, volume, orderTime, stopOrder=False):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""卖出开仓"""
|
2015-10-24 00:01:02 +00:00
|
|
|
|
print u'strategyEngine.py StrategyTemplate({3}) short(symbol:{0}, price:{1},volume:{2},time:{4})'.\
|
|
|
|
|
format(self.symbol, price, volume, self.name, orderTime)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
if self.trading:
|
|
|
|
|
if stopOrder:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 止损单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
so = self.engine.placeStopOrder(self.symbol, DIRECTION_SELL,
|
2015-05-21 07:14:20 +00:00
|
|
|
|
OFFSET_OPEN, price, volume, self)
|
|
|
|
|
return so
|
|
|
|
|
else:
|
2015-10-17 16:23:24 +00:00
|
|
|
|
# 委托单
|
2015-05-26 08:05:04 +00:00
|
|
|
|
ref = self.engine.sendOrder(self.symbol, DIRECTION_SELL,
|
2015-10-14 16:41:45 +00:00
|
|
|
|
OFFSET_OPEN, price, volume, orderTime, self)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
return ref
|
|
|
|
|
else:
|
|
|
|
|
return None
|
2015-10-24 00:01:02 +00:00
|
|
|
|
|
2015-10-10 09:08:13 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-26 08:05:04 +00:00
|
|
|
|
def cancelOrder(self, orderRef):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""撤单"""
|
2015-05-26 08:05:04 +00:00
|
|
|
|
self.engine.cancelOrder(orderRef)
|
2015-10-23 23:25:53 +00:00
|
|
|
|
|
2015-05-21 07:14:20 +00:00
|
|
|
|
#----------------------------------------------------------------------
|
2015-05-26 08:05:04 +00:00
|
|
|
|
def cancelStopOrder(self, so):
|
2015-05-21 07:14:20 +00:00
|
|
|
|
"""撤销停止单"""
|
2015-05-26 08:05:04 +00:00
|
|
|
|
self.engine.cancelStopOrder(so)
|
2015-05-21 07:14:20 +00:00
|
|
|
|
|
|
|
|
|
|