471 lines
15 KiB
Go
471 lines
15 KiB
Go
package newTest
|
||
|
||
import (
|
||
"context"
|
||
"fmt"
|
||
"git.qtrade.icu/lychiyu/qbtrade/pkg/fixedpoint"
|
||
"git.qtrade.icu/lychiyu/qbtrade/pkg/qbtrade"
|
||
"git.qtrade.icu/lychiyu/qbtrade/pkg/strategy/common"
|
||
"git.qtrade.icu/lychiyu/qbtrade/pkg/types"
|
||
"git.qtrade.icu/lychiyu/qbtrade/pkg/util"
|
||
log "github.com/sirupsen/logrus"
|
||
"strings"
|
||
"sync"
|
||
)
|
||
|
||
const ID = "new_test"
|
||
|
||
const (
|
||
ShortTag = "short"
|
||
ShortProfitTag = "short_profit"
|
||
ShortLossTag = "short_loss"
|
||
|
||
LongTag = "long"
|
||
LongProfitTag = "long_profit"
|
||
LongLossTag = "long_loss"
|
||
)
|
||
|
||
func init() {
|
||
qbtrade.RegisterStrategy(ID, &Strategy{})
|
||
}
|
||
|
||
type Strategy struct {
|
||
*common.Strategy
|
||
|
||
Environment *qbtrade.Environment
|
||
markets map[string]types.Market
|
||
|
||
// persistence fields
|
||
Positions map[string]*types.Position `persistence:"position"`
|
||
ProfitStats *types.ProfitStats `persistence:"profit_stats"`
|
||
//TradeStats *types.TradeStats `persistence:"trade_stats"`
|
||
|
||
//配置文件
|
||
ReCalculate bool `json:"recalculate"`
|
||
OrderType types.OrderType `json:"orderType"`
|
||
Symbols []string `json:"symbols"`
|
||
Interval types.Interval `json:"interval"`
|
||
NRCount int `json:"nr_count"`
|
||
DryRun bool `json:"dry_run"`
|
||
ProfitRange fixedpoint.Value `json:"profitRange"`
|
||
LossRange fixedpoint.Value `json:"lossRange"`
|
||
StrictMode bool `json:"strict_mode"`
|
||
Leverage fixedpoint.Value `json:"leverage"`
|
||
qbtrade.QuantityOrAmount
|
||
|
||
// 计算NR的历史kline
|
||
CalKLines map[string][]types.KLine
|
||
// 符合NR的kline
|
||
LastNRCandles map[string]*types.KLine
|
||
|
||
session *qbtrade.ExchangeSession
|
||
orderExecutors map[string]*qbtrade.GeneralOrderExecutor
|
||
|
||
//AccountValueCalculator *qbtrade.AccountValueCalculator
|
||
|
||
qbtrade.StrategyController
|
||
|
||
ordered map[string]bool // 是否已经下单
|
||
orderedSide map[string]string // 成交单的方向
|
||
|
||
LongOrder map[string]types.SubmitOrder
|
||
LongProfitOrder map[string]types.SubmitOrder
|
||
LongLossOrder map[string]types.SubmitOrder
|
||
ShortOrder map[string]types.SubmitOrder
|
||
ShortProfitOrder map[string]types.SubmitOrder
|
||
ShortLossOrder map[string]types.SubmitOrder
|
||
}
|
||
|
||
func (s *Strategy) Defaults() error {
|
||
if s.OrderType == "" {
|
||
log.Infof("order type is not set, using limit maker order type")
|
||
s.OrderType = types.OrderTypeLimit
|
||
//s.OrderType = types.OrderTypeStopLimit
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (s *Strategy) Initialize() error {
|
||
if s.Strategy == nil {
|
||
s.Strategy = &common.Strategy{}
|
||
}
|
||
|
||
return nil
|
||
}
|
||
|
||
func (s *Strategy) ID() string {
|
||
return ID
|
||
}
|
||
|
||
func (s *Strategy) InstanceID() string {
|
||
return fmt.Sprintf("%s:%s:%s", ID, strings.Join(s.Symbols, "-"), s.Interval)
|
||
}
|
||
|
||
func (s *Strategy) Subscribe(session *qbtrade.ExchangeSession) {
|
||
for _, symbol := range s.Symbols {
|
||
session.Subscribe(types.KLineChannel, symbol, types.SubscribeOptions{Interval: s.Interval})
|
||
session.Subscribe(types.MarketTradeChannel, symbol, types.SubscribeOptions{})
|
||
}
|
||
}
|
||
|
||
func (s *Strategy) OnKLineClosed(ctx context.Context, kline types.KLine, symbol string) {
|
||
if s.ordered[symbol] {
|
||
return
|
||
}
|
||
|
||
calKLines := s.CalKLines[symbol]
|
||
|
||
if len(s.CalKLines) < s.NRCount {
|
||
return
|
||
}
|
||
nr := calKLines[len(calKLines)-1]
|
||
preNr := calKLines[len(calKLines)-2]
|
||
isNR := true
|
||
|
||
if preNr.High < nr.High || preNr.Low > nr.Low {
|
||
isNR = false
|
||
return
|
||
}
|
||
|
||
for i := len(calKLines) - s.NRCount; i < len(calKLines); i++ {
|
||
// 这种是所有的kline都要高于nr
|
||
//if s.CalKLines[i].High > nr.High || s.CalKLines[i].Low < nr.Low {
|
||
// isNR = false
|
||
// break
|
||
//}
|
||
if s.StrictMode {
|
||
if calKLines[i].High-calKLines[i].Low < nr.High-nr.Low {
|
||
isNR = false
|
||
break
|
||
}
|
||
} else {
|
||
if (calKLines[i].High-calKLines[i].Low)/calKLines[i].Low < (nr.High-nr.Low)/nr.Low {
|
||
isNR = false
|
||
break
|
||
}
|
||
}
|
||
|
||
}
|
||
|
||
if isNR {
|
||
s.LastNRCandles[symbol] = &nr
|
||
log.Infof("交易信号(%s):%+v", symbol, kline)
|
||
s.placeOrders(ctx, symbol)
|
||
return
|
||
}
|
||
}
|
||
|
||
func (s *Strategy) cancelSideOrder(ctx context.Context, symbol string) {
|
||
if s.orderedSide[symbol] == "" || len(s.orderExecutors[symbol].ActiveMakerOrders().Orders()) <= 0 {
|
||
return
|
||
}
|
||
|
||
if s.orderedSide[symbol] == LongTag {
|
||
log.Infof("the long order is filled (%s), will cancel short order", symbol)
|
||
s.orderExecutors[symbol].CancelOrders(ctx, types.Order{SubmitOrder: s.ShortOrder[symbol]})
|
||
s.orderExecutors[symbol].CancelOrders(ctx, types.Order{SubmitOrder: s.ShortLossOrder[symbol]})
|
||
s.orderExecutors[symbol].CancelOrders(ctx, types.Order{SubmitOrder: s.ShortProfitOrder[symbol]})
|
||
} else {
|
||
log.Infof("the short order is filled (%s), will cancel short order", symbol)
|
||
s.orderExecutors[symbol].CancelOrders(ctx, types.Order{SubmitOrder: s.LongOrder[symbol]})
|
||
s.orderExecutors[symbol].CancelOrders(ctx, types.Order{SubmitOrder: s.LongLossOrder[symbol]})
|
||
s.orderExecutors[symbol].CancelOrders(ctx, types.Order{SubmitOrder: s.LongProfitOrder[symbol]})
|
||
}
|
||
}
|
||
|
||
func (s *Strategy) Run(ctx context.Context, orderExecutor qbtrade.OrderExecutor, session *qbtrade.ExchangeSession) error {
|
||
s.session = session
|
||
s.markets = s.session.Markets()
|
||
|
||
s.Positions = make(map[string]*types.Position)
|
||
s.CalKLines = make(map[string][]types.KLine)
|
||
s.LastNRCandles = make(map[string]*types.KLine)
|
||
s.orderExecutors = make(map[string]*qbtrade.GeneralOrderExecutor)
|
||
s.ordered = make(map[string]bool)
|
||
s.orderedSide = make(map[string]string)
|
||
s.LongOrder = make(map[string]types.SubmitOrder)
|
||
s.LongLossOrder = make(map[string]types.SubmitOrder)
|
||
s.LongProfitOrder = make(map[string]types.SubmitOrder)
|
||
s.ShortOrder = make(map[string]types.SubmitOrder)
|
||
s.ShortLossOrder = make(map[string]types.SubmitOrder)
|
||
s.ShortProfitOrder = make(map[string]types.SubmitOrder)
|
||
|
||
qbtrade.Notify("NR4策略开始执行...")
|
||
|
||
//for _, symbol := range s.Symbols {
|
||
// s.Strategy.Initialize(ctx, s.Environment, session, s.markets[symbol], ID, s.InstanceID())
|
||
//}
|
||
//
|
||
for _, symbol := range s.Symbols {
|
||
s.Positions[symbol] = types.NewPositionFromMarket(s.markets[symbol])
|
||
}
|
||
//
|
||
//if s.ProfitStats == nil {
|
||
// s.ProfitStats = types.NewProfitStats(s.Market)
|
||
//}
|
||
|
||
//if s.TradeStats == nil {
|
||
// s.TradeStats = types.NewTradeStats(s.Symbol)
|
||
//}
|
||
|
||
s.OnSuspend(func() {
|
||
// Cancel active orders
|
||
for _, symbol := range s.Symbols {
|
||
_ = s.orderExecutors[symbol].GracefulCancel(ctx)
|
||
}
|
||
})
|
||
|
||
s.OnEmergencyStop(func() {
|
||
// Cancel active orders
|
||
for _, symbol := range s.Symbols {
|
||
_ = s.orderExecutors[symbol].GracefulCancel(ctx)
|
||
}
|
||
// Close 100% position
|
||
//_ = s.ClosePosition(ctx, fixedpoint.One)
|
||
})
|
||
for _, symbol := range s.Symbols {
|
||
s.orderExecutors[symbol] = qbtrade.NewGeneralOrderExecutor(session, symbol, ID, s.InstanceID(), s.Positions[symbol])
|
||
s.orderExecutors[symbol].BindEnvironment(s.Environment)
|
||
_ = s.orderExecutors[symbol].GracefulCancel(ctx)
|
||
//s.orderExecutors[symbol].BindProfitStats(s.ProfitStats)
|
||
//s.orderExecutor.BindTradeStats(s.TradeStats)
|
||
|
||
//s.orderExecutors[symbol].TradeCollector().OnPositionUpdate(func(position *types.Position) {
|
||
// log.Infof("position is updated, symbol (%s): %+v", symbol, position)
|
||
//})
|
||
}
|
||
|
||
// AccountValueCalculator
|
||
//s.AccountValueCalculator = qbtrade.NewAccountValueCalculator(s.session, s.Market.QuoteCurrency)
|
||
|
||
s.session.MarketDataStream.OnKLineClosed(func(kline types.KLine) {
|
||
for _, symbol := range s.Symbols {
|
||
if kline.Symbol != symbol {
|
||
continue
|
||
}
|
||
if !s.ordered[symbol] {
|
||
// 在下一根k线时没有成交订单则取消所有订单
|
||
s.cancelOrders(ctx, symbol)
|
||
} else {
|
||
// 如果有订单则不再进行NR的计算
|
||
return
|
||
}
|
||
s.CalKLines[symbol] = []types.KLine{}
|
||
// 获取历史最近的4根K线
|
||
if !s.ReCalculate {
|
||
lines, err := s.session.Exchange.QueryKLines(ctx, symbol, s.Interval, types.KLineQueryOptions{Limit: s.NRCount})
|
||
s.CalKLines[symbol] = lines
|
||
if err != nil {
|
||
util.LogErr(err, fmt.Sprintf("failed to close position %s", symbol))
|
||
}
|
||
}
|
||
if len(s.CalKLines) < s.NRCount {
|
||
s.CalKLines[symbol] = append(s.CalKLines[symbol], kline)
|
||
} else {
|
||
s.OnKLineClosed(ctx, kline, symbol)
|
||
}
|
||
}
|
||
})
|
||
|
||
// 监听市场的交易事件
|
||
//session.MarketDataStream.OnMarketTrade(func(trade types.Trade) {
|
||
// // handle market trade event here
|
||
// fmt.Println(trade)
|
||
//})
|
||
session.UserDataStream.OnOrderUpdate(func(order types.Order) {
|
||
orderSymbol := order.Symbol
|
||
log.Infof("the order is: %+v,id is %d type is %s, status is %s", order, order.OrderID, order.Type, order.Status)
|
||
s.cancelSideOrder(ctx, orderSymbol)
|
||
if order.Status == types.OrderStatusFilled {
|
||
if order.Type == types.OrderTypeLimit && order.Side == types.SideTypeBuy {
|
||
log.Infof("the long order is filled: %+v,id is %d, symbol is %s, type is %s, status is %s",
|
||
order, order.OrderID, orderSymbol, order.Type, order.Status)
|
||
s.ordered[orderSymbol] = true
|
||
s.orderedSide[orderSymbol] = LongTag
|
||
qbtrade.Notify("订单成交通知:\n 币种:%s, 方向:%s, 价格:%s, 数量:%s", order.Symbol, LongTag, order.Price, order.Quantity)
|
||
}
|
||
if order.Type == types.OrderTypeLimit && order.Side == types.SideTypeSell {
|
||
log.Infof("the short order is filled: %+v,id is %d, symbol is %s, type is %s, status is %s",
|
||
order, order.OrderID, orderSymbol, order.Type, order.Status)
|
||
s.ordered[orderSymbol] = true
|
||
s.orderedSide[orderSymbol] = ShortTag
|
||
qbtrade.Notify("订单成交通知:\n 币种:%s, 方向:%s, 价格:%s, 数量:%s", order.Symbol, ShortTag, order.Price, order.Quantity)
|
||
}
|
||
if order.Type == types.OrderTypeMarket {
|
||
log.Infof("the loss or profit order is filled: %+v,id is %d, symbol is %s, type is %s, status is %s",
|
||
order, order.OrderID, orderSymbol, order.Type, order.Status)
|
||
qbtrade.Notify("订单止盈或止损通知:\n %s:", order.Symbol, order.Price)
|
||
s.ordered[orderSymbol] = false
|
||
}
|
||
} else {
|
||
log.Infof("the order is: %+v,id is %d, symbol is %s, type is %s, status is %s",
|
||
order, order.OrderID, orderSymbol, order.Type, order.Status)
|
||
}
|
||
})
|
||
|
||
session.UserDataStream.OnTradeUpdate(func(trade types.Trade) {
|
||
log.Infof("trade price %f, fee %f %s", trade.Price.Float64(), trade.Fee.Float64(), trade.FeeCurrency)
|
||
})
|
||
|
||
session.UserDataStream.OnBalanceUpdate(func(balances types.BalanceMap) {
|
||
log.Infof("balance update: %+v", balances)
|
||
})
|
||
|
||
qbtrade.OnShutdown(ctx, func(ctx context.Context, wg *sync.WaitGroup) {
|
||
defer wg.Done()
|
||
qbtrade.Sync(ctx, s)
|
||
})
|
||
|
||
return nil
|
||
}
|
||
|
||
func (s *Strategy) cancelOrders(ctx context.Context, symbol string) {
|
||
if len(s.orderExecutors[symbol].ActiveMakerOrders().Orders()) <= 0 {
|
||
return
|
||
}
|
||
log.Infof("the order is not filled, will cancel all orders")
|
||
if err := s.orderExecutors[symbol].GracefulCancel(ctx); err != nil {
|
||
log.WithError(err).Errorf("failed to cancel orders")
|
||
}
|
||
}
|
||
|
||
func (s *Strategy) placeOrders(ctx context.Context, symbol string) {
|
||
orders, err := s.generateOrders(ctx, symbol)
|
||
if err != nil {
|
||
log.WithError(err).Error(fmt.Sprintf("failed to generate orders (%s)", symbol))
|
||
return
|
||
}
|
||
log.Infof("orders: %+v", orders)
|
||
|
||
if s.DryRun {
|
||
log.Infof("dry run, not submitting orders (%s)", symbol)
|
||
return
|
||
}
|
||
|
||
createdOrders, err := s.orderExecutors[symbol].SubmitOrders(ctx, orders...)
|
||
if err != nil {
|
||
log.WithError(err).Error(fmt.Sprintf("failed to submit orders (%s)", symbol))
|
||
return
|
||
}
|
||
log.Infof("created orders (%s): %+v", symbol, createdOrders)
|
||
}
|
||
|
||
func (s *Strategy) generateOrders(ctx context.Context, symbol string) ([]types.SubmitOrder, error) {
|
||
var orders []types.SubmitOrder
|
||
// 卖价
|
||
sellPrice := fixedpoint.NewFromFloat(s.LastNRCandles[symbol].High.Float64())
|
||
// 买价
|
||
buyPrice := fixedpoint.NewFromFloat(s.LastNRCandles[symbol].Low.Float64())
|
||
|
||
buyQuantity := s.QuantityOrAmount.CalculateQuantity(buyPrice).Mul(s.Leverage)
|
||
sellQuantity := s.QuantityOrAmount.CalculateQuantity(sellPrice).Mul(s.Leverage)
|
||
log.Infof("generateOrders (%s), sellPrice is %s, sellQuantity is %s, "+
|
||
"buyPrice is %s, buyQuantity is %s", symbol, sellPrice, sellQuantity, buyPrice, buyQuantity)
|
||
s.ShortOrder[symbol] = types.SubmitOrder{
|
||
Symbol: symbol,
|
||
Side: types.SideTypeSell,
|
||
Type: s.OrderType,
|
||
Price: sellPrice,
|
||
PositionSide: types.PositionSideTypeShort,
|
||
Quantity: sellQuantity,
|
||
TimeInForce: types.TimeInForceGTC,
|
||
Market: s.markets[symbol],
|
||
Tag: ShortTag,
|
||
}
|
||
|
||
s.ShortProfitOrder[symbol] = types.SubmitOrder{
|
||
Symbol: symbol,
|
||
Side: types.SideTypeBuy,
|
||
Type: types.OrderTypeTakeProfitMarket,
|
||
PositionSide: types.PositionSideTypeShort,
|
||
StopPrice: sellPrice.Sub(sellPrice.Mul(s.ProfitRange)),
|
||
TimeInForce: types.TimeInForceGTC,
|
||
Market: s.markets[symbol],
|
||
Tag: ShortProfitTag,
|
||
ClosePosition: true,
|
||
}
|
||
|
||
s.ShortLossOrder[symbol] = types.SubmitOrder{
|
||
Symbol: symbol,
|
||
Side: types.SideTypeBuy,
|
||
Type: types.OrderTypeStopMarket,
|
||
PositionSide: types.PositionSideTypeShort,
|
||
StopPrice: buyPrice.Add(sellPrice.Mul(s.LossRange)),
|
||
TimeInForce: types.TimeInForceGTC,
|
||
Market: s.markets[symbol],
|
||
Tag: ShortLossTag,
|
||
ClosePosition: true,
|
||
}
|
||
|
||
s.LongOrder[symbol] = types.SubmitOrder{
|
||
Symbol: symbol,
|
||
Side: types.SideTypeBuy,
|
||
Type: s.OrderType,
|
||
Price: buyPrice,
|
||
PositionSide: types.PositionSideTypeLong,
|
||
Quantity: buyQuantity,
|
||
TimeInForce: types.TimeInForceGTC,
|
||
Market: s.markets[symbol],
|
||
Tag: LongTag,
|
||
}
|
||
|
||
s.LongProfitOrder[symbol] = types.SubmitOrder{
|
||
Symbol: symbol,
|
||
Side: types.SideTypeSell,
|
||
Type: types.OrderTypeTakeProfitMarket,
|
||
PositionSide: types.PositionSideTypeLong,
|
||
StopPrice: buyPrice.Add(buyPrice.Mul(s.ProfitRange)),
|
||
TimeInForce: types.TimeInForceGTC,
|
||
Market: s.markets[symbol],
|
||
Tag: LongProfitTag,
|
||
ClosePosition: true,
|
||
}
|
||
|
||
s.LongLossOrder[symbol] = types.SubmitOrder{
|
||
Symbol: symbol,
|
||
Side: types.SideTypeSell,
|
||
Type: types.OrderTypeStopMarket,
|
||
PositionSide: types.PositionSideTypeLong,
|
||
StopPrice: sellPrice.Sub(buyPrice.Mul(s.LossRange)),
|
||
TimeInForce: types.TimeInForceGTC,
|
||
Market: s.markets[symbol],
|
||
Tag: LongLossTag,
|
||
ClosePosition: true,
|
||
}
|
||
|
||
//// 挂空单
|
||
//orders = append(orders, s.ShortOrder[symbol])
|
||
//// 挂多单
|
||
//orders = append(orders, s.LongOrder[symbol])
|
||
//
|
||
//// 空单止盈
|
||
//orders = append(orders, s.ShortProfitOrder[symbol])
|
||
//// 空单止损
|
||
//orders = append(orders, s.ShortLossOrder[symbol])
|
||
//
|
||
//// 多单止盈
|
||
//orders = append(orders, s.LongProfitOrder[symbol])
|
||
//// 多单止损
|
||
//orders = append(orders, s.LongLossOrder[symbol])
|
||
|
||
if s.LastNRCandles[symbol].Open > s.LastNRCandles[symbol].Close {
|
||
// 挂空单
|
||
orders = append(orders, s.ShortOrder[symbol])
|
||
// 空单止盈
|
||
orders = append(orders, s.ShortProfitOrder[symbol])
|
||
// 空单止损
|
||
orders = append(orders, s.ShortLossOrder[symbol])
|
||
}
|
||
|
||
if s.LastNRCandles[symbol].Open < s.LastNRCandles[symbol].Close {
|
||
// 挂多单
|
||
orders = append(orders, s.LongOrder[symbol])
|
||
// 多单止盈
|
||
orders = append(orders, s.LongProfitOrder[symbol])
|
||
// 多单止损
|
||
orders = append(orders, s.LongLossOrder[symbol])
|
||
}
|
||
|
||
return orders, nil
|
||
}
|