qbtrade/pkg/risk/dynamicrisk/dynamic_exposure.go
2024-06-27 22:42:38 +08:00

87 lines
2.8 KiB
Go

package dynamicrisk
import (
"git.qtrade.icu/lychiyu/qbtrade/pkg/fixedpoint"
"git.qtrade.icu/lychiyu/qbtrade/pkg/indicator"
"git.qtrade.icu/lychiyu/qbtrade/pkg/qbtrade"
"git.qtrade.icu/lychiyu/qbtrade/pkg/types"
"github.com/pkg/errors"
log "github.com/sirupsen/logrus"
"math"
)
type DynamicExposure struct {
// BollBandExposure calculates the max exposure with the Bollinger Band
BollBandExposure *DynamicExposureBollBand `json:"bollBandExposure"`
}
// Initialize dynamic exposure
func (d *DynamicExposure) Initialize(symbol string, session *qbtrade.ExchangeSession) {
switch {
case d.BollBandExposure != nil:
d.BollBandExposure.initialize(symbol, session)
}
}
func (d *DynamicExposure) IsEnabled() bool {
return d.BollBandExposure != nil
}
// GetMaxExposure returns the max exposure
func (d *DynamicExposure) GetMaxExposure(price float64, trend types.Direction) (maxExposure fixedpoint.Value, err error) {
switch {
case d.BollBandExposure != nil:
return d.BollBandExposure.getMaxExposure(price, trend)
default:
return fixedpoint.Zero, errors.New("dynamic exposure is not enabled")
}
}
// DynamicExposureBollBand calculates the max exposure with the Bollinger Band
type DynamicExposureBollBand struct {
// DynamicExposureBollBandScale is used to define the exposure range with the given percentage.
DynamicExposureBollBandScale *qbtrade.PercentageScale `json:"dynamicExposurePositionScale"`
types.IntervalWindowBandWidth
dynamicExposureBollBand *indicator.BOLL
}
// initialize dynamic exposure with Bollinger Band
func (d *DynamicExposureBollBand) initialize(symbol string, session *qbtrade.ExchangeSession) {
d.dynamicExposureBollBand = session.StandardIndicatorSet(symbol).BOLL(d.IntervalWindow, d.BandWidth)
// Subscribe kline
session.Subscribe(types.KLineChannel, symbol, types.SubscribeOptions{
Interval: d.dynamicExposureBollBand.Interval,
})
}
// getMaxExposure returns the max exposure
func (d *DynamicExposureBollBand) getMaxExposure(price float64, trend types.Direction) (fixedpoint.Value, error) {
downBand := d.dynamicExposureBollBand.DownBand.Last(0)
upBand := d.dynamicExposureBollBand.UpBand.Last(0)
sma := d.dynamicExposureBollBand.SMA.Last(0)
log.Infof("dynamicExposureBollBand bollinger band: up %f sma %f down %f", upBand, sma, downBand)
bandPercentage := 0.0
if price < sma {
// should be negative percentage
bandPercentage = (price - sma) / math.Abs(sma-downBand)
} else if price > sma {
// should be positive percentage
bandPercentage = (price - sma) / math.Abs(upBand-sma)
}
// Reverse if downtrend
if trend == types.DirectionDown {
bandPercentage = 0 - bandPercentage
}
v, err := d.DynamicExposureBollBandScale.Scale(bandPercentage)
if err != nil {
return fixedpoint.Zero, err
}
return fixedpoint.NewFromFloat(v), nil
}