diff --git a/examples/VnTrader/run.py b/examples/VnTrader/run.py index 8c9b45f1..78d67aea 100644 --- a/examples/VnTrader/run.py +++ b/examples/VnTrader/run.py @@ -20,7 +20,8 @@ from vnpy.trader.gateway import (ctpGateway, oandaGateway, ibGateway, huobiGateway, okcoinGateway) if system == 'Windows': - from vnpy.trader.gateway import femasGateway, xspeedGateway + from vnpy.trader.gateway import (femasGateway, xspeedGateway, + futuGateway) if system == 'Linux': from vnpy.trader.gateway import xtpGateway @@ -51,6 +52,7 @@ def main(): if system == 'Windows': me.addGateway(femasGateway) me.addGateway(xspeedGateway) + me.addGateway(futuGateway) if system == 'Linux': me.addGateway(xtpGateway) diff --git a/vnpy/trader/gateway/futuGateway/Futu_connect.json b/vnpy/trader/gateway/futuGateway/Futu_connect.json new file mode 100644 index 00000000..d922ddf8 --- /dev/null +++ b/vnpy/trader/gateway/futuGateway/Futu_connect.json @@ -0,0 +1,5 @@ +{ + "market": "HK", + "password": "123456", + "env": 1 +} \ No newline at end of file diff --git a/vnpy/trader/gateway/futuGateway/__init__.py b/vnpy/trader/gateway/futuGateway/__init__.py new file mode 100644 index 00000000..0b477e7f --- /dev/null +++ b/vnpy/trader/gateway/futuGateway/__init__.py @@ -0,0 +1,10 @@ +# encoding: UTF-8 + +from vnpy.trader import vtConstant +from futuGateway import FutuGateway + +gatewayClass = FutuGateway +gatewayName = 'FUTU' +gatewayDisplayName = u'富途证券' +gatewayType = vtConstant.GATEWAYTYPE_INTERNATIONAL +gatewayQryEnabled = True diff --git a/vnpy/trader/gateway/futuGateway/futuGateway.py b/vnpy/trader/gateway/futuGateway/futuGateway.py new file mode 100644 index 00000000..7275a395 --- /dev/null +++ b/vnpy/trader/gateway/futuGateway/futuGateway.py @@ -0,0 +1,501 @@ +# encoding: UTF-8 + +''' +富途证券的gateway接入 +''' + +import json +from collections import OrderedDict +from threading import Thread +from time import sleep + +import futuquant as ft +from futuquant.open_context import (RET_ERROR, RET_OK, + StockQuoteHandlerBase, OrderBookHandlerBase, + USTradeOrderHandlerBase, USTradeDealHandlerBase, + HKTradeOrderHandlerBase, HKTradeDealHandlerBase) + +from vnpy.trader.vtGateway import * +from vnpy.trader.vtConstant import GATEWAYTYPE_INTERNATIONAL +from vnpy.trader.vtFunction import getJsonPath + +productMap = OrderedDict() +productMap[PRODUCT_EQUITY] = 'STOCK' +productMap[PRODUCT_INDEX] = 'IDX' +productMap[PRODUCT_ETF] = 'ETF' +productMap[PRODUCT_WARRANT] = 'WARRANT' +productMap[PRODUCT_BOND] = 'BOND' + +directionMap = {} +directionMap[DIRECTION_LONG] = '0' +directionMap[DIRECTION_SHORT] = '1' +directionMapReverse = {v:k for k,v in directionMap.items()} + +statusMapReverse = {} +statusMapReverse['0'] = STATUS_UNKNOWN +statusMapReverse['1'] = STATUS_NOTTRADED +statusMapReverse['2'] = STATUS_PARTTRADED +statusMapReverse['3'] = STATUS_ALLTRADED +statusMapReverse['4'] = STATUS_CANCELLED +statusMapReverse['5'] = STATUS_REJECTED +statusMapReverse['6'] = STATUS_CANCELLED +statusMapReverse['7'] = STATUS_CANCELLED +statusMapReverse['8'] = STATUS_UNKNOWN +statusMapReverse['21'] = STATUS_UNKNOWN +statusMapReverse['22'] = STATUS_UNKNOWN +statusMapReverse['23'] = STATUS_UNKNOWN + + + +######################################################################## +class FutuGateway(VtGateway): + """富途接口""" + + #---------------------------------------------------------------------- + def __init__(self, eventEngine, gatewayName='FUTU'): + """Constructor""" + super(FutuGateway, self).__init__(eventEngine, gatewayName) + + self.quoteCtx = None + self.tradeCtx = None + + self.market = '' + self.password = '' + self.env = 1 # 默认仿真交易 + + self.fileName = self.gatewayName + '_connect.json' + self.filePath = getJsonPath(self.fileName, __file__) + + self.tickDict = {} + + self.qryEnabled = True + self.qryThread = Thread(target=self.qryData) + + #---------------------------------------------------------------------- + def writeLog(self, content): + """输出日志""" + log = VtLogData() + log.gatewayName = self.gatewayName + log.logContent = content + self.onLog(log) + + #---------------------------------------------------------------------- + def writeError(self, code, msg): + """输出错误""" + error = VtErrorData() + error.gatewayName = self.gatewayName + error.errorID = code + error.errorMsg = msg + self.onError(error) + + #---------------------------------------------------------------------- + def connect(self): + """连接""" + self.connectQuote() + self.connectTrade() + + self.qryThread.start() + + #---------------------------------------------------------------------- + def qryData(self): + """初始化时查询数据""" + # 查询合约、成交、委托、持仓、账户 + self.qryContract() + self.qryTrade() + self.qryOrder() + self.qryPosition() + self.qryAccount() + + # 启动循环查询 + self.initQuery() + + #---------------------------------------------------------------------- + def connectQuote(self): + """连接行情功能""" + self.quoteCtx = ft.OpenQuoteContext() + + # 继承实现处理器类 + class QuoteHandler(StockQuoteHandlerBase): + """报价处理器""" + gateway = self # 缓存Gateway对象 + + def on_recv_rsp(self, rsp_str): + print 'quote handler' + ret_code, content = super(QuoteHandler, self).on_recv_rsp(rsp_str) + if ret_code != RET_OK: + return RET_ERROR, content + self.gateway.processQuote(content) + return RET_OK, content + + class OrderBookHandler(OrderBookHandlerBase): + """订单簿处理器""" + gateway = self + + def on_recv_rsp(self, rsp_str): + print 'book handler' + ret_code, content = super(OrderBookHandler, self).on_recv_rsp(rsp_str) + if ret_code != RET_OK: + return RET_ERROR, content + self.gateway.processOrderBook(content) + return RET_OK, content + + # 设置回调处理对象 + self.quoteCtx.set_handler(QuoteHandler()) + self.quoteCtx.set_handler(OrderBookHandler()) + + # 启动行情 + self.quoteCtx.start() + + self.writeLog(u'行情接口连接成功') + + #---------------------------------------------------------------------- + def connectTrade(self): + """连接交易功能""" + # 载入配置 + try: + f = open(self.filePath) + setting = json.load(f) + self.market = setting['market'] + self.password = setting['password'] + self.env = setting['env'] + except: + self.writeLog(u'载入配置文件出错') + return + + # 连接交易接口 + if self.market == 'US': + self.tradeCtx = ft.OpenUSTradeContext() + OrderHandlerBase = USTradeOrderHandlerBase + DealHandlerBase = USTradeDealHandlerBase + else: + self.tradeCtx = ft.OpenHKTradeContext() + OrderHandlerBase = HKTradeOrderHandlerBase + DealHandlerBase = HKTradeDealHandlerBase + + # 继承实现处理器类 + class OrderHandler(OrderHandlerBase): + """委托处理器""" + gateway = self # 缓存Gateway对象 + + def on_recv_rsp(self, rsp_str): + ret_code, content = super(OrderHandler, self).on_recv_rsp(rsp_str) + if ret_code != RET_OK: + return RET_ERROR, content + self.gateway.processOrder(content) + return RET_OK, content + + class DealHandler(DealHandlerBase): + """订单簿处理器""" + gateway = self + + def on_recv_rsp(self, rsp_str): + ret_code, content = super(DealHandler, self).on_recv_rsp(rsp_str) + if ret_code != RET_OK: + return RET_ERROR, content + self.gateway.processTrade(content) + return RET_OK, content + + # 设置回调处理对象 + #self.tradeCtx.set_handler(OrderHandler()) + #self.tradeCtx.set_handler(DealHandler()) + + # 启动交易接口 + self.tradeCtx.start() + + # 只有港股实盘交易才需要解锁 + if self.market == 'HK' and self.env == 0: + self.tradeCtx.unlock_trade(self.password) + + self.writeLog(u'交易接口连接成功') + + #---------------------------------------------------------------------- + def subscribe(self, subscribeReq): + """订阅行情""" + for data_type in ['QUOTE', 'ORDER_BOOK']: + code, data = self.quoteCtx.subscribe(subscribeReq.symbol, data_type, True) + + if code: + self.writeError(code, u'订阅行情失败:%s' %data) + + #---------------------------------------------------------------------- + def sendOrder(self, orderReq): + """发单""" + side = directionMap[orderReq.direction] + priceType = 0 # 只支持限价单 + + code, data = self.tradeCtx.place_order(orderReq.price, orderReq.volume, + orderReq.symbol, side, + priceType, self.env) + + if code: + self.writeError(code, u'委托失败:%s' %data) + return '' + + for ix, row in data.iterrows(): + orderID = str(row['orderid']) + + vtOrderID = '.'.join([self.gatewayName, orderID]) + + return vtOrderID + + #---------------------------------------------------------------------- + def cancelOrder(self, cancelOrderReq): + """撤单""" + code, data = self.tradeCtx.set_order_status(0, int(cancelOrderReq.orderID), + self.env) + + if code: + self.writeError(code, u'撤单失败:%s' %data) + return + + #---------------------------------------------------------------------- + def qryContract(self): + """查询合约""" + for vtProductClass, product in productMap.items(): + code, data = self.quoteCtx.get_stock_basicinfo(self.market, product) + + if code: + self.writeError(code, u'查询合约信息失败:%s' %data) + return + + for ix, row in data.iterrows(): + contract = VtContractData() + contract.gatewayName = self.gatewayName + + contract.symbol = row['code'] + contract.vtSymbol = contract.symbol + contract.name = row['name'] + contract.productClass = vtProductClass + contract.size = int(row['lot_size']) + contract.priceTick = 0.01 + + self.onContract(contract) + + self.writeLog(u'合约信息查询成功') + + #---------------------------------------------------------------------- + def qryAccount(self): + """查询账户资金""" + code, data = self.tradeCtx.accinfo_query(self.env) + + if code: + self.writeError(code, u'查询账户资金失败:%s' %data) + return + + for ix, row in data.iterrows(): + account = VtAccountData() + account.gatewayName = self.gatewayName + + account.accountID = '%s_%s' %(self.gatewayName, self.market) + account.vtAccountID = '.'.join([self.gatewayName, account.accountID]) + account.balance = float(row['ZCJZ']) + account.margin = float(row['GPBZJ']) + account.available = float(row['XJJY']) + + self.onAccount(account) + + #---------------------------------------------------------------------- + def qryPosition(self): + """查询持仓""" + code, data = self.tradeCtx.position_list_query(self.env) + + if code: + self.writeError(code, u'查询持仓失败:%s' %data) + return + + for ix, row in data.iterrows(): + pos = VtPositionData() + pos.gatewayName = self.gatewayName + + pos.symbol = row['code'] + pos.vtSymbol = pos.symbol + + pos.direction = DIRECTION_LONG + pos.vtPositionName = '.'.join([pos.vtSymbol, pos.direction]) + + pos.position = int(data['qty']) + pos.price = float(data['cost_price']) + pos.positionProfit = float(data['pl_val']) + pos.frozen = int(data['qty']) - int(data['can_sell_qty']) + + self.onPosition(pos) + + #---------------------------------------------------------------------- + def qryOrder(self): + """查询委托""" + code, data = self.tradeCtx.order_list_query("", self.env) + + if code: + self.writeError(code, u'查询委托失败:%s' %data) + return + + self.processOrder(data) + self.writeLog(u'委托查询成功') + + #---------------------------------------------------------------------- + def qryTrade(self): + """查询成交""" + code, data = self.tradeCtx.deal_list_query(self.env) + + if code: + self.writeError(code, u'查询成交失败:%s' %data) + return + + self.processDeal(data) + self.writeLog(u'成交查询成功') + + #---------------------------------------------------------------------- + def close(self): + """关闭""" + self.quoteCtx.stop() + self.tradeCtx.stop() + + #---------------------------------------------------------------------- + def initQuery(self): + """初始化连续查询""" + if self.qryEnabled: + # 需要循环的查询函数列表 + self.qryFunctionList = [self.qryAccount, self.qryPosition] + + self.qryCount = 0 # 查询触发倒计时 + self.qryTrigger = 2 # 查询触发点 + self.qryNextFunction = 0 # 上次运行的查询函数索引 + + self.startQuery() + + #---------------------------------------------------------------------- + def query(self, event): + """注册到事件处理引擎上的查询函数""" + self.qryCount += 1 + + if self.qryCount > self.qryTrigger: + # 清空倒计时 + self.qryCount = 0 + + # 执行查询函数 + function = self.qryFunctionList[self.qryNextFunction] + function() + + # 计算下次查询函数的索引,如果超过了列表长度,则重新设为0 + self.qryNextFunction += 1 + if self.qryNextFunction == len(self.qryFunctionList): + self.qryNextFunction = 0 + + #---------------------------------------------------------------------- + def startQuery(self): + """启动连续查询""" + self.eventEngine.register(EVENT_TIMER, self.query) + + #---------------------------------------------------------------------- + def setQryEnabled(self, qryEnabled): + """设置是否要启动循环查询""" + self.qryEnabled = qryEnabled + + #---------------------------------------------------------------------- + def processQuote(self, data): + """报价推送""" + print 'quote-----------' + print type(data), data + for ix, row in data.iterrows(): + symbol = row['code'] + + tick = self.tickDict.get(symbol, None) + if not tick: + tick = VtTickData() + tick.symbol = symbol + tick.vtSymbol = tick.symbol + tick.gatewayName = self.gatewayName + self.tickDict[symbol] = tick + + tick.date = row['data_date'] + tick.time = row['data_time'] + tick.datetime = None + + tick.openPrice = row['open_price'] + tick.highPrice = row['high_price'] + tick.lowPrice = row['low_price'] + tick.preClosePrice = row['prev_close_price'] + + tick.lastPrice = row['last_price'] + tick.volume = row['volume'] + + self.onTick(tick) + + #---------------------------------------------------------------------- + def processOrderBook(self, data): + """订单簿推送""" + print 'book-----------' + print data + + symbol = data['stock_code'] + + tick = self.tickDict.get(symbol, None) + if not tick: + tick = VtTickData() + tick.symbol = symbol + tick.vtSymbol = tick.symbol + tick.gatewayName = self.gatewayName + self.tickDict[symbol] = tick + + d = tick.__dict__ + for i in range(5): + bidData = data['Bid'][i] + askData = data['Ask'][i] + n = i + 1 + + d['bidPrice%s' %n] = bidData[0] + d['bidVolume%s' %n] = bidData[1] + d['askPrice%s' %n] = askData[0] + d['askVolume%s' %n] = askData[1] + + self.onTick(tick) + + #---------------------------------------------------------------------- + def processOrder(self, data): + """处理委托推送""" + print 'order -----------------' + print data + for ix, row in data.iterrows(): + order = VtOrderData() + order.gatewayName = self.gatewayName + + order.symbol = row['code'] + order.vtSymbol = order.symbol + + order.orderID = row['orderid'] + order.vtOrderID = '.'.join([self.gatewayName, order.orderID]) + + order.price = float(row['price']) + order.totalVolume = int(row['qty']) + order.tradedVolume = int(row['dealt_qty']) + order.orderTime = row['submited_time'] + order.status = statusMapReverse.get(row['status'], STATUS_UNKNOWN) + order.direction = directionMapReverse[row['order_side']] + self.onOrder(order) + + #---------------------------------------------------------------------- + def processDeal(self, data): + """处理成交推送""" + print 'deal -------------' + print data + for ix, row in data.iterrows(): + trade = VtTradeData() + trade.gatewayName = self.gatewayName + + trade.symbol = row['code'] + trade.vtSymbol = trade.symbol + + trade.tradeID = row['dealid'] + trade.vtTradeID = '.'.join([self.gatewayName, trade.tradeID]) + + trade.orderID = row['orderid'] + trade.vtOrderID = '.'.join([self.gatewayName, trade.orderID]) + + trade.price = float(row['price']) + trade.volume = float(row['qty']) + trade.direction = directionMapReverse[row['orderside']] + trade.tradeTime = row['time'] + + self.onTrade(trade) + + diff --git a/vnpy/trader/vtEngine.py b/vnpy/trader/vtEngine.py index 598e8bb1..a7fa6260 100644 --- a/vnpy/trader/vtEngine.py +++ b/vnpy/trader/vtEngine.py @@ -730,7 +730,7 @@ class PositionDetail(object): self.shortYd = pos.ydPosition self.shortTd = self.shortPos - self.shortYd - self.output() + #self.output() #---------------------------------------------------------------------- def updateOrderReq(self, req, vtOrderID): @@ -759,7 +759,7 @@ class PositionDetail(object): self.longPos = self.longTd + self.longYd self.shortPos = self.shortTd + self.shortYd - self.output() + #self.output() #---------------------------------------------------------------------- def calculateFrozen(self): @@ -812,7 +812,7 @@ class PositionDetail(object): self.longPosFrozen = self.longYdFrozen + self.longTdFrozen self.shortPosFrozen = self.shortYdFrozen + self.shortTdFrozen - self.output() + #self.output() #---------------------------------------------------------------------- def output(self):