代码拉取完成,页面将自动刷新
同步操作将从 海风/goctp 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
package goctp
import (
"bytes"
"encoding/gob"
"fmt"
"os"
"strings"
"sync"
"time"
"unsafe"
ctp "gitee.com/haifengat/goctp/ctpdefine"
)
// HFTrade 交易接口
type HFTrade struct {
UserID string // 交易员
InvestorID string // 帐号
BrokerID string // 经纪商
TradingDay string // 交易日
passWord string
SessionID int // 判断是否自己的委托用
Instruments sync.Map // 合约列表 (key: InstrumentID, value: *InstrumentField)
InstrumentStatuss sync.Map // 合约状态 (key: InstrumentID, value: *InstrumentStatus)
posiDetail map[string]*sync.Map // 原始持仓
Positions sync.Map // 合成后的持仓 (key: instrument_long/short value: *ctp.CThostFtdcInvestorPositionField)
Orders sync.Map // 委托 (key: sessionID_OrderRef, value: *OrderField)
Trades sync.Map // 成交 (key: TradeID_buy/sell, value: &TradeField)
sysID4Order sync.Map // key:OrderSysID,value: *OrderField
Account *AccountField // 帐户权益
UserAccounts map[string]*AccountField // 交易员:多帐户权益 string->*AccountField
UserPositions map[string]*sync.Map // 交易员:多帐户持仓
Investors map[string]struct{} // 多个帐号(交易员)
IsLogin bool // 登录成功
Version string // 版本号,如 v6.5.1_20200908 10:25:08
PrivateMode ctp.THOST_TE_RESUME_TYPE // 私有流模式
// qryTicker *time.Ticker // 循环查询
waitLogin sync.WaitGroup // 登录信号
waitQry sync.WaitGroup // 查询指令后等待响应完成
reqID int // requestid
cntOrder int // 计算order数量
cntTrade int // 计算trade数量
onFrontConnected OnFrontConnectedType // 事件
onFrontDisConnected OnFrontDisConnectedType
onRspUserLogin OnRspUserLoginType
onRtnOrder OnRtnOrderType
onRtnCancel OnRtnOrderType
onErrRtnOrder OnRtnErrOrderType
onErrAction OnRtnErrActionType
onRtnTrade OnRtnTradeType
onRtnInstrumentStatus OnRtnInstrumentStatusType
onRtnBankToFuture OnRtnFromBankToFutureByFuture
onRtnFutureToBank OnRtnFromFutureToBankByFuture
// 继承类要实现的函数
ReqConnect ReqConnectType
ReleaseAPI ReleaseAPIType
ReqUserLogin ReqUserLoginType
ReqAuthenticate ReqAuthenticateType
ReqSettlementInfoConfirm ReqSettlementInfoConfirmType
ReqQryInstrument ReqQryInstrumentType
ReqQryClassifiedInstrument ReqQryClassifiedInstrumentType
ReqQryTradingAccount ReqQryTradingAccountType
ReqQryInvestorPosition ReqQryInvestorPositionType
ReqOrder ReqOrderInsertType
ReqAction ReqOrderActionType
ReqFromBankToFutureByFuture ReqTransferType
ReqFromFutureToBankByFuture ReqTransferType
GetVersion GetVersionType
ReqQryInvestor ReqQryInvestorType
ReqQryOrder ReqQryOrderType
ReqQryTrade ReqQryTradeType
}
type ReqAuthenticateType func(*ctp.CThostFtdcReqAuthenticateField, int)
type ReqUserLoginType func(*ctp.CThostFtdcReqUserLoginField, int)
type ReqSettlementInfoConfirmType func(*ctp.CThostFtdcSettlementInfoConfirmField, int)
type ReqQryInstrumentType func(*ctp.CThostFtdcQryInstrumentField, int)
type ReqQryClassifiedInstrumentType func(*ctp.CThostFtdcQryClassifiedInstrumentField, int)
type ReqQryTradingAccountType func(*ctp.CThostFtdcQryTradingAccountField, int)
type ReqQryInvestorPositionType func(*ctp.CThostFtdcQryInvestorPositionField, int)
type ReqOrderInsertType func(*ctp.CThostFtdcInputOrderField, int)
type ReqOrderActionType = func(*ctp.CThostFtdcInputOrderActionField, int)
type ReqTransferType = func(*ctp.CThostFtdcReqTransferField, int)
type ReqConnectType = func(string)
type ReleaseAPIType func()
type GetVersionType func() string
type ReqQryInvestorType = func(*ctp.CThostFtdcQryInvestorField, int)
type ReqQryOrderType = func(f *ctp.CThostFtdcQryOrderField, i int)
type ReqQryTradeType = func(f *ctp.CThostFtdcQryTradeField, i int)
func (t *HFTrade) Init() {
t.PrivateMode = ctp.THOST_TERT_RESTART // 默认 restart
t.posiDetail = make(map[string]*sync.Map)
t.UserAccounts = make(map[string]*AccountField)
t.UserPositions = make(map[string]*sync.Map)
t.Investors = make(map[string]struct{})
for _, r := range []interface{}{t.ReqQryInvestor, t.ReqAuthenticate, t.ReqUserLogin, t.ReqSettlementInfoConfirm, t.ReqQryInstrument, t.ReqQryClassifiedInstrument, t.ReqQryTradingAccount, t.ReqQryInvestorPosition, t.ReqOrder, t.ReqAction, t.GetVersion} {
if r == nil {
panic("缺少继承函数")
}
}
t.Version = t.GetVersion()
fmt.Println("trade api version: ", t.Version)
// 执行目录下创建 log目录
_, err := os.Stat("log")
if err != nil {
os.Mkdir("log", os.ModePerm)
}
t.waitLogin = sync.WaitGroup{}
t.Account = new(AccountField)
}
// SetQuick 以quick模式启动(须在NewTrade前调用)
func (t *HFTrade) SetQuick(investorAraay ...string) {
t.PrivateMode = ctp.THOST_TERT_QUICK
for _, v := range investorAraay {
t.Investors[v] = struct{}{}
}
}
func (t *HFTrade) Release() {
if t.IsLogin {
t.waitLogin.Add(1)
t.IsLogin = false
t.waitLogin.Wait()
// 前置开,而后台关时, release 报下面的错误, 不处理则会返回 n 个4096后崩溃
// CThostFtdcUserApiImplBase::OnSessionDisconnected[0x7f1a3c000b68][1137639425][ 4097]
// DesignError:pthread_mutex_unlock in line 116 of file ../../source/event/Mutex.h
t.ReleaseAPI() // 未登录会报错
}
t.FrontDisConnected(0) // 需手动触发
}
func (t *HFTrade) getReqID() int {
t.reqID++
return t.reqID
}
// ReqLogin 登录
func (t *HFTrade) ReqLogin(user, pwd, broker, appID, authCode string) {
t.UserID = user
t.passWord = pwd
t.BrokerID = broker
f := ctp.CThostFtdcReqAuthenticateField{}
copy(f.BrokerID[:], broker)
copy(f.UserID[:], user)
copy(f.AppID[:], appID)
copy(f.AuthCode[:], authCode)
t.ReqAuthenticate(&f, t.getReqID())
}
//------------------- 函数封装 ----------------------
// ReqOrderInsert 限价委托
func (t *HFTrade) ReqOrderInsertByUser(investor, instrument string, buySell DirectionType, openClose OffsetFlagType, price float64, volume int) string {
f := ctp.CThostFtdcInputOrderField{}
copy(f.BrokerID[:], t.BrokerID)
if info, ok := t.Instruments.Load(instrument); ok {
copy(f.ExchangeID[:], info.(*InstrumentField).ExchangeID)
}
copy(f.UserID[:], t.UserID)
copy(f.InvestorID[:], investor)
copy(f.AccountID[:], investor)
f.IsAutoSuspend = ctp.TThostFtdcBoolType(0)
f.IsSwapOrder = ctp.TThostFtdcBoolType(0)
f.ForceCloseReason = ctp.THOST_FTDC_FCC_NotForceClose
// 参数赋值
id := t.getReqID()
copy(f.OrderRef[:], fmt.Sprintf("%012d", id))
copy(f.InstrumentID[:], instrument)
f.Direction = ctp.TThostFtdcDirectionType(buySell)
f.CombOffsetFlag[0] = byte(openClose)
f.CombHedgeFlag[0] = byte(HedgeFlagSpeculation)
// 不同类型的Order
f.OrderPriceType = ctp.THOST_FTDC_OPT_LimitPrice
f.TimeCondition = ctp.THOST_FTDC_TC_GFD
f.VolumeCondition = ctp.THOST_FTDC_VC_AV
f.ContingentCondition = ctp.THOST_FTDC_CC_Immediately
f.LimitPrice = ctp.TThostFtdcPriceType(price)
f.VolumeTotalOriginal = ctp.TThostFtdcVolumeType(volume)
t.ReqOrder(&f, id)
return fmt.Sprintf("%d_%s", t.SessionID, Bytes2String(f.OrderRef[:]))
}
// ReqOrderInsert 限价委托
func (t *HFTrade) ReqOrderInsert(instrument string, buySell DirectionType, openClose OffsetFlagType, price float64, volume int) string {
f := ctp.CThostFtdcInputOrderField{}
copy(f.BrokerID[:], t.BrokerID)
if info, ok := t.Instruments.Load(instrument); ok {
copy(f.ExchangeID[:], info.(*InstrumentField).ExchangeID)
}
copy(f.UserID[:], t.UserID)
copy(f.InvestorID[:], t.InvestorID)
copy(f.AccountID[:], t.InvestorID)
f.IsAutoSuspend = ctp.TThostFtdcBoolType(0)
f.IsSwapOrder = ctp.TThostFtdcBoolType(0)
f.ForceCloseReason = ctp.THOST_FTDC_FCC_NotForceClose
// 参数赋值
id := t.getReqID()
copy(f.OrderRef[:], fmt.Sprintf("%012d", id))
copy(f.InstrumentID[:], instrument)
f.Direction = ctp.TThostFtdcDirectionType(buySell)
f.CombOffsetFlag[0] = byte(openClose)
f.CombHedgeFlag[0] = byte(HedgeFlagSpeculation)
// 不同类型的Order
f.OrderPriceType = ctp.THOST_FTDC_OPT_LimitPrice
f.TimeCondition = ctp.THOST_FTDC_TC_GFD
f.VolumeCondition = ctp.THOST_FTDC_VC_AV
f.ContingentCondition = ctp.THOST_FTDC_CC_Immediately
f.LimitPrice = ctp.TThostFtdcPriceType(price)
f.VolumeTotalOriginal = ctp.TThostFtdcVolumeType(volume)
t.ReqOrder(&f, id)
return fmt.Sprintf("%d_%s", t.SessionID, Bytes2String(f.OrderRef[:]))
}
// ReqOrderInsertMarket 市价委托
func (t *HFTrade) ReqOrderInsertMarket(instrument string, buySell DirectionType, openClose OffsetFlagType, volume int) string {
f := ctp.CThostFtdcInputOrderField{}
copy(f.BrokerID[:], t.BrokerID)
if info, ok := t.Instruments.Load(instrument); ok {
copy(f.ExchangeID[:], info.(*InstrumentField).ExchangeID)
}
copy(f.UserID[:], t.UserID)
copy(f.InvestorID[:], t.InvestorID)
copy(f.AccountID[:], t.InvestorID)
f.IsAutoSuspend = ctp.TThostFtdcBoolType(0)
f.IsSwapOrder = ctp.TThostFtdcBoolType(0)
f.ForceCloseReason = ctp.THOST_FTDC_FCC_NotForceClose
// 参数赋值
id := t.getReqID()
copy(f.OrderRef[:], fmt.Sprintf("%012d", id))
copy(f.InstrumentID[:], instrument)
f.Direction = ctp.TThostFtdcDirectionType(buySell)
f.CombOffsetFlag[0] = byte(openClose)
f.CombHedgeFlag[0] = byte(HedgeFlagSpeculation)
// 不同类型的Order
f.OrderPriceType = ctp.THOST_FTDC_OPT_AnyPrice
f.TimeCondition = ctp.THOST_FTDC_TC_IOC
f.VolumeCondition = ctp.THOST_FTDC_VC_AV
f.ContingentCondition = ctp.THOST_FTDC_CC_Immediately
f.LimitPrice = ctp.TThostFtdcPriceType(0)
f.VolumeTotalOriginal = ctp.TThostFtdcVolumeType(volume)
t.ReqOrder(&f, id)
return fmt.Sprintf("%d_%s", t.SessionID, Bytes2String(f.OrderRef[:]))
}
// ReqOrderInsertFOK FOK委托[部成撤单]
func (t *HFTrade) ReqOrderInsertFOK(instrument string, buySell DirectionType, openClose OffsetFlagType, price float64, volume int) string {
f := ctp.CThostFtdcInputOrderField{}
copy(f.BrokerID[:], t.BrokerID)
if info, ok := t.Instruments.Load(instrument); ok {
copy(f.ExchangeID[:], info.(*InstrumentField).ExchangeID)
}
copy(f.UserID[:], t.UserID)
copy(f.InvestorID[:], t.InvestorID)
copy(f.AccountID[:], t.InvestorID)
f.IsAutoSuspend = ctp.TThostFtdcBoolType(0)
f.IsSwapOrder = ctp.TThostFtdcBoolType(0)
f.ForceCloseReason = ctp.THOST_FTDC_FCC_NotForceClose
// 参数赋值
id := t.getReqID()
copy(f.OrderRef[:], fmt.Sprintf("%012d", id))
copy(f.InstrumentID[:], instrument)
f.Direction = ctp.TThostFtdcDirectionType(buySell)
f.CombOffsetFlag[0] = byte(openClose)
f.CombHedgeFlag[0] = byte(HedgeFlagSpeculation)
// 不同类型的Order
f.OrderPriceType = ctp.THOST_FTDC_OPT_LimitPrice
f.TimeCondition = ctp.THOST_FTDC_TC_GFD
f.VolumeCondition = ctp.THOST_FTDC_VC_CV
f.ContingentCondition = ctp.THOST_FTDC_CC_Immediately
f.LimitPrice = ctp.TThostFtdcPriceType(price)
f.VolumeTotalOriginal = ctp.TThostFtdcVolumeType(volume)
t.ReqOrder(&f, id)
return fmt.Sprintf("%d_%s", t.SessionID, Bytes2String(f.OrderRef[:]))
}
// ReqOrderInsertFAK FAK委托[全成or撤单]
func (t *HFTrade) ReqOrderInsertFAK(instrument string, buySell DirectionType, openClose OffsetFlagType, price float64, volume int) string {
f := ctp.CThostFtdcInputOrderField{}
copy(f.BrokerID[:], t.BrokerID)
if info, ok := t.Instruments.Load(instrument); ok {
copy(f.ExchangeID[:], info.(*InstrumentField).ExchangeID)
}
copy(f.UserID[:], t.UserID)
copy(f.InvestorID[:], t.InvestorID)
copy(f.AccountID[:], t.InvestorID)
f.IsAutoSuspend = ctp.TThostFtdcBoolType(0)
f.IsSwapOrder = ctp.TThostFtdcBoolType(0)
f.ForceCloseReason = ctp.THOST_FTDC_FCC_NotForceClose
// 参数赋值
id := t.getReqID()
copy(f.OrderRef[:], fmt.Sprintf("%012d", id))
copy(f.InstrumentID[:], instrument)
f.Direction = ctp.TThostFtdcDirectionType(buySell)
f.CombOffsetFlag[0] = byte(openClose)
f.CombHedgeFlag[0] = byte(HedgeFlagSpeculation)
// 不同类型的Order
f.OrderPriceType = ctp.THOST_FTDC_OPT_LimitPrice
f.TimeCondition = ctp.THOST_FTDC_TC_IOC
f.VolumeCondition = ctp.THOST_FTDC_VC_AV
f.ContingentCondition = ctp.THOST_FTDC_CC_Immediately
f.LimitPrice = ctp.TThostFtdcPriceType(price)
f.VolumeTotalOriginal = ctp.TThostFtdcVolumeType(volume)
t.ReqOrder(&f, id)
return fmt.Sprintf("%d_%s", t.SessionID, Bytes2String(f.OrderRef[:]))
}
// ReqOrderAction 撤单
func (t *HFTrade) ReqOrderAction(orderID string) int {
if o, ok := t.Orders.Load(orderID); ok {
var order = o.(*OrderField)
f := ctp.CThostFtdcInputOrderActionField{}
copy(f.BrokerID[:], t.BrokerID)
copy(f.UserID[:], t.UserID)
copy(f.InvestorID[:], order.InvestorID)
copy(f.InstrumentID[:], order.InstrumentID)
copy(f.ExchangeID[:], order.ExchangeID)
copy(f.OrderRef[:], order.OrderRef)
f.ActionFlag = ctp.THOST_FTDC_AF_Delete
f.FrontID = ctp.TThostFtdcFrontIDType(order.FrontID)
f.SessionID = ctp.TThostFtdcSessionIDType(order.SessionID)
t.ReqAction(&f, t.getReqID())
return 0
}
return -1
}
// ReqBankToFuture 银行转期货
func (t *HFTrade) ReqBankToFuture(bankID, bankAccount, bankPwd string, amount float64) {
f := ctp.CThostFtdcReqTransferField{}
copy(f.TradeCode[:], "202001")
copy(f.BankBranchID[:], "0000")
copy(f.BrokerID[:], t.BrokerID)
copy(f.UserID[:], t.UserID)
copy(f.AccountID[:], t.InvestorID)
copy(f.Password[:], t.passWord)
copy(f.CurrencyID[:], "CNY")
f.LastFragment = ctp.THOST_FTDC_LF_Yes
f.IdCardType = ctp.THOST_FTDC_ICT_IDCard
f.CustType = ctp.THOST_FTDC_CUSTT_Person
f.InstallID = 1
f.FutureSerial = 0
f.VerifyCertNoFlag = ctp.THOST_FTDC_YNI_No
f.FutureFetchAmount = 0
f.CustFee = 0
f.BrokerFee = 0
f.SecuPwdFlag = ctp.THOST_FTDC_BPWDF_BlankCheck
f.RequestID = ctp.TThostFtdcRequestIDType(t.getReqID())
f.TID = 0
copy(f.BankID[:], bankID)
copy(f.BankAccount[:], bankAccount)
copy(f.BankPassWord[:], bankPwd)
f.TradeAmount = ctp.TThostFtdcTradeAmountType(amount)
t.ReqFromBankToFutureByFuture(&f, t.getReqID())
}
// ReqFutureToBank 期货转银行
func (t *HFTrade) ReqFutureToBank(bankID, bankAccount string, amount float64) {
f := ctp.CThostFtdcReqTransferField{}
copy(f.TradeCode[:], "202002")
copy(f.BankBranchID[:], "0000")
copy(f.BrokerID[:], t.BrokerID)
copy(f.UserID[:], t.UserID)
copy(f.AccountID[:], t.InvestorID)
copy(f.Password[:], t.passWord)
copy(f.CurrencyID[:], "CNY")
f.LastFragment = ctp.THOST_FTDC_LF_Yes
f.IdCardType = ctp.THOST_FTDC_ICT_IDCard
f.CustType = ctp.THOST_FTDC_CUSTT_Person
f.InstallID = 1
f.FutureSerial = 0
f.VerifyCertNoFlag = ctp.THOST_FTDC_YNI_No
f.FutureFetchAmount = 0
f.CustFee = 0
f.BrokerFee = 0
f.SecuPwdFlag = ctp.THOST_FTDC_BPWDF_BlankCheck
f.RequestID = ctp.TThostFtdcRequestIDType(t.getReqID())
f.TID = 0
copy(f.BankID[:], bankID)
copy(f.BankAccount[:], bankAccount)
// copy(f.BankPassWord[:], bankPwd)
f.TradeAmount = ctp.TThostFtdcTradeAmountType(amount)
t.ReqFromFutureToBankByFuture(&f, t.getReqID())
}
//-------------------- 响应封装 -----------------------
// RegOnFrontConnected 注册连接响应
func (t *HFTrade) RegOnFrontConnected(on OnFrontConnectedType) {
t.onFrontConnected = on
}
// RegOnFrontDisConnected 注册连接响应
func (t *HFTrade) RegOnFrontDisConnected(on OnFrontDisConnectedType) {
t.onFrontDisConnected = on
}
// RegOnRspUserLogin 注册登陆响应
/* <error id="INVALID_DATA_SYNC_STATUS" value="1" prompt="CTP:不在已同步状态"/>
<error id="INCONSISTENT_INFORMATION" value="2" prompt="CTP:会话信息不一致"/>
<error id="INVALID_LOGIN" value="3" prompt="CTP:不合法的登录"/>
<error id="USER_NOT_ACTIVE" value="4" prompt="CTP:用户不活跃"/>
<error id="DUPLICATE_LOGIN" value="5" prompt="CTP:重复的登录"/>
<error id="NOT_LOGIN_YET" value="6" prompt="CTP:还没有登录"/>
<error id="NOT_INITED" value="7" prompt="CTP:还没有初始化"/>
<error id="FRONT_NOT_ACTIVE" value="8" prompt="CTP:前置不活跃"/>
<error id="LOGIN_FORBIDDEN" value="75" prompt="CTP:连续登录失败次数超限,登录被禁止"/>*/
func (t *HFTrade) RegOnRspUserLogin(on OnRspUserLoginType) {
t.onRspUserLogin = on
}
// RegOnRtnOrder 注册委托响应
func (t *HFTrade) RegOnRtnOrder(on OnRtnOrderType) {
t.onRtnOrder = on
}
// RegOnErrRtnOrder 注册委托响应
func (t *HFTrade) RegOnErrRtnOrder(on OnRtnErrOrderType) {
t.onErrRtnOrder = on
}
// RegOnErrAction 注册撤单响应
func (t *HFTrade) RegOnErrAction(on OnRtnErrActionType) {
t.onErrAction = on
}
// RegOnRtnCancel 注册撤单响应
func (t *HFTrade) RegOnRtnCancel(on OnRtnOrderType) {
t.onRtnCancel = on
}
// RegOnRtnTrade 注册成交响应
func (t *HFTrade) RegOnRtnTrade(on OnRtnTradeType) {
t.onRtnTrade = on
}
// RegOnRtnInstrumentStatus 注册合约状态变化
func (t *HFTrade) RegOnRtnInstrumentStatus(on OnRtnInstrumentStatusType) {
t.onRtnInstrumentStatus = on
}
// RegOnRtnFromBankToFuture 注册银行转期货
func (t *HFTrade) RegOnRtnFromBankToFuture(on OnRtnFromBankToFutureByFuture) {
t.onRtnBankToFuture = on
}
// RegOnRtnFromFutureToBank 注册期货转银行
func (t *HFTrade) RegOnRtnFromFutureToBank(on OnRtnFromFutureToBankByFuture) {
t.onRtnFutureToBank = on
}
// RtnFromBankToFutureByFuture 银行转期货-期货端
func (t *HFTrade) RtnFromBankToFutureByFuture(field *ctp.CThostFtdcRspTransferField) {
if t.onRtnBankToFuture != nil {
f := TransferField{
Amout: float64(field.TradeAmount),
CurrencyID: Bytes2String(field.CurrencyID[:]),
ErrorID: int(field.ErrorID),
ErrorMsg: Bytes2String(field.ErrorMsg[:]),
}
t.onRtnBankToFuture(&f)
}
}
// RtnFromFutureToBankByFuture // 期货转银行-期货端
func (t *HFTrade) RtnFromFutureToBankByFuture(field *ctp.CThostFtdcRspTransferField) {
if t.onRtnFutureToBank != nil {
f := TransferField{
Amout: float64(field.TradeAmount),
CurrencyID: Bytes2String(field.CurrencyID[:]),
ErrorID: int(field.ErrorID),
ErrorMsg: Bytes2String(field.ErrorMsg[:]),
}
t.onRtnFutureToBank(&f)
}
}
// RtnInstrumentStatus 合约/品种/交易所 状态响应
func (t *HFTrade) RtnInstrumentStatus(field *ctp.CThostFtdcInstrumentStatusField) {
status, loaded := t.InstrumentStatuss.LoadOrStore(Bytes2String(field.InstrumentID[:]), &InstrumentStatus{
ExchangeID: Bytes2String(field.ExchangeID[:]),
InstrumentID: Bytes2String(field.InstrumentID[:]),
InstrumentStatus: InstrumentStatusType(field.InstrumentStatus),
EnterTime: Bytes2String(field.EnterTime[:]),
})
if loaded {
status.(*InstrumentStatus).InstrumentStatus = InstrumentStatusType(field.InstrumentStatus)
status.(*InstrumentStatus).EnterTime = Bytes2String(field.EnterTime[:])
}
if t.onRtnInstrumentStatus != nil {
t.onRtnInstrumentStatus(status.(*InstrumentStatus))
}
}
// RtnTrade 成交响应
func (t *HFTrade) RtnTrade(field *ctp.CThostFtdcTradeField) {
if _, exists := t.Investors[Bytes2String(field.InvestorID[:])]; !exists {
return
}
t.cntTrade++
var key string
tradeid := Bytes2String(field.TradeID[:])
if field.Direction == ctp.THOST_FTDC_D_Buy {
key = fmt.Sprintf("%s_buy", tradeid)
} else if field.Direction == ctp.THOST_FTDC_D_Sell {
key = fmt.Sprintf("%s_sell", tradeid)
} else {
key = "error"
}
tf, _ := t.Trades.LoadOrStore(key, &TradeField{
InvestorID: Bytes2String(field.InvestorID[:]),
Direction: DirectionType(field.Direction),
HedgeFlag: HedgeFlagType(field.HedgeFlag),
InstrumentID: Bytes2String(field.InstrumentID[:]),
ExchangeID: Bytes2String(field.ExchangeID[:]),
TradingDay: Bytes2String(field.TradingDay[:]),
Volume: int(field.Volume),
OffsetFlag: OffsetFlagType(field.OffsetFlag),
OrderSysID: Bytes2String(field.OrderSysID[:]),
Price: float64(field.Price),
TradeDate: Bytes2String(field.TradeDate[:]),
TradeTime: Bytes2String(field.TradeTime[:]),
TradeID: tradeid,
})
var f = tf.(*TradeField)
if t.IsLogin && len(t.Investors) == 1 { // 登录后:更新持仓 // 交易员不处理
if f.OffsetFlag == OffsetFlagOpen { // 开仓
var key string
var posiDire = PosiDirectionLong
if f.Direction == DirectionBuy {
key = fmt.Sprintf("%s_long", f.InstrumentID)
} else {
key = fmt.Sprintf("%s_short", f.InstrumentID)
posiDire = PosiDirectionShort
}
pf, _ := t.Positions.LoadOrStore(key, &PositionField{
InvestorID: f.InvestorID,
InstrumentID: f.InstrumentID,
PositionDirection: posiDire,
HedgeFlag: f.HedgeFlag,
ExchangeID: f.ExchangeID,
})
var p = pf.(*PositionField)
p.OpenVolume += f.Volume
p.OpenAmount += f.Price * float64(f.Volume)
if info, ok := t.Instruments.Load(f.InstrumentID); ok {
p.OpenCost += f.Price * float64(f.Volume) * float64(info.(*InstrumentField).VolumeMultiple)
}
p.Position += f.Volume
p.TodayPosition += f.Volume
} else {
var key string
if f.Direction == DirectionBuy {
key = fmt.Sprintf("%s_short", f.InstrumentID)
} else {
key = fmt.Sprintf("%s_long", f.InstrumentID)
}
if posi, ok := t.Positions.Load(key); ok {
var p = posi.(*PositionField)
p.OpenVolume -= f.Volume
p.OpenAmount -= f.Price * float64(f.Volume)
if info, ok := t.Instruments.Load(f.InstrumentID); ok {
p.OpenCost -= f.Price * float64(f.Volume) * float64(info.(*InstrumentField).VolumeMultiple)
}
p.Position -= f.Volume
// 解锁冻结,
if f.Direction == DirectionBuy { // 冻结空头
p.LongFrozen -= f.Volume
} else { // 冻结多头
p.ShortFrozen -= f.Volume
}
if f.OffsetFlag == OffsetFlagCloseToday {
p.TodayPosition -= f.Volume
} else { // 先平昨
lots := f.Volume
if p.YdPosition > 0 {
if lots >= p.YdPosition { // 昨仓全平
lots -= p.YdPosition
p.YdPosition = 0
} else { // 昨仓减少
p.YdPosition -= lots
lots = 0
}
}
p.TodayPosition -= lots
}
}
}
}
// 处理对应的Order
if ord, ok := t.sysID4Order.Load(f.OrderSysID); ok {
var o = ord.(*OrderField)
if o.TradePrice == 0 { // 在 VolumeLeft 前计算
o.TradePrice = f.Price
} else { // 计算均价
o.TradePrice = (o.TradePrice*float64(o.VolumeTotalOriginal-o.VolumeLeft) + f.Price*float64(f.Volume)) / float64(o.VolumeTotalOriginal-o.VolumeLeft+f.Volume)
}
o.LastTradeTime = f.TradeTime
o.VolumeTraded = f.Volume
o.VolumeLeft -= f.Volume
if o.VolumeLeft == 0 {
o.OrderStatus = OrderStatusAllTraded
o.StatusMsg = "全部成交"
} else {
o.OrderStatus = OrderStatusPartTradedQueueing
o.StatusMsg = "部分成交"
}
if t.IsLogin && t.onRtnOrder != nil {
t.onRtnOrder(o)
}
}
// 客户端响应
if t.IsLogin && t.onRtnTrade != nil {
t.onRtnTrade(f)
}
}
// RtnOrder 委托响应
func (t *HFTrade) RtnOrder(field *ctp.CThostFtdcOrderField) {
// 由rspQryOrder调用时,可能为空指针进而创建struct, 传至此处.
// 交易员模式时, qryorder返回所有用户的数据,亦在此处过滤.
if _, exists := t.Investors[Bytes2String(field.InvestorID[:])]; !exists {
return
}
t.cntOrder++
key := fmt.Sprintf("%d_%s", field.SessionID, Bytes2String(field.OrderRef[:]))
if of, exists := t.Orders.LoadOrStore(key, &OrderField{
InvestorID: Bytes2String(field.InvestorID[:]),
InstrumentID: Bytes2String(field.InstrumentID[:]),
SessionID: int(field.SessionID),
FrontID: int(field.FrontID),
OrderRef: Bytes2String(field.OrderRef[:]),
Direction: DirectionType(field.Direction),
OffsetFlag: OffsetFlagType(field.CombOffsetFlag[0]),
HedgeFlag: HedgeFlagType(field.CombHedgeFlag[0]),
LimitPrice: float64(field.LimitPrice),
VolumeTotalOriginal: int(field.VolumeTotalOriginal),
VolumeLeft: int(field.VolumeTotalOriginal),
ExchangeID: Bytes2String(field.ExchangeID[:]),
InsertDate: Bytes2String(field.InsertDate[:]),
InsertTime: Bytes2String(field.InsertTime[:]),
OrderStatus: OrderStatusNoTradeQueueing, // OrderStatusType(orderField.OrderStatus)
StatusMsg: "委托已提交", // bytes2GBKbytes2GBKString(orderField.StatusMsg[:])
IsLocal: int(field.SessionID) == t.SessionID,
}); !exists { // 新添加
if t.IsLogin && t.onRtnOrder != nil {
// 平仓指令, 冻结持仓(随后的持仓查询会进行修正),冻结持仓恢复会滞后 <=2s
f := of.(*OrderField)
if f.OffsetFlag != OffsetFlagOpen {
if f.Direction == DirectionBuy { // 冻结空头
key := fmt.Sprintf("%s_short", f.InstrumentID)
if posiField, ok := t.Positions.Load(key); ok {
posiField.(*PositionField).LongFrozen += f.VolumeTotalOriginal
}
} else {
key := fmt.Sprintf("%s_long", f.InstrumentID)
if posiField, ok := t.Positions.Load(key); ok { // 冻结多头
posiField.(*PositionField).ShortFrozen += f.VolumeTotalOriginal
}
}
}
t.onRtnOrder(f)
}
} else {
var f = of.(*OrderField)
if OrderStatusType(field.OrderStatus) == OrderStatusCanceled { // 处理撤单
f.OrderStatus = OrderStatusCanceled
f.StatusMsg = Bytes2String(field.StatusMsg[:])
f.CancelTime = Bytes2String(field.CancelTime[:])
// 错单
if t.IsLogin { // 登录前不响应
// 解锁冻结,
if f.OffsetFlag != OffsetFlagOpen {
if f.Direction == DirectionBuy { // 冻结空头
key := fmt.Sprintf("%s_short", f.InstrumentID)
if posiField, ok := t.Positions.Load(key); ok {
posiField.(*PositionField).LongFrozen -= f.VolumeLeft
}
} else {
key := fmt.Sprintf("%s_long", f.InstrumentID)
if posiField, ok := t.Positions.Load(key); ok { // 冻结多头
posiField.(*PositionField).ShortFrozen -= f.VolumeLeft
}
}
}
if strings.Contains(f.StatusMsg, "被拒绝") {
if t.onErrRtnOrder != nil {
t.onErrRtnOrder(f, &RspInfoField{
ErrorID: -1,
ErrorMsg: f.StatusMsg,
})
}
} else if t.onRtnCancel != nil {
t.onRtnCancel(f)
}
}
} else {
f.OrderSysID = Bytes2String(field.OrderSysID[:])
if len(f.OrderSysID) > 0 {
t.sysID4Order.Store(f.OrderSysID, f)
}
}
}
}
// ErrRtnOrderAction 撤单错误
func (t *HFTrade) ErrRtnOrderAction(field *ctp.CThostFtdcOrderActionField, info *ctp.CThostFtdcRspInfoField) {
if t.IsLogin && t.onErrAction != nil {
t.onErrAction(fmt.Sprintf("%d_%s", field.SessionID, field.OrderRef), &RspInfoField{
ErrorID: int(info.ErrorID),
ErrorMsg: Bytes2String(info.ErrorMsg[:]),
})
}
}
// ErrRtnOrderInsert 委托错误O
func (t *HFTrade) ErrRtnOrderInsert(field *ctp.CThostFtdcInputOrderField, info *ctp.CThostFtdcRspInfoField) {
if !t.IsLogin { // 过滤当日以前登录时的错误委托
return
}
key := fmt.Sprintf("%d_%s", t.SessionID, Bytes2String(field.OrderRef[:]))
of, _ := t.Orders.LoadOrStore(key, &OrderField{
InvestorID: Bytes2String(field.InvestorID[:]),
InstrumentID: Bytes2String(field.InstrumentID[:]),
SessionID: t.SessionID,
FrontID: 0,
OrderRef: Bytes2String(field.OrderRef[:]),
Direction: DirectionType(field.Direction),
OffsetFlag: OffsetFlagType(field.CombOffsetFlag[0]),
HedgeFlag: HedgeFlagType(field.CombHedgeFlag[0]),
LimitPrice: float64(field.LimitPrice),
VolumeTotalOriginal: int(field.VolumeTotalOriginal),
VolumeLeft: int(field.VolumeTotalOriginal),
ExchangeID: Bytes2String(field.ExchangeID[:]),
IsLocal: true,
})
var o = of.(*OrderField)
o.OrderStatus = OrderStatusCanceled
if t.onErrRtnOrder != nil {
t.onErrRtnOrder(o, &RspInfoField{ErrorID: int(info.ErrorID), ErrorMsg: Bytes2String(info.ErrorMsg[:])})
}
}
func (t *HFTrade) positionCom() {
for investor, detail := range t.posiDetail {
mpPosition, ok := t.UserPositions[investor]
if !ok {
mpPosition = &sync.Map{}
t.UserPositions[investor] = mpPosition
}
detail.Range(func(key, ps interface{}) bool {
pFinal := PositionField{
InvestorID: investor,
}
for _, p := range ps.([]*ctp.CThostFtdcInvestorPositionField) {
pFinal.InstrumentID = Bytes2String(p.InstrumentID[:])
pFinal.PositionDirection = PosiDirectionType(p.PosiDirection)
pFinal.HedgeFlag = HedgeFlagType(p.HedgeFlag)
pFinal.ExchangeID = Bytes2String(p.ExchangeID[:])
pFinal.PreSettlementPrice = float64(p.PreSettlementPrice)
pFinal.SettlementPrice = float64(p.SettlementPrice)
pFinal.Position += int(p.Position)
pFinal.TodayPosition += int(p.TodayPosition)
// pFinal.YdPosition += int(p.YdPosition) // 直接取值还需要减去当日平仓
pFinal.YdPosition = pFinal.Position - pFinal.TodayPosition
pFinal.LongFrozen += int(p.LongFrozen)
pFinal.ShortFrozen += int(p.ShortFrozen)
pFinal.LongFrozenAmount += float64(p.LongFrozenAmount)
pFinal.ShortFrozenAmount += float64(p.ShortFrozenAmount)
pFinal.OpenVolume += int(p.OpenVolume)
pFinal.CloseVolume += int(p.CloseVolume)
pFinal.OpenAmount += float64(p.OpenAmount)
pFinal.CloseAmount += float64(p.CloseAmount)
pFinal.PositionCost += float64(p.PositionCost)
pFinal.PreMargin += float64(p.PreMargin)
pFinal.UseMargin += float64(p.UseMargin)
pFinal.FrozenMargin += float64(p.FrozenMargin)
pFinal.FrozenCash += float64(p.FrozenCash)
pFinal.FrozenCommission += float64(p.FrozenCommission)
pFinal.CashIn += float64(p.CashIn)
pFinal.Commission += float64(p.Commission)
pFinal.CloseProfit += float64(p.CloseProfit)
pFinal.PositionProfit += float64(p.PositionProfit)
pFinal.OpenCost += float64(p.OpenCost)
pFinal.ExchangeMargin += float64(p.ExchangeMargin)
pFinal.CombPosition += int(p.CombPosition)
pFinal.CombLongFrozen += int(p.CombLongFrozen)
pFinal.CombShortFrozen += int(p.CombShortFrozen)
pFinal.CloseProfitByDate += float64(p.CloseProfitByDate)
pFinal.CloseProfitByTrade += float64(p.CloseProfitByTrade)
pFinal.StrikeFrozen += int(p.StrikeFrozen)
pFinal.StrikeFrozenAmount += float64(p.StrikeFrozenAmount)
pFinal.AbandonFrozen += int(p.AbandonFrozen)
pFinal.YdStrikeFrozen += int(p.YdStrikeFrozen)
pFinal.PositionCostOffset += float64(p.PositionCostOffset)
}
mpPosition.Store(key, &pFinal)
return true
})
if investor == t.InvestorID {
t.Positions = *mpPosition
}
t.posiDetail[investor] = &sync.Map{} // 数据清空
}
}
// RspQryInvestorPosition 持仓
func (t *HFTrade) RspQryInvestorPosition(field *ctp.CThostFtdcInvestorPositionField, b bool) {
// 多帐号处理
investor := Bytes2String(field.InvestorID[:])
detail, ok := t.posiDetail[investor]
if !ok {
detail = &sync.Map{}
t.posiDetail[investor] = detail
}
// 复制接口中的数据
var buf bytes.Buffer
var p = new(ctp.CThostFtdcInvestorPositionField)
gob.NewEncoder(&buf).Encode(field)
gob.NewDecoder(bytes.NewBuffer(buf.Bytes())).Decode(p)
instrumentID := Bytes2String(p.InstrumentID[:])
if len(instrumentID) > 0 { // 偶尔出现NULL的数据导致数据转换错误
if _, ok := t.Instruments.Load(instrumentID); ok { // 解决交易所自主合成某些不可交易的套利合约的问题如 SPC y2005&p2001
var key string
if p.PosiDirection == ctp.THOST_FTDC_PD_Long {
key = fmt.Sprintf("%s_long", Bytes2String(p.InstrumentID[:]))
} else if p.PosiDirection == ctp.THOST_FTDC_PD_Short {
key = fmt.Sprintf("%s_short", Bytes2String(p.InstrumentID[:]))
} else {
key = fmt.Sprintf("%s_net", Bytes2String(p.InstrumentID[:]))
}
// key := fmt.Sprintf("%s_%c", Bytes2String(p.InstrumentID[:]), PosiDirectionType(p.PosiDirection))
ps, _ := detail.LoadOrStore(key, make([]*ctp.CThostFtdcInvestorPositionField, 0))
ps = append(ps.([]*ctp.CThostFtdcInvestorPositionField), p)
detail.Store(key, ps) // append后指针有变化,需重新赋值
}
}
if b {
t.positionCom()
if !t.IsLogin {
t.waitLogin.Done() // 通知:1. 登录响应可以发了 2. release 可以继续了
time.Sleep(100 * time.Millisecond) //登录过程中是否要等待 islogin 的赋值
}
go func() {
if t.IsLogin { // release 后不处理
time.Sleep(1100 * time.Millisecond)
faccount := ctp.CThostFtdcQryTradingAccountField{}
copy(faccount.BrokerID[:], t.BrokerID)
t.ReqQryTradingAccount(&faccount, t.getReqID())
}
}()
}
}
// RspQryTradingAccount 权益
func (t *HFTrade) RspQryTradingAccount(field *ctp.CThostFtdcTradingAccountField, b bool) {
//infoField := (* ctp.CThostFtdcRspInfoField)(unsafe.Pointer(info))
accID := Bytes2String(field.AccountID[:])
acc, ok := t.UserAccounts[accID]
if !ok {
acc = &AccountField{}
t.UserAccounts[accID] = acc
if accID == t.InvestorID {
t.Account = acc
}
}
acc.InvestorID = accID
acc.PreMortgage = float64(field.PreMortgage)
acc.PreDeposit = float64(field.PreDeposit)
acc.PreBalance = float64(field.PreBalance)
acc.PreMargin = float64(field.PreMargin)
acc.InterestBase = float64(field.InterestBase)
acc.Interest = float64(field.Interest)
acc.Deposit = float64(field.Deposit)
acc.Withdraw = float64(field.Withdraw)
acc.FrozenMargin = float64(field.FrozenMargin)
acc.FrozenCash = float64(field.FrozenCash)
acc.FrozenCommission = float64(field.FrozenCommission)
acc.CurrMargin = float64(field.CurrMargin)
acc.CashIn = float64(field.CashIn)
acc.Commission = float64(field.Commission)
acc.CloseProfit = float64(field.CloseProfit)
acc.PositionProfit = float64(field.PositionProfit)
acc.Balance = float64(field.Balance)
acc.Available = float64(field.Available)
acc.WithdrawQuota = float64(field.WithdrawQuota)
acc.Reserve = float64(field.Reserve)
acc.Credit = float64(field.Credit)
acc.Mortgage = float64(field.Mortgage)
acc.ExchangeMargin = float64(field.ExchangeMargin)
acc.DeliveryMargin = float64(field.DeliveryMargin)
acc.ExchangeDeliveryMargin = float64(field.ExchangeDeliveryMargin)
acc.ReserveBalance = float64(field.ReserveBalance)
acc.CurrencyID = Bytes2String(field.CurrencyID[:])
acc.PreFundMortgageIn = float64(field.PreFundMortgageIn)
acc.PreFundMortgageOut = float64(field.PreFundMortgageOut)
acc.FundMortgageIn = float64(field.FundMortgageIn)
acc.FundMortgageOut = float64(field.FundMortgageOut)
acc.FundMortgageAvailable = float64(field.FundMortgageAvailable)
acc.MortgageableFund = float64(field.MortgageableFund)
if b { // 查询完成
go func() { // 查持仓
time.Sleep(1100 * time.Millisecond)
fposition := ctp.CThostFtdcQryInvestorPositionField{}
copy(fposition.BrokerID[:], t.BrokerID)
t.ReqQryInvestorPosition(&fposition, t.getReqID())
}()
}
}
// 循环查询持仓&资金
func (t *HFTrade) qryUser() {
time.Sleep(1500 * time.Millisecond) // 遇到登录过程中停止,请增加此处的延时时间
// 等待之前的Order响应完再发送登录通知
var ordCnt, trdCnt int
for {
if ordCnt == t.cntOrder && trdCnt == t.cntTrade {
break
}
ordCnt = t.cntOrder
trdCnt = t.cntTrade
time.Sleep(500 * time.Millisecond)
}
fmt.Println("orders: ", ordCnt, " trades: ", trdCnt)
// 改为响应中相互调用,以避免release时,查询处理未完成造成的异常
faccount := ctp.CThostFtdcQryTradingAccountField{}
copy(faccount.BrokerID[:], t.BrokerID)
t.ReqQryTradingAccount(&faccount, t.getReqID())
}
// RspQryOrder 查委托响应
func (t *HFTrade) RspQryOrder(field *ctp.CThostFtdcOrderField, b bool) {
if len(Bytes2String(field.InvestorID[:])) > 0 { // 处理当日无委托时的空响应
t.RtnOrder(field) // 处理两次,以触发自定义处理的代码
t.RtnOrder(field)
}
if b {
t.waitQry.Done()
}
}
// RspQryTrade 查成交响应
func (t *HFTrade) RspQryTrade(field *ctp.CThostFtdcTradeField, b bool) {
if len(Bytes2String(field.InvestorID[:])) > 0 { // 无成交
t.RtnTrade(field) // 处理两次,以触发自定义处理的代码
}
if b {
t.waitQry.Done()
}
}
// RspQryInvestor 查用户(交易员下有多个帐号)
func (t *HFTrade) RspQryInvestor(field *ctp.CThostFtdcInvestorField, b bool) {
investorID := Bytes2String(field.InvestorID[:])
t.Investors[investorID] = struct{}{}
if b {
go func() {
// qry order
time.Sleep(1200 * time.Millisecond)
t.waitQry.Add(1)
fmt.Println(time.Now().Local().Format("2006-01-02 15:04:05"), " qry order")
qryOrder := ctp.CThostFtdcQryOrderField{}
copy(qryOrder.BrokerID[:], t.BrokerID)
t.ReqQryOrder(&qryOrder, t.getReqID())
t.waitQry.Wait()
time.Sleep(1200 * time.Millisecond)
t.waitQry.Add(1)
fmt.Println(time.Now().Local().Format("2006-01-02 15:04:05"), " qry trade")
qryTrade := ctp.CThostFtdcQryTradeField{}
copy(qryTrade.BrokerID[:], t.BrokerID)
t.ReqQryTrade(&qryTrade, t.getReqID())
t.waitQry.Wait()
fmt.Println(time.Now().Local().Format("2006-01-02 15:04:05"), " qry finished.")
// 循环查持仓/权益
t.qryUser()
}()
}
}
// RspQryInstrument 合约
func (t *HFTrade) RspQryInstrument(field *ctp.CThostFtdcInstrumentField, b bool) {
if field != nil {
t.Instruments.Store(Bytes2String(field.InstrumentID[:]), &InstrumentField{
InstrumentID: Bytes2String(field.InstrumentID[:]),
ExchangeID: Bytes2String(field.ExchangeID[:]),
ProductID: Bytes2String(field.ProductID[:]),
ProductClass: ProductClassType(field.ProductClass),
MaxMarketOrderVolume: int(field.MaxMarketOrderVolume),
MinMarketOrderVolume: int(field.MinMarketOrderVolume),
MaxLimitOrderVolume: int(field.MaxLimitOrderVolume),
MinLimitOrderVolume: int(field.MinLimitOrderVolume),
VolumeMultiple: int(field.VolumeMultiple),
PriceTick: float64(field.PriceTick),
PositionType: PositionTypeType(field.PositionType),
UseMaxMarginSideAlgorithm: field.MaxMarginSideAlgorithm == '1',
UnderlyingInstrID: Bytes2String(field.UnderlyingInstrID[:]),
StrikePrice: float64(field.StrikePrice),
OptionsType: OptionsTypeType(field.OptionsType),
UnderlyingMultiple: float64(field.UnderlyingMultiple),
CombinationType: CombinationTypeType(field.CombinationType),
ExpireDate: Bytes2String(field.ExpireDate[:]),
StartDelivDate: Bytes2String(field.StartDelivDate[:]),
EndDelivDate: Bytes2String(field.EndDelivDate[:]),
})
}
if b && !t.IsLogin {
if t.PrivateMode == ctp.THOST_TERT_QUICK { // 交易员模式
f := ctp.CThostFtdcQryInvestorField{}
copy(f.BrokerID[:], t.BrokerID)
go func() {
time.Sleep(1100 * time.Millisecond)
t.ReqQryInvestor(&f, t.getReqID())
}()
} else {
go t.qryUser()
}
}
}
// RspSettlementInfoConfirm 确认结算
func (t *HFTrade) RspSettlementInfoConfirm() {
if strings.Compare(t.Version, "v6.5.1") < 0 {
t.ReqQryInstrument(&ctp.CThostFtdcQryInstrumentField{}, t.getReqID())
} else {
f := ctp.CThostFtdcQryClassifiedInstrumentField{
TradingType: ctp.THOST_FTDC_TD_TRADE,
ClassType: ctp.THOST_FTDC_INS_ALL,
}
t.ReqQryClassifiedInstrument(&f, t.getReqID())
}
}
// RspUserLogin 登录
func (t *HFTrade) RspUserLogin(loginField *ctp.CThostFtdcRspUserLoginField, infoField *ctp.CThostFtdcRspInfoField) {
if infoField.ErrorID == 0 {
t.SessionID = int(loginField.SessionID)
t.TradingDay = Bytes2String(loginField.TradingDay[:])
// investor 赋值
t.InvestorID = t.UserID
if t.PrivateMode == ctp.THOST_TERT_RESTART {
t.Investors[t.UserID] = struct{}{}
}
// 用waitgroup控制登录消息发送信号
if t.onRspUserLogin != nil {
t.waitLogin.Add(1)
go func(field *RspUserLoginField) {
f := ctp.CThostFtdcSettlementInfoConfirmField{}
copy(f.InvestorID[:], t.UserID)
// copy(f.AccountID[:], t.InvestorID)
copy(f.BrokerID[:], t.BrokerID)
t.ReqSettlementInfoConfirm(&f, t.getReqID())
t.waitLogin.Wait()
// 登录成功响应
t.IsLogin = true
t.onRspUserLogin(field, &RspInfoField{ErrorID: 0, ErrorMsg: "成功"})
}(&RspUserLoginField{
TradingDay: t.TradingDay,
LoginTime: Bytes2String(loginField.LoginTime[:]),
BrokerID: t.BrokerID,
UserID: t.UserID,
FrontID: int(loginField.FrontID),
SessionID: t.SessionID,
MaxOrderRef: Bytes2String(loginField.MaxOrderRef[:]),
SystemName: Bytes2String(loginField.SystemName[:]),
SysVersion: Bytes2String(loginField.SysVersion[:]),
})
}
} else {
t.onRspUserLogin(&RspUserLoginField{}, &RspInfoField{ErrorID: int(infoField.ErrorID), ErrorMsg: Bytes2String(infoField.ErrorMsg[:])})
}
}
// RspAuthenticate 认证
func (t *HFTrade) RspAuthenticate(info *ctp.CThostFtdcRspInfoField) {
if info.ErrorID == 0 {
f := ctp.CThostFtdcReqUserLoginField{}
copy(f.UserID[:], t.UserID)
copy(f.BrokerID[:], t.BrokerID)
copy(f.Password[:], t.passWord)
copy(f.UserProductInfo[:], "@HF")
t.ReqUserLogin(&f, t.getReqID())
} else if t.onRspUserLogin != nil {
infoField := (*ctp.CThostFtdcRspInfoField)(unsafe.Pointer(info))
t.onRspUserLogin(&RspUserLoginField{}, &RspInfoField{ErrorID: int(infoField.ErrorID), ErrorMsg: Bytes2String(infoField.ErrorMsg[:])})
}
}
// FrontDisConnected 断开响应
func (t *HFTrade) FrontDisConnected(reason int) {
if t.onFrontDisConnected != nil {
t.onFrontDisConnected(reason)
}
}
// FrontConnected 连接
func (t *HFTrade) FrontConnected() {
if t.onFrontConnected != nil {
t.onFrontConnected()
}
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。