mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-14 07:26:47 +00:00
* initial concept of a nice validation tester for exchanges * adds some datahandler design * expand testing * more tests and fixes * minor end of day fix for bithumb * fixes implementation issues * more test coverage and improvements, but not sure if i should continue * fix more wrapper implementations * adds error type, more fixes * changes signature, fixes implementations * fixes more wrapper implementations * one more bit * more cleanup * WOW things work? * lintle 1/1337 * mini bump * fixes all linting * neaten * GetOrderInfo+ asset pair fixes+improvements * adds new websocket test * expand ws testing * fix bug, expand tests, improve implementation * code coverage of a lot of new codes * fixes everything * reverts accidental changes * minor fixes from reviewing code * removes Bitfinex cancelBatchOrder implementation * fixes dumb baby typo for babies * mini nit fixes * so many nits to address * addresses all the nits * Titlecase * switcheroo * removes websocket testing for now * fix appveyor, minor test fix * fixes typo, re-kindles killed kode * skip binance wrapper tests when running CI * expired context, huobi okx fixes * kodespull * fix ordering * time fix because why not * fix exmo, others * hopefully this fixes all of my life's problems * last thing today * huobi, more like hypotrophy * golangci-lint, more like mypooroldknee-splint * fix huobi times by removing them * should fix okx currency issues * blocks the application * adds last little contingency for pairs * addresses most nits and new problems * lovely fixed before seeing why okx sucks * fixes issues with okx websocket * the classic receieieivaier * lintle * adds test and fixes existing tests * expands error handling messages during setup * fixes dumb okx bugs introduced * quick fix for lint and exmo * fixes nixes * fix exmo deposit issue * lint * fixes issue with extra asset runs missing * fix surprise race * all the lint and merge fixes * fixes surprise bugs in OKx * fixes issues with times and chains * fixing all the merge stuff * merge fix * rm logs and a panic potential * lovely lint lament * an easy demonstration of scenario, but not of initial purpose * put it in the bin * Revert "put it in the bin" This reverts commit 15c6490f713233d43f10957367fcbf18e3818bdd. * re-add after immediate error popup * fix mini poor test design * okx okay * merge fixes * fixes issues discovered in lovely test * I FORGOT TO COMMIT THIS * nit fixaroonaboo * forgoetten test fix * revert old okx asset intrument work * fixes * revert problems I didnt understand. update bybit * fix merge bugs * test cleanup * further improvements * reshuffle and lint * rm redundant CI_TEST by rm the CI_TEST field that is redundant * path fix * move to its own section, dont run on 32 bit + appveyor * lint * fix lbank * address nits * let it rip * fix failing test time range * niteroo boogaloo * mod tidy, use common.SimpleTimeFormat
509 lines
12 KiB
Go
509 lines
12 KiB
Go
package gct
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
objects "github.com/d5/tengo/v2"
|
|
"github.com/thrasher-corp/gocryptotrader/common"
|
|
"github.com/thrasher-corp/gocryptotrader/common/file"
|
|
"github.com/thrasher-corp/gocryptotrader/gctscript/modules/ta/indicators"
|
|
"github.com/thrasher-corp/gocryptotrader/log"
|
|
)
|
|
|
|
var commonModule = map[string]objects.Object{
|
|
"writeascsv": &objects.UserFunction{Name: "writeascsv", Value: WriteAsCSV},
|
|
}
|
|
|
|
// OutputDir is the default script output directory
|
|
var OutputDir string
|
|
|
|
// WriteAsCSV takes in a slice matrix to save to file
|
|
func WriteAsCSV(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) == 0 {
|
|
return nil, errors.New("cannot write to file, no data present")
|
|
}
|
|
|
|
var bucket [][]string
|
|
var err error
|
|
var target string
|
|
for i := range args {
|
|
if args[i] == nil {
|
|
return nil, errors.New("data is nil")
|
|
}
|
|
var front bool
|
|
var temp [][]string
|
|
switch args[i].TypeName() {
|
|
case indicators.AverageTrueRange:
|
|
temp, err = convertATR(args[i])
|
|
case indicators.BollingerBands:
|
|
temp, err = convertBollingerBands(args[i])
|
|
case indicators.ExponentialMovingAverage:
|
|
temp, err = convertEMA(args[i])
|
|
case indicators.MovingAverageConvergenceDivergence:
|
|
temp, err = convertMACD(args[i])
|
|
case indicators.MoneyFlowIndex:
|
|
temp, err = convertMFI(args[i])
|
|
case indicators.OnBalanceVolume:
|
|
temp, err = convertOBV(args[i])
|
|
case indicators.RelativeStrengthIndex:
|
|
temp, err = convertRSI(args[i])
|
|
case indicators.SimpleMovingAverage:
|
|
temp, err = convertSMA(args[i])
|
|
case indicators.CorrelationCoefficient:
|
|
temp, err = convertCorrelationCoefficient(args[i])
|
|
case indicators.OHLCV:
|
|
temp, err = convertOHLCV(args[i])
|
|
front = true
|
|
case "scriptContext":
|
|
if target != "" {
|
|
return nil, fmt.Errorf("filename already set, extra string %v cannot be processed", args[i])
|
|
}
|
|
scriptCtx, ok := objects.ToInterface(args[i]).(*Context)
|
|
if !ok {
|
|
return nil, common.GetTypeAssertError("*gct.Context", args[i])
|
|
}
|
|
|
|
scriptDetails, ok := scriptCtx.Value["script"]
|
|
if !ok {
|
|
return nil, errors.New("no script details")
|
|
}
|
|
|
|
target, ok = objects.ToString(scriptDetails)
|
|
if !ok {
|
|
return nil, errors.New("failed to convert incoming output to string")
|
|
}
|
|
|
|
target = processTarget(target)
|
|
case "string":
|
|
if target != "" {
|
|
return nil, fmt.Errorf("filename already set, extra string %v cannot be processed", args[i])
|
|
}
|
|
var ok bool
|
|
target, ok = objects.ToString(args[i])
|
|
if !ok {
|
|
return nil, errors.New("failed to convert incoming output to string")
|
|
}
|
|
|
|
if target == "" {
|
|
return nil, errors.New("script context details not specified")
|
|
}
|
|
|
|
target = processTarget(target)
|
|
default:
|
|
err = fmt.Errorf("%s type is not handled", args[i].TypeName())
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if front {
|
|
var newBucket [][]string
|
|
newBucket = append(newBucket, temp...)
|
|
for x := range bucket {
|
|
newBucket[x] = append(newBucket[x], bucket[x]...)
|
|
}
|
|
|
|
bucket = newBucket
|
|
front = false
|
|
continue
|
|
}
|
|
|
|
if len(bucket) == 0 {
|
|
bucket = temp
|
|
} else {
|
|
for i := range temp {
|
|
bucket[i] = append(bucket[i], temp[i]...)
|
|
}
|
|
}
|
|
}
|
|
|
|
if target == "" {
|
|
return nil, errors.New("filename unset please set in writeascsv as ctx or client defined filename")
|
|
}
|
|
|
|
err = file.WriteAsCSV(target, bucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
log.Debugf(log.GCTScriptMgr,
|
|
"CSV file successfully saved to: %s",
|
|
target)
|
|
return nil, nil
|
|
}
|
|
|
|
func convertATR(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.ATR)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.AverageTrueRange,
|
|
},
|
|
{
|
|
fmt.Sprintf("Period:%d", obj.Period),
|
|
},
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertBollingerBands(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.BBands)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
upperS := fmt.Sprintf("Upper_Band (NBDevUp:%f)", obj.STDDevUp)
|
|
lowerS := fmt.Sprintf("Lower_band(NBDevDown:%f)", obj.STDDevDown)
|
|
middleS := fmt.Sprintf("Middle_Band (Period:%d)", obj.Period)
|
|
MAType := "MA_TYPE:SMA"
|
|
if obj.MAType != 0 {
|
|
MAType = "MA_TYPE:EMA"
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.BollingerBands, "", MAType,
|
|
},
|
|
{
|
|
upperS, middleS, lowerS,
|
|
},
|
|
}
|
|
|
|
for x := range obj.Value {
|
|
element := obj.Value[x].Iterate()
|
|
var upper, middle, lower string
|
|
for i := 0; element.Next(); i++ {
|
|
switch i {
|
|
case 0:
|
|
upper, ok = objects.ToString(element.Value())
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
case 1:
|
|
middle, ok = objects.ToString(element.Value())
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
case 2:
|
|
lower, ok = objects.ToString(element.Value())
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
}
|
|
}
|
|
bucket = append(bucket, []string{upper, middle, lower})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertEMA(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.EMA)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.ExponentialMovingAverage,
|
|
},
|
|
{
|
|
fmt.Sprintf("Period:%d", obj.Period),
|
|
},
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertMACD(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.MACD)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.MovingAverageConvergenceDivergence,
|
|
fmt.Sprintf("Period:%d Fast:%d Slow:%d",
|
|
obj.Period,
|
|
obj.PeriodFast,
|
|
obj.PeriodSlow),
|
|
"",
|
|
},
|
|
{
|
|
"MACD", "Signal", "Histogram",
|
|
},
|
|
}
|
|
|
|
for x := range obj.Value {
|
|
element := obj.Value[x].Iterate()
|
|
var macd, signal, hist string
|
|
for i := 0; element.Next(); i++ {
|
|
switch i {
|
|
case 0:
|
|
macd, ok = objects.ToString(element.Value())
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
case 1:
|
|
signal, ok = objects.ToString(element.Value())
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
case 2:
|
|
hist, ok = objects.ToString(element.Value())
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
}
|
|
}
|
|
bucket = append(bucket, []string{macd, signal, hist})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertMFI(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.MFI)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.MoneyFlowIndex,
|
|
},
|
|
{
|
|
fmt.Sprintf("Period:%d", obj.Period),
|
|
},
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertOBV(a objects.Object) ([][]string, error) {
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.OnBalanceVolume,
|
|
},
|
|
{
|
|
"",
|
|
},
|
|
}
|
|
|
|
obj, ok := objects.ToInterface(a).(*indicators.OBV)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertRSI(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.RSI)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.RelativeStrengthIndex,
|
|
},
|
|
{
|
|
fmt.Sprintf("Period:%d", obj.Period),
|
|
},
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertSMA(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.SMA)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.SimpleMovingAverage,
|
|
},
|
|
{
|
|
fmt.Sprintf("Period:%d", obj.Period),
|
|
},
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertCorrelationCoefficient(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*indicators.Correlation)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.CorrelationCoefficient,
|
|
},
|
|
{
|
|
fmt.Sprintf("Period:%d", obj.Period),
|
|
},
|
|
}
|
|
|
|
var val string
|
|
for i := range obj.Value {
|
|
val, ok = objects.ToString(obj.Value[i])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
bucket = append(bucket, []string{val})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func convertOHLCV(a objects.Object) ([][]string, error) {
|
|
obj, ok := objects.ToInterface(a).(*OHLCV)
|
|
if !ok {
|
|
return nil, errors.New("casting failure")
|
|
}
|
|
|
|
exchange, ok := objects.ToString(obj.Value["exchange"])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
|
|
pair, ok := objects.ToString(obj.Value["pair"])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
|
|
asset, ok := objects.ToString(obj.Value["asset"])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
|
|
interval, ok := objects.ToString(obj.Value["intervals"])
|
|
if !ok {
|
|
return nil, errors.New("cannot convert object to string")
|
|
}
|
|
|
|
var bucket = [][]string{
|
|
{
|
|
indicators.OHLCV, "Exchange:" + exchange, pair, asset, interval, "",
|
|
},
|
|
{
|
|
"Date", "Volume", "Open", "High", "Low", "Close",
|
|
},
|
|
}
|
|
|
|
candles, ok := obj.Value["candles"]
|
|
if !ok {
|
|
return nil, errors.New("candles not found in object map")
|
|
}
|
|
|
|
data := candles.Iterate()
|
|
|
|
for data.Next() {
|
|
var date, open, high, low, closed, volume string
|
|
candle := data.Value().Iterate()
|
|
for i := 0; candle.Next(); i++ {
|
|
switch i {
|
|
case 0:
|
|
date, ok = objects.ToString(candle.Value())
|
|
case 1:
|
|
open, ok = objects.ToString(candle.Value())
|
|
case 2:
|
|
high, ok = objects.ToString(candle.Value())
|
|
case 3:
|
|
low, ok = objects.ToString(candle.Value())
|
|
case 4:
|
|
closed, ok = objects.ToString(candle.Value())
|
|
case 5:
|
|
volume, ok = objects.ToString(candle.Value())
|
|
}
|
|
if !ok {
|
|
return nil, errors.New("failed to convert")
|
|
}
|
|
}
|
|
bucket = append(bucket, []string{date, volume, open, high, low, closed})
|
|
}
|
|
return bucket, nil
|
|
}
|
|
|
|
func processTarget(target string) string {
|
|
// Removes file transversal
|
|
target = filepath.Base(target)
|
|
|
|
// checks to see if file is context defined, if not it will allow
|
|
// a client defined filename and append a date, forces the use of
|
|
// .csv file extension
|
|
switch {
|
|
case filepath.Ext(target) != ".csv" && strings.Contains(target, common.GctExt):
|
|
target += ".csv"
|
|
case filepath.Ext(target) == ".csv":
|
|
s := strings.Split(target, ".")
|
|
if len(s) == 2 {
|
|
target = s[0] + "-" + strconv.FormatInt(time.Now().UnixNano(), 10) + ".csv"
|
|
}
|
|
default:
|
|
target += "-" + strconv.FormatInt(time.Now().UnixNano(), 10) + ".csv"
|
|
}
|
|
return filepath.Join(OutputDir, target)
|
|
}
|