autotrader/trader.go
2023-05-15 19:52:40 -05:00

161 lines
3.9 KiB
Go

package autotrader
import (
"fmt"
"log"
"os"
"strconv"
"strings"
"time"
"github.com/go-co-op/gocron"
"github.com/rocketlaunchr/dataframe-go"
)
// Trader acts as the primary interface to the broker and strategy. To the strategy, it provides all the information
// about the current state of the market and the portfolio. To the broker, it provides the orders to be executed and
// requests for the current state of the portfolio.
type Trader struct {
Broker Broker
Strategy Strategy
Symbol string
Frequency string
CandlesToKeep int
Log *log.Logger
EOF bool
data *DataFrame
sched *gocron.Scheduler
stats *DataFrame // Performance (financial) reporting and statistics.
}
func (t *Trader) Data() *DataFrame {
return t.data
}
func (t *Trader) Stats() *DataFrame {
return t.stats
}
// Run starts the trader. This is a blocking call.
func (t *Trader) Run() {
t.sched = gocron.NewScheduler(time.UTC)
capitalizedFreq := strings.ToUpper(t.Frequency)
if strings.HasPrefix(capitalizedFreq, "S") {
seconds, err := strconv.Atoi(t.Frequency[1:])
if err != nil {
panic(err)
}
t.sched.Every(seconds).Seconds()
} else if strings.HasPrefix(capitalizedFreq, "M") {
minutes, err := strconv.Atoi(t.Frequency[1:])
if err != nil {
panic(err)
}
t.sched.Every(minutes).Minutes()
} else if strings.HasPrefix(capitalizedFreq, "H") {
hours, err := strconv.Atoi(t.Frequency[1:])
if err != nil {
panic(err)
}
t.sched.Every(hours).Hours()
} else {
switch capitalizedFreq {
case "D":
t.sched.Every(1).Day()
case "W":
t.sched.Every(1).Day()
case "M":
t.sched.Every(1).Day()
default:
panic(fmt.Sprintf("invalid frequency: %s", t.Frequency))
}
}
t.sched.Do(t.Tick) // Set the function to be run when the interval repeats.
t.Init()
t.sched.StartBlocking()
}
func (t *Trader) Init() {
t.Strategy.Init(t)
t.stats = NewDataFrame(
NewDataSeries(dataframe.NewSeriesTime("Date", nil)),
NewDataSeries(dataframe.NewSeriesFloat64("Equity", nil)),
)
}
// Tick updates the current state of the market and runs the strategy.
func (t *Trader) Tick() {
t.fetchData() // Fetch the latest candlesticks from the broker.
// t.Log.Println(t.data.Close(-1))
t.Strategy.Next(t) // Run the strategy.
// Update the stats.
t.stats.PushValues(map[string]interface{}{
"Date": t.data.Date(-1),
"Equity": t.Broker.NAV(),
})
}
func (t *Trader) fetchData() {
var err error
t.data, err = t.Broker.Candles(t.Symbol, t.Frequency, t.CandlesToKeep)
if err == ErrEOF {
t.EOF = true
t.Log.Println("End of data")
if t.sched != nil && t.sched.IsRunning() {
t.sched.Clear()
}
} else if err != nil {
panic(err) // TODO: implement safe shutdown procedure
}
}
func (t *Trader) Buy(units float64) {
t.Log.Printf("Buy %f units", units)
t.closeOrdersAndPositions()
t.Broker.MarketOrder(t.Symbol, units, 0.0, 0.0)
}
func (t *Trader) Sell(units float64) {
t.Log.Printf("Sell %f units", units)
t.closeOrdersAndPositions()
t.Broker.MarketOrder(t.Symbol, -units, 0.0, 0.0)
}
func (t *Trader) closeOrdersAndPositions() {
for _, order := range t.Broker.OpenOrders() {
if order.Symbol() == t.Symbol {
order.Cancel()
}
}
for _, position := range t.Broker.OpenPositions() {
if position.Symbol() == t.Symbol {
position.Close()
}
}
}
type TraderConfig struct {
Broker Broker
Strategy Strategy
Symbol string
Frequency string
CandlesToKeep int
}
// NewTrader initializes a new Trader which can be used for live trading or backtesting.
func NewTrader(config TraderConfig) *Trader {
logger := log.New(os.Stdout, "autotrader: ", log.LstdFlags|log.Lshortfile)
return &Trader{
Broker: config.Broker,
Strategy: config.Strategy,
Symbol: config.Symbol,
Frequency: config.Frequency,
CandlesToKeep: config.CandlesToKeep,
Log: logger,
stats: NewDataFrame(),
}
}