mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-13 23:16:45 +00:00
* modernise: Run new gopls modernise tool against codebase
* Address shazbert's nits
* apichecker, gctcli: Simplify HTML scraping functions and improve depth limit handling
* refactor: Create minSyncInterval const and update order book limit handling for binance and binanceUS
* refactor: Various slice usage improvements and rename TODO
* tranches: Revert deleteByID changes due to performance decrease
Shazbert was a F1 driver in a past lifetime 🏎️
* tranches: Simply retrieve copy
Thanks to shazbert
* documentation: Sort contributors list by contributions
* tranches: Remove deadcode in deleteByID
146 lines
3.7 KiB
Go
146 lines
3.7 KiB
Go
package indicators
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
objects "github.com/d5/tengo/v2"
|
|
"github.com/thrasher-corp/gct-ta/indicators"
|
|
"github.com/thrasher-corp/gocryptotrader/gctscript/modules"
|
|
"github.com/thrasher-corp/gocryptotrader/gctscript/wrappers/validator"
|
|
)
|
|
|
|
// BBandsModule bollinger bands indicator commands
|
|
var BBandsModule = map[string]objects.Object{
|
|
"calculate": &objects.UserFunction{Name: "calculate", Value: bbands},
|
|
}
|
|
|
|
// BollingerBands is the string constant
|
|
const BollingerBands = "Bollinger Bands"
|
|
|
|
// BBands defines a custom Bollinger Bands indicator tengo object
|
|
type BBands struct {
|
|
objects.Array
|
|
Period int
|
|
STDDevUp, STDDevDown float64
|
|
MAType indicators.MaType
|
|
}
|
|
|
|
// TypeName returns the name of the custom type.
|
|
func (o *BBands) TypeName() string {
|
|
return BollingerBands
|
|
}
|
|
|
|
func bbands(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 6 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
r := new(BBands)
|
|
if validator.IsTestExecution.Load() == true {
|
|
return r, nil
|
|
}
|
|
|
|
ohlcIndicatorType, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, fmt.Errorf(modules.ErrParameterConvertFailed, ohlcIndicatorType)
|
|
}
|
|
|
|
selector, errIndSelector := ParseIndicatorSelector(ohlcIndicatorType)
|
|
if errIndSelector != nil {
|
|
return nil, errIndSelector
|
|
}
|
|
|
|
ohlcvInput := objects.ToInterface(args[1])
|
|
ohlcvInputData, valid := ohlcvInput.([]any)
|
|
if !valid {
|
|
return nil, fmt.Errorf(modules.ErrParameterConvertFailed, OHLCV)
|
|
}
|
|
|
|
ohlcvData := make([][]float64, 6)
|
|
var allErrors []string
|
|
for x := range ohlcvInputData {
|
|
var t []any
|
|
t, ok = ohlcvInputData[x].([]any)
|
|
if !ok {
|
|
return nil, errors.New("ohlcvInputData type assert failed")
|
|
}
|
|
if len(t) < 6 {
|
|
return nil, errors.New("ohlcvInputData invalid data length")
|
|
}
|
|
value, err := toFloat64(t[2])
|
|
if err != nil {
|
|
allErrors = append(allErrors, err.Error())
|
|
}
|
|
ohlcvData[2] = append(ohlcvData[2], value)
|
|
|
|
value, err = toFloat64(t[3])
|
|
if err != nil {
|
|
allErrors = append(allErrors, err.Error())
|
|
}
|
|
ohlcvData[3] = append(ohlcvData[3], value)
|
|
|
|
value, err = toFloat64(t[4])
|
|
if err != nil {
|
|
allErrors = append(allErrors, err.Error())
|
|
}
|
|
ohlcvData[4] = append(ohlcvData[4], value)
|
|
|
|
value, err = toFloat64(t[5])
|
|
if err != nil {
|
|
allErrors = append(allErrors, err.Error())
|
|
}
|
|
ohlcvData[5] = append(ohlcvData[5], value)
|
|
}
|
|
|
|
inTimePeriod, ok := objects.ToInt(args[2])
|
|
if !ok {
|
|
allErrors = append(allErrors, fmt.Sprintf(modules.ErrParameterConvertFailed, inTimePeriod))
|
|
}
|
|
|
|
inNbDevUp, ok := objects.ToFloat64(args[3])
|
|
if !ok {
|
|
allErrors = append(allErrors, fmt.Sprintf(modules.ErrParameterConvertFailed, inNbDevUp))
|
|
}
|
|
|
|
inNbDevDn, ok := objects.ToFloat64(args[4])
|
|
if !ok {
|
|
allErrors = append(allErrors, fmt.Sprintf(modules.ErrParameterConvertFailed, inNbDevDn))
|
|
}
|
|
|
|
inMAType, ok := objects.ToString(args[5])
|
|
if !ok {
|
|
allErrors = append(allErrors, fmt.Sprintf(modules.ErrParameterConvertFailed, inMAType))
|
|
}
|
|
|
|
if len(allErrors) > 0 {
|
|
return nil, errors.New(strings.Join(allErrors, ", "))
|
|
}
|
|
|
|
MAType, err := ParseMAType(inMAType)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
r.Period = inTimePeriod
|
|
r.STDDevDown = inNbDevDn
|
|
r.STDDevUp = inNbDevUp
|
|
r.MAType = MAType
|
|
|
|
retUpper, retMiddle, retLower := indicators.BBANDS(ohlcvData[selector], inTimePeriod, inNbDevDn, inNbDevDn, MAType)
|
|
for x := range retMiddle {
|
|
temp := &objects.Array{}
|
|
temp.Value = append(temp.Value, &objects.Float{Value: retMiddle[x]})
|
|
if retUpper != nil {
|
|
temp.Value = append(temp.Value, &objects.Float{Value: retUpper[x]})
|
|
}
|
|
if retLower != nil {
|
|
temp.Value = append(temp.Value, &objects.Float{Value: retLower[x]})
|
|
}
|
|
r.Value = append(r.Value, temp)
|
|
}
|
|
|
|
return r, nil
|
|
}
|