2023-04-08 13:58:47 -06:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-05-30 21:50:59 -06:00
|
|
|
"log"
|
2023-05-31 00:37:51 -06:00
|
|
|
"time"
|
2023-06-01 13:11:32 -06:00
|
|
|
"errors"
|
2023-04-08 13:58:47 -06:00
|
|
|
)
|
|
|
|
|
2023-05-30 20:45:16 -06:00
|
|
|
type Member struct {
|
|
|
|
BaseResource
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewMember(name string) * Member {
|
|
|
|
member := &Member{
|
2023-06-01 22:42:47 -06:00
|
|
|
BaseResource: NewBaseResource(name, "A Team Member", []Resource{}),
|
2023-05-30 20:45:16 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return member
|
|
|
|
}
|
|
|
|
|
2023-04-08 13:58:47 -06:00
|
|
|
type Team struct {
|
|
|
|
BaseResource
|
|
|
|
Org string
|
|
|
|
Team string
|
|
|
|
}
|
|
|
|
|
2023-05-30 20:45:16 -06:00
|
|
|
func (team * Team) Members() []*Member {
|
|
|
|
ret := make([]*Member, len(team.children))
|
|
|
|
for idx, member := range(team.children) {
|
|
|
|
ret[idx] = member.(*Member)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTeam(org string, team string, members []*Member) * Team {
|
2023-04-08 13:58:47 -06:00
|
|
|
name := fmt.Sprintf("%s%s", org, team)
|
|
|
|
description := fmt.Sprintf("Team %s", name)
|
|
|
|
resource := &Team{
|
2023-06-01 22:42:47 -06:00
|
|
|
BaseResource: NewBaseResource(name, description, make([]Resource, len(members))),
|
2023-04-08 13:58:47 -06:00
|
|
|
Org: org,
|
|
|
|
Team: team,
|
|
|
|
}
|
2023-06-01 22:42:47 -06:00
|
|
|
|
2023-05-30 20:45:16 -06:00
|
|
|
for idx, member := range(members) {
|
|
|
|
resource.children[idx] = member
|
|
|
|
}
|
2023-06-01 22:42:47 -06:00
|
|
|
|
2023-04-08 13:58:47 -06:00
|
|
|
return resource
|
|
|
|
}
|
|
|
|
|
|
|
|
type Alliance struct {
|
|
|
|
BaseResource
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAlliance(team0 * Team, team1 * Team) * Alliance {
|
|
|
|
name := fmt.Sprintf("Alliance %s/%s", team0.Name(), team1.Name())
|
|
|
|
description := ""
|
|
|
|
|
|
|
|
resource := &Alliance{
|
2023-06-01 22:42:47 -06:00
|
|
|
BaseResource: NewBaseResource(name, description, []Resource{team0, team1}),
|
2023-04-08 13:58:47 -06:00
|
|
|
}
|
|
|
|
return resource
|
|
|
|
}
|
2023-05-30 20:45:16 -06:00
|
|
|
|
|
|
|
type Match struct {
|
|
|
|
BaseEvent
|
2023-05-30 21:50:59 -06:00
|
|
|
state string
|
2023-05-31 00:37:51 -06:00
|
|
|
control string
|
|
|
|
control_start time.Time
|
2023-05-30 20:45:16 -06:00
|
|
|
}
|
|
|
|
|
2023-06-01 13:11:32 -06:00
|
|
|
type Arena struct {
|
|
|
|
BaseResource
|
|
|
|
connected bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewVirtualArena(name string) * Arena {
|
|
|
|
arena := &Arena{
|
2023-06-01 22:42:47 -06:00
|
|
|
BaseResource: NewBaseResource(name, "A virtual vex arena", []Resource{}),
|
2023-06-01 13:11:32 -06:00
|
|
|
connected: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
return arena
|
|
|
|
}
|
|
|
|
|
2023-06-02 17:31:29 -06:00
|
|
|
func (arena * Arena) lock(event Event) error {
|
2023-06-01 13:11:32 -06:00
|
|
|
if arena.connected == false {
|
|
|
|
log.Printf("ARENA NOT CONNECTED: %s", arena.Name())
|
|
|
|
error_str := fmt.Sprintf("%s is not connected, cannot lock", arena.Name())
|
|
|
|
return errors.New(error_str)
|
|
|
|
}
|
2023-06-02 17:31:29 -06:00
|
|
|
return nil
|
2023-06-01 13:11:32 -06:00
|
|
|
}
|
|
|
|
|
2023-06-02 17:31:29 -06:00
|
|
|
func (arena * Arena) update(signal GraphSignal) {
|
|
|
|
log.Printf("ARENA_UPDATE: %s", arena.Name())
|
|
|
|
arena.signal <- signal
|
2023-06-01 22:42:47 -06:00
|
|
|
}
|
|
|
|
|
2023-06-01 13:11:32 -06:00
|
|
|
func (arena * Arena) Connect(abort chan error) bool {
|
|
|
|
log.Printf("Connecting %s", arena.Name())
|
|
|
|
go func(arena * Arena, abort chan error) {
|
|
|
|
owner := arena.Owner()
|
|
|
|
update_str := fmt.Sprintf("VIRTUAL_ARENA connected: %s", arena.Name())
|
2023-06-01 22:42:47 -06:00
|
|
|
signal := NewSignal(arena, "arena_connected")
|
|
|
|
signal.description = update_str
|
2023-06-02 17:31:29 -06:00
|
|
|
arena.connected = true
|
2023-06-03 01:38:35 -06:00
|
|
|
go SendUpdate(arena, signal)
|
2023-06-01 13:11:32 -06:00
|
|
|
log.Printf("VIRTUAL_ARENA goroutine starting: %s", arena.Name())
|
|
|
|
for true {
|
|
|
|
select {
|
|
|
|
case <- abort:
|
|
|
|
log.Printf("Virtual arena %s aborting", arena.Name())
|
|
|
|
break
|
2023-06-01 22:42:47 -06:00
|
|
|
case update := <- arena.signal:
|
2023-06-03 01:38:35 -06:00
|
|
|
log.Printf("%s update: %+v", arena.Name(), update)
|
2023-06-01 13:11:32 -06:00
|
|
|
new_owner := arena.Owner()
|
|
|
|
if new_owner != owner {
|
|
|
|
log.Printf("NEW_OWNER for %s", arena.Name())
|
|
|
|
if new_owner != nil {
|
|
|
|
log.Printf("new: %s", new_owner.Name())
|
|
|
|
} else {
|
|
|
|
log.Printf("new: nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
if owner != nil {
|
|
|
|
log.Printf("old: %s", owner.Name())
|
|
|
|
} else {
|
|
|
|
log.Printf("old: nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
owner = new_owner
|
|
|
|
if owner != nil {
|
|
|
|
} else {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}(arena, abort)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2023-05-31 00:37:51 -06:00
|
|
|
const start_slack = 3000 * time.Millisecond
|
2023-06-03 01:38:35 -06:00
|
|
|
const TEMP_AUTON_TIME = time.Second * 3
|
|
|
|
const TEMP_DRIVE_TIME = time.Second * 5
|
2023-05-31 00:37:51 -06:00
|
|
|
|
2023-05-30 20:45:16 -06:00
|
|
|
func NewMatch(alliance0 * Alliance, alliance1 * Alliance, arena * Arena) * Match {
|
2023-06-01 13:11:32 -06:00
|
|
|
name := fmt.Sprintf("Match: %s vs. %s on %s", alliance0.Name(), alliance1.Name(), arena.Name())
|
2023-05-30 20:45:16 -06:00
|
|
|
description := "A vex match"
|
|
|
|
|
|
|
|
match := &Match{
|
|
|
|
BaseEvent: NewBaseEvent(name, description, []Resource{alliance0, alliance1, arena}),
|
2023-05-30 21:50:59 -06:00
|
|
|
state: "init",
|
2023-05-31 00:37:51 -06:00
|
|
|
control: "init",
|
|
|
|
control_start: time.UnixMilli(0),
|
2023-05-30 20:45:16 -06:00
|
|
|
}
|
|
|
|
|
2023-05-30 21:50:59 -06:00
|
|
|
match.actions["start"] = func() (string, error) {
|
2023-06-01 13:48:38 -06:00
|
|
|
log.Printf("STARTING_MATCH %s", match.Name())
|
2023-05-31 00:37:51 -06:00
|
|
|
match.control = "none"
|
2023-05-30 21:50:59 -06:00
|
|
|
match.state = "scheduled"
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
|
2023-06-03 01:38:35 -06:00
|
|
|
match.handlers["queue_autonomous"] = func(signal GraphSignal) (string, error) {
|
|
|
|
if match.state != "scheduled" {
|
|
|
|
log.Printf("BAD_STATE: %s: %s", signal.Type(), match.state)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
2023-05-31 00:37:51 -06:00
|
|
|
match.control = "none"
|
|
|
|
match.state = "autonomous_queued"
|
|
|
|
match.control_start = time.Now().Add(start_slack)
|
2023-06-03 01:38:35 -06:00
|
|
|
go SendUpdate(match, NewSignal(match, "autonomous_queued"))
|
2023-05-31 00:37:51 -06:00
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
|
2023-06-03 01:38:35 -06:00
|
|
|
match.handlers["start_autonomous"] = func(signal GraphSignal) (string, error) {
|
|
|
|
if match.state != "autonomous_queued" {
|
|
|
|
log.Printf("BAD_STATE: %s: %s", signal.Type(), match.state)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
match.control = "program"
|
2023-05-31 00:37:51 -06:00
|
|
|
match.state = "autonomous_running"
|
2023-06-03 01:38:35 -06:00
|
|
|
// TODO replace with typed protobuf
|
|
|
|
match.control_start = signal.Time()
|
|
|
|
go SendUpdate(match, NewSignal(match, "autonomous_running"))
|
|
|
|
go func(match * Match) {
|
|
|
|
control_wait := time.Until(match.control_start.Add(TEMP_AUTON_TIME))
|
|
|
|
time.Sleep(control_wait)
|
|
|
|
SendUpdate(match, NewSignal(match, "autonomous_done"))
|
|
|
|
}(match)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
match.handlers["autonomous_done"] = func(signal GraphSignal) (string, error) {
|
|
|
|
if match.state != "autonomous_running" {
|
|
|
|
log.Printf("BAD_STATE: %s: %s", signal.Type(), match.state)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
match.control = "none"
|
|
|
|
match.state = "autonomous_done"
|
|
|
|
|
2023-05-30 21:50:59 -06:00
|
|
|
return "wait", nil
|
2023-05-30 20:45:16 -06:00
|
|
|
}
|
|
|
|
|
2023-06-03 01:38:35 -06:00
|
|
|
match.handlers["queue_driver"] = func(signal GraphSignal) (string, error) {
|
|
|
|
if match.state != "autonomous_done"{
|
|
|
|
log.Printf("BAD_STATE: %s: %s", signal.Type(), match.state)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
match.control = "none"
|
|
|
|
match.state = "driver_queued"
|
|
|
|
match.control_start = time.Now().Add(start_slack)
|
|
|
|
go SendUpdate(match, NewSignal(match, "driver_queued"))
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
match.handlers["start_driver"] = func(signal GraphSignal) (string, error) {
|
|
|
|
if match.state != "driver_queued" {
|
|
|
|
log.Printf("BAD_STATE: %s: %s", signal.Type(), match.state)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
match.control = "driver"
|
|
|
|
match.state = "driver_running"
|
|
|
|
match.control_start = signal.Time()
|
|
|
|
|
|
|
|
go SendUpdate(match, NewSignal(match, "driver_running"))
|
|
|
|
go func(match * Match) {
|
|
|
|
control_wait := time.Until(match.control_start.Add(TEMP_DRIVE_TIME))
|
|
|
|
time.Sleep(control_wait)
|
|
|
|
SendUpdate(match, NewSignal(match, "driver_done"))
|
|
|
|
}(match)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
match.handlers["driver_done"] = func(signal GraphSignal) (string, error) {
|
|
|
|
if match.state != "driver_running" {
|
|
|
|
log.Printf("BAD_STATE: %s: %s", signal.Type(), match.state)
|
|
|
|
return "wait", nil
|
|
|
|
}
|
|
|
|
match.control = "none"
|
|
|
|
match.state = "driver_done"
|
|
|
|
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
2023-05-30 20:45:16 -06:00
|
|
|
return match
|
|
|
|
}
|