Files
gocryptotrader/gctscript/modules/ta/indicators/ema.go
Adrian Gallagher 9a4eb9de84 CI: Fix golangci-lint linter issues, add prealloc linter and bump version depends for Go 1.18 (#915)
* Bump CI versions

* Specifically set go version as 1.17.x bumps it to 1.18

* Another

* Adjust AppVeyor

* Part 1 of linter issues

* Part 2

* Fix various linters and improvements

* Part 3

* Finishing touches

* Tests and EqualFold

* Fix nitterinos plus bonus requester jobs bump for exchanges with large number of tests

* Fix nitterinos and bump golangci-lint timeout for AppVeyor

* Address nits, ensure all books are returned on err due to syncer regression

* Fix the wiggins

* Fix duplication

* Fix nitterinos
2022-04-20 13:45:15 +10:00

85 lines
2.0 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"
)
// EMAModule EMA indicator commands
var EMAModule = map[string]objects.Object{
"calculate": &objects.UserFunction{Name: "calculate", Value: ema},
}
// ExponentialMovingAverage is the string constant
const ExponentialMovingAverage = "Exponential Moving Average"
// EMA defines a custom Exponential Moving Average indicator tengo object
type EMA struct {
objects.Array
Period int
}
// TypeName returns the name of the custom type.
func (o *EMA) TypeName() string {
return ExponentialMovingAverage
}
func ema(args ...objects.Object) (objects.Object, error) {
if len(args) != 2 {
return nil, objects.ErrWrongNumArguments
}
r := new(EMA)
if validator.IsTestExecution.Load() == true {
return r, nil
}
ohlcvInput := objects.ToInterface(args[0])
ohlcvInputData, valid := ohlcvInput.([]interface{})
if !valid {
return nil, fmt.Errorf(modules.ErrParameterConvertFailed, OHLCV)
}
ohlcvClose := make([]float64, len(ohlcvInputData))
var allErrors []string
for x := range ohlcvInputData {
t, ok := ohlcvInputData[x].([]interface{})
if !ok {
return nil, errors.New("ohlcvInputData type assert failed")
}
if len(t) < 5 {
return nil, errors.New("ohlcvInputData invalid data length")
}
value, err := toFloat64(t[4])
if err != nil {
allErrors = append(allErrors, err.Error())
}
ohlcvClose[x] = value
}
inTimePeriod, ok := objects.ToInt(args[1])
if !ok {
allErrors = append(allErrors, fmt.Sprintf(modules.ErrParameterConvertFailed, inTimePeriod))
}
if len(allErrors) > 0 {
return nil, errors.New(strings.Join(allErrors, ", "))
}
r.Period = inTimePeriod
ret := indicators.EMA(ohlcvClose, inTimePeriod)
for x := range ret {
r.Value = append(r.Value, &objects.Float{Value: ret[x]})
}
return r, nil
}