[golang] Bootstrap contracting and refactor the agent code
This commit is contained in:
parent
8fbd912414
commit
3dad3f60f2
8 changed files with 158 additions and 106 deletions
|
@ -2,12 +2,12 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"log/slog"
|
"log/slog"
|
||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
|
|
||||||
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/agent"
|
||||||
"git.adyxax.org/adyxax/spacetraders/golang/pkg/api"
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/api"
|
||||||
"git.adyxax.org/adyxax/spacetraders/golang/pkg/database"
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/database"
|
||||||
)
|
)
|
||||||
|
@ -37,7 +37,7 @@ func main() {
|
||||||
|
|
||||||
apiClient := api.NewClient(ctx)
|
apiClient := api.NewClient(ctx)
|
||||||
defer apiClient.Close()
|
defer apiClient.Close()
|
||||||
if err := run(
|
if err := agent.Run(
|
||||||
apiClient,
|
apiClient,
|
||||||
db,
|
db,
|
||||||
os.Getenv,
|
os.Getenv,
|
||||||
|
@ -49,76 +49,3 @@ func main() {
|
||||||
os.Exit(2)
|
os.Exit(2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func run(
|
|
||||||
apiClient *api.Client,
|
|
||||||
db *database.DB,
|
|
||||||
getenv func(string) string,
|
|
||||||
) error {
|
|
||||||
accountToken := getenv("SPACETRADERS_ACCOUNT_TOKEN")
|
|
||||||
if accountToken == "" {
|
|
||||||
return fmt.Errorf("the SPACETRADERS_ACCOUNT_TOKEN environment variable is not set")
|
|
||||||
}
|
|
||||||
agent := getenv("SPACETRADERS_AGENT")
|
|
||||||
if agent == "" {
|
|
||||||
return fmt.Errorf("the SPACETRADERS_AGENT environment variable is not set")
|
|
||||||
}
|
|
||||||
faction := getenv("SPACETRADERS_FACTION")
|
|
||||||
if faction == "" {
|
|
||||||
return fmt.Errorf("the SPACETRADERS_FACTION environment variable is not set")
|
|
||||||
}
|
|
||||||
// ----- Get token or register ---------------------------------------------
|
|
||||||
apiClient.SetToken(accountToken)
|
|
||||||
register, err := apiClient.Register(faction, agent)
|
|
||||||
if err != nil {
|
|
||||||
apiError := &api.APIError{}
|
|
||||||
if errors.As(err, &apiError) {
|
|
||||||
switch apiError.Code {
|
|
||||||
case 4111: // Agent symbol has already been claimed
|
|
||||||
token, err := db.GetToken()
|
|
||||||
if err != nil || token == "" {
|
|
||||||
return fmt.Errorf("failed to register and failed to get a token from the database: someone stole our agent's callsign: %w", err)
|
|
||||||
}
|
|
||||||
apiClient.SetToken(token)
|
|
||||||
agent, err := apiClient.MyAgent()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to get agent: %w", err)
|
|
||||||
}
|
|
||||||
slog.Info("agent", "/my/agent", agent)
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("failed to register: %w", err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return fmt.Errorf("failed to register with an invalid apiError: %w", err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
token, err := db.GetToken()
|
|
||||||
if err != nil || token == "" {
|
|
||||||
if err := db.AddToken(register.Token); err != nil {
|
|
||||||
return fmt.Errorf("failed to save token: %w", err)
|
|
||||||
}
|
|
||||||
apiClient.SetToken(register.Token)
|
|
||||||
} else {
|
|
||||||
// We successfully registered but have a tainted database
|
|
||||||
slog.Error("token", "token", register.Token)
|
|
||||||
return fmt.Errorf("TODO server reset not implemented yet")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// ----- run agent ---------------------------------------------------------
|
|
||||||
ships, err := apiClient.MyShips()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
slog.Info("start", "ship", ships[0].Nav.Status, "err", err)
|
|
||||||
err = apiClient.Orbit(&ships[0])
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
slog.Info("orbit", "ship", ships[0].Nav.Status, "err", err)
|
|
||||||
err = apiClient.Dock(&ships[0])
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
slog.Info("dock", "ship", ships[0].Nav.Status, "err", err)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,26 +0,0 @@
|
||||||
package agent
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sync"
|
|
||||||
|
|
||||||
"git.adyxax.org/adyxax/spacetraders/golang/pkg/model"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Agent struct {
|
|
||||||
data *model.Agent
|
|
||||||
mutex sync.Mutex
|
|
||||||
}
|
|
||||||
|
|
||||||
var agent Agent
|
|
||||||
|
|
||||||
func GetAgent() *model.Agent {
|
|
||||||
agent.mutex.Lock()
|
|
||||||
defer agent.mutex.Unlock()
|
|
||||||
return agent.data
|
|
||||||
}
|
|
||||||
|
|
||||||
func SetAgent(data *model.Agent) {
|
|
||||||
agent.mutex.Lock()
|
|
||||||
defer agent.mutex.Unlock()
|
|
||||||
agent.data = data
|
|
||||||
}
|
|
78
golang/pkg/agent/run.go
Normal file
78
golang/pkg/agent/run.go
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
package agent
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"log/slog"
|
||||||
|
|
||||||
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/api"
|
||||||
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/database"
|
||||||
|
)
|
||||||
|
|
||||||
|
func Run(
|
||||||
|
apiClient *api.Client,
|
||||||
|
db *database.DB,
|
||||||
|
getenv func(string) string,
|
||||||
|
) error {
|
||||||
|
accountToken := getenv("SPACETRADERS_ACCOUNT_TOKEN")
|
||||||
|
if accountToken == "" {
|
||||||
|
return fmt.Errorf("the SPACETRADERS_ACCOUNT_TOKEN environment variable is not set")
|
||||||
|
}
|
||||||
|
agent := getenv("SPACETRADERS_AGENT")
|
||||||
|
if agent == "" {
|
||||||
|
return fmt.Errorf("the SPACETRADERS_AGENT environment variable is not set")
|
||||||
|
}
|
||||||
|
faction := getenv("SPACETRADERS_FACTION")
|
||||||
|
if faction == "" {
|
||||||
|
return fmt.Errorf("the SPACETRADERS_FACTION environment variable is not set")
|
||||||
|
}
|
||||||
|
// ----- Get token or register ---------------------------------------------
|
||||||
|
apiClient.SetToken(accountToken)
|
||||||
|
register, err := apiClient.Register(faction, agent)
|
||||||
|
if err != nil {
|
||||||
|
apiError := &api.APIError{}
|
||||||
|
if errors.As(err, &apiError) {
|
||||||
|
switch apiError.Code {
|
||||||
|
case 4111: // Agent symbol has already been claimed
|
||||||
|
token, err := db.GetToken()
|
||||||
|
if err != nil || token == "" {
|
||||||
|
return fmt.Errorf("failed to register and failed to get a token from the database: someone stole our agent's callsign: %w", err)
|
||||||
|
}
|
||||||
|
apiClient.SetToken(token)
|
||||||
|
agent, err := apiClient.MyAgent()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to get agent: %w", err)
|
||||||
|
}
|
||||||
|
slog.Info("agent", "/my/agent", agent)
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("failed to register: %w", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("failed to register with an invalid apiError: %w", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
token, err := db.GetToken()
|
||||||
|
if err != nil || token == "" {
|
||||||
|
if err := db.AddToken(register.Token); err != nil {
|
||||||
|
return fmt.Errorf("failed to save token: %w", err)
|
||||||
|
}
|
||||||
|
apiClient.SetToken(register.Token)
|
||||||
|
} else {
|
||||||
|
// We successfully registered but have a tainted database
|
||||||
|
slog.Error("token", "token", register.Token)
|
||||||
|
return fmt.Errorf("TODO server reset not implemented yet")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----- run agent ---------------------------------------------------------
|
||||||
|
contracts, err := apiClient.MyContracts()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
slog.Info("start", "contract", contracts[0], "err", err)
|
||||||
|
ships, err := apiClient.MyShips()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
slog.Info("start", "ship", ships[0].Nav.Status, "err", err)
|
||||||
|
return nil
|
||||||
|
}
|
17
golang/pkg/api/contracts.go
Normal file
17
golang/pkg/api/contracts.go
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (c *Client) MyContracts() ([]model.Contract, error) {
|
||||||
|
uriRef := url.URL{Path: "my/contracts"}
|
||||||
|
var contracts []model.Contract
|
||||||
|
if err := c.Send("GET", &uriRef, nil, &contracts); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to get contracts: %w", err)
|
||||||
|
}
|
||||||
|
return contracts, nil
|
||||||
|
}
|
|
@ -5,12 +5,11 @@ import (
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
"git.adyxax.org/adyxax/spacetraders/golang/pkg/agent"
|
|
||||||
"git.adyxax.org/adyxax/spacetraders/golang/pkg/database"
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/database"
|
||||||
"git.adyxax.org/adyxax/spacetraders/golang/pkg/model"
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (c *Client) Dock(s *model.Ship) error {
|
func (c *Client) dock(s *model.Ship) error {
|
||||||
if s.Nav.Status == "DOCKED" {
|
if s.Nav.Status == "DOCKED" {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -35,7 +34,7 @@ func (c *Client) MyShips() ([]model.Ship, error) {
|
||||||
return ships, nil
|
return ships, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Client) Orbit(s *model.Ship) error {
|
func (c *Client) orbit(s *model.Ship) error {
|
||||||
if s.Nav.Status == "IN_ORBIT" {
|
if s.Nav.Status == "IN_ORBIT" {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -55,7 +54,7 @@ func (c *Client) Refuel(s *model.Ship, db *database.DB) error {
|
||||||
if s.Fuel.Current == s.Fuel.Capacity {
|
if s.Fuel.Current == s.Fuel.Capacity {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if err := c.Dock(s); err != nil {
|
if err := c.dock(s); err != nil {
|
||||||
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
||||||
}
|
}
|
||||||
uriRef := url.URL{Path: path.Join("my/ships", s.Symbol, "refuel")}
|
uriRef := url.URL{Path: path.Join("my/ships", s.Symbol, "refuel")}
|
||||||
|
@ -68,7 +67,9 @@ func (c *Client) Refuel(s *model.Ship, db *database.DB) error {
|
||||||
if err := c.Send("POST", &uriRef, nil, &response); err != nil {
|
if err := c.Send("POST", &uriRef, nil, &response); err != nil {
|
||||||
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
||||||
}
|
}
|
||||||
agent.SetAgent(response.Agent)
|
if err := db.SaveAgent(response.Agent); err != nil {
|
||||||
|
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
||||||
|
}
|
||||||
s.Fuel = response.Fuel
|
s.Fuel = response.Fuel
|
||||||
if err := db.AppendTransaction(response.Transaction); err != nil {
|
if err := db.AppendTransaction(response.Transaction); err != nil {
|
||||||
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
return fmt.Errorf("failed to refuel ship %s: %w", s.Symbol, err)
|
||||||
|
|
19
golang/pkg/database/agents.go
Normal file
19
golang/pkg/database/agents.go
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
package database
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"git.adyxax.org/adyxax/spacetraders/golang/pkg/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (db *DB) SaveAgent(agent *model.Agent) error {
|
||||||
|
data, err := json.Marshal(agent)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to marshal agent: %w", err)
|
||||||
|
}
|
||||||
|
if _, err := db.Exec(`INSERT INTO agents SET data = (json(?));`, data); err != nil {
|
||||||
|
return fmt.Errorf("failed to insert agent data: %w", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -1,3 +1,7 @@
|
||||||
|
CREATE TABLE agents (
|
||||||
|
id INTEGER PRIMARY KEY,
|
||||||
|
data TEXT NOT NULL
|
||||||
|
);
|
||||||
CREATE TABLE markets (
|
CREATE TABLE markets (
|
||||||
id INTEGER PRIMARY KEY,
|
id INTEGER PRIMARY KEY,
|
||||||
systemSymbol TEXT NOT NULL,
|
systemSymbol TEXT NOT NULL,
|
||||||
|
|
32
golang/pkg/model/contract.go
Normal file
32
golang/pkg/model/contract.go
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
package model
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
type Contract struct {
|
||||||
|
Accepted bool `json:"accepted"`
|
||||||
|
DeadlineToAccept time.Time `json:"deadlineToAccept"`
|
||||||
|
Expiration time.Time `json:"expiration"`
|
||||||
|
FactionSymbol string `json:"factionSymbol"`
|
||||||
|
Fullfilled bool `json:"fulfilled"`
|
||||||
|
Id string `json:"id"`
|
||||||
|
Terms *Terms `json:"terms"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Deliver struct {
|
||||||
|
DestinationSymbol string `json:"destinationSymbol"`
|
||||||
|
TradeSymbol string `json:"tradeSymbol"`
|
||||||
|
UnitsFulfilled int `json:"unitsFulfilled"`
|
||||||
|
UnitsRequired int `json:"unitsRequired"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Payment struct {
|
||||||
|
OnAccepted int `json:"onAccepted"`
|
||||||
|
OnFulfilled int `json:"onFulfilled"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Terms struct {
|
||||||
|
Deadline time.Time `json:"deadline"`
|
||||||
|
Payment *Payment `json:"payment"`
|
||||||
|
Deliver []Deliver `json:"deliver"`
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue