package techsignal import ( "context" "errors" "strings" "time" "github.com/sirupsen/logrus" "git.qtrade.icu/lychiyu/qbtrade/pkg/exchange/binance" "git.qtrade.icu/lychiyu/qbtrade/pkg/fixedpoint" "git.qtrade.icu/lychiyu/qbtrade/pkg/qbtrade" "git.qtrade.icu/lychiyu/qbtrade/pkg/types" ) const ID = "techsignal" var log = logrus.WithField("strategy", ID) func init() { // Register the pointer of the strategy struct, // so that qbtrade knows what struct to be used to unmarshal the configs (YAML or JSON) // Note: built-in strategies need to imported manually in the qbtrade cmd package. qbtrade.RegisterStrategy(ID, &Strategy{}) } type Strategy struct { // These fields will be filled from the config file (it translates YAML to JSON) Symbol string `json:"symbol"` Market types.Market `json:"-"` FundingRate *struct { High fixedpoint.Value `json:"high"` Neutral fixedpoint.Value `json:"neutral"` DiffThreshold fixedpoint.Value `json:"diffThreshold"` } `json:"fundingRate"` SupportDetection []struct { Interval types.Interval `json:"interval"` // MovingAverageType is the moving average indicator type that we want to use, // it could be SMA or EWMA MovingAverageType string `json:"movingAverageType"` // MovingAverageInterval is the interval of k-lines for the moving average indicator to calculate, // it could be "1m", "5m", "1h" and so on. note that, the moving averages are calculated from // the k-line data we subscribed MovingAverageInterval types.Interval `json:"movingAverageInterval"` // MovingAverageWindow is the number of the window size of the moving average indicator. // The number of k-lines in the window. generally used window sizes are 7, 25 and 99 in the TradingView. MovingAverageWindow int `json:"movingAverageWindow"` MinVolume fixedpoint.Value `json:"minVolume"` MinQuoteVolume fixedpoint.Value `json:"minQuoteVolume"` } `json:"supportDetection"` } func (s *Strategy) ID() string { return ID } func (s *Strategy) Subscribe(session *qbtrade.ExchangeSession) { // session.Subscribe(types.BookChannel, s.Symbol, types.SubscribeOptions{}) for _, detection := range s.SupportDetection { session.Subscribe(types.KLineChannel, s.Symbol, types.SubscribeOptions{ Interval: detection.Interval, }) session.Subscribe(types.KLineChannel, s.Symbol, types.SubscribeOptions{ Interval: detection.MovingAverageInterval, }) } } func (s *Strategy) Validate() error { if len(s.Symbol) == 0 { return errors.New("symbol is required") } return nil } func (s *Strategy) listenToFundingRate(ctx context.Context, exchange *binance.Exchange) { var previousIndex, fundingRate24HoursLowIndex *types.PremiumIndex fundingRateTicker := time.NewTicker(1 * time.Hour) defer fundingRateTicker.Stop() for { select { case <-ctx.Done(): return case <-fundingRateTicker.C: index, err := exchange.QueryPremiumIndex(ctx, s.Symbol) if err != nil { log.WithError(err).Error("can not query last funding rate") continue } fundingRate := index.LastFundingRate if fundingRate.Compare(s.FundingRate.High) >= 0 { qbtrade.Notify("%s funding rate %s is too high! threshold %s", s.Symbol, fundingRate.Percentage(), s.FundingRate.High.Percentage(), ) } else { if previousIndex != nil { if s.FundingRate.DiffThreshold.IsZero() { // 0.6% s.FundingRate.DiffThreshold = fixedpoint.NewFromFloat(0.006 * 0.01) } diff := fundingRate.Sub(previousIndex.LastFundingRate) if diff.Abs().Compare(s.FundingRate.DiffThreshold) > 0 { qbtrade.Notify("%s funding rate changed %s, current funding rate %s", s.Symbol, diff.SignedPercentage(), fundingRate.Percentage(), ) } } } previousIndex = index if fundingRate24HoursLowIndex != nil { if fundingRate24HoursLowIndex.Time.Before(time.Now().Add(24 * time.Hour)) { fundingRate24HoursLowIndex = index } if fundingRate.Compare(fundingRate24HoursLowIndex.LastFundingRate) < 0 { fundingRate24HoursLowIndex = index } } else { fundingRate24HoursLowIndex = index } } } } func (s *Strategy) Run(ctx context.Context, orderExecutor qbtrade.OrderExecutor, session *qbtrade.ExchangeSession) error { standardIndicatorSet := session.StandardIndicatorSet(s.Symbol) if s.FundingRate != nil { if binanceExchange, ok := session.Exchange.(*binance.Exchange); ok { go s.listenToFundingRate(ctx, binanceExchange) } else { log.Error("exchange does not support funding rate api") } } session.MarketDataStream.OnKLineClosed(func(kline types.KLine) { // skip k-lines from other symbols if kline.Symbol != s.Symbol { return } for _, detection := range s.SupportDetection { if kline.Interval != detection.Interval { continue } closePrice := kline.GetClose() var ma types.Float64Indicator switch strings.ToLower(detection.MovingAverageType) { case "sma": ma = standardIndicatorSet.SMA(types.IntervalWindow{ Interval: detection.MovingAverageInterval, Window: detection.MovingAverageWindow, }) case "ema", "ewma": ma = standardIndicatorSet.EWMA(types.IntervalWindow{ Interval: detection.MovingAverageInterval, Window: detection.MovingAverageWindow, }) default: ma = standardIndicatorSet.EWMA(types.IntervalWindow{ Interval: detection.MovingAverageInterval, Window: detection.MovingAverageWindow, }) } var lastMA = ma.Last(0) // skip if the closed price is above the moving average if closePrice.Float64() > lastMA { log.Infof("skip %s support closed price %f > last ma %f", s.Symbol, closePrice.Float64(), lastMA) return } prettyBaseVolume := s.Market.BaseCurrencyFormatter() prettyQuoteVolume := s.Market.QuoteCurrencyFormatter() if detection.MinVolume.Sign() > 0 && kline.Volume.Compare(detection.MinVolume) > 0 { qbtrade.Notify("Detected %s %s support base volume %s > min base volume %s, quote volume %s", s.Symbol, detection.Interval.String(), prettyBaseVolume.FormatMoney(kline.Volume.Trunc()), prettyBaseVolume.FormatMoney(detection.MinVolume.Trunc()), prettyQuoteVolume.FormatMoney(kline.QuoteVolume.Trunc()), ) qbtrade.Notify(kline) } else if detection.MinQuoteVolume.Sign() > 0 && kline.QuoteVolume.Compare(detection.MinQuoteVolume) > 0 { qbtrade.Notify("Detected %s %s support quote volume %s > min quote volume %s, base volume %s", s.Symbol, detection.Interval.String(), prettyQuoteVolume.FormatMoney(kline.QuoteVolume.Trunc()), prettyQuoteVolume.FormatMoney(detection.MinQuoteVolume.Trunc()), prettyBaseVolume.FormatMoney(kline.Volume.Trunc()), ) qbtrade.Notify(kline) } } }) return nil }