Files
gocryptotrader/gctscript/modules/ta/indicators/bbands.go
Adrian Gallagher f0d45aa1d2 golangci-lint/CI: Bump versions and introduce new linters (#798)
* golangci-lint/CI: Bump versions

Fix remaining linter issues

* Specifically set AppVeyor version

* Fix the infamous typos 👀

* Add go env cmd to AppVeyor

* Add go version cmd to AppVeyor

* Specify AppVeyor image, adjust linters

* Update go get to go install due to deprecation

* Bump golangci-lint timeout time for AppVeyor

* Change NW contract to NQ

* Address nitters

* GetRandomPair -> Pair{}

* Address nits

* Address time nitterinos plus additional tweaks

* More time inception upgrades!

* Bending time and space
2021-10-14 16:38:53 +11:00

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.([]interface{})
if !valid {
return nil, fmt.Errorf(modules.ErrParameterConvertFailed, OHLCV)
}
ohlcvData := make([][]float64, 6)
var allErrors []string
for x := range ohlcvInputData {
var t []interface{}
t, ok = ohlcvInputData[x].([]interface{})
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
}