hybridgroup.gobot/robot.go

280 lines
6.8 KiB
Go
Raw Normal View History

2014-04-30 23:10:44 +08:00
package gobot
2013-10-23 07:45:31 +08:00
import (
2013-11-14 12:44:54 +08:00
"fmt"
2013-12-31 14:04:23 +08:00
"log"
"os"
"os/signal"
2018-08-30 12:30:38 +08:00
"sync"
"sync/atomic"
2018-08-30 12:30:38 +08:00
multierror "github.com/hashicorp/go-multierror"
2013-10-23 07:45:31 +08:00
)
// JSONRobot a JSON representation of a Robot.
2014-06-11 06:16:11 +08:00
type JSONRobot struct {
2014-05-16 02:50:45 +08:00
Name string `json:"name"`
Commands []string `json:"commands"`
2014-06-11 06:16:11 +08:00
Connections []*JSONConnection `json:"connections"`
Devices []*JSONDevice `json:"devices"`
2014-05-16 02:50:45 +08:00
}
// NewJSONRobot returns a JSONRobot given a Robot.
func NewJSONRobot(robot *Robot) *JSONRobot {
jsonRobot := &JSONRobot{
Name: robot.Name,
Commands: []string{},
Connections: []*JSONConnection{},
Devices: []*JSONDevice{},
}
for command := range robot.Commands() {
jsonRobot.Commands = append(jsonRobot.Commands, command)
}
robot.Devices().Each(func(device Device) {
jsonDevice := NewJSONDevice(device)
jsonRobot.Connections = append(jsonRobot.Connections, NewJSONConnection(robot.Connection(jsonDevice.Connection)))
jsonRobot.Devices = append(jsonRobot.Devices, jsonDevice)
})
return jsonRobot
}
2016-07-14 00:44:47 +08:00
// Robot is a named entity that manages a collection of connections and devices.
// It contains its own work routine and a collection of
2014-11-14 03:06:57 +08:00
// custom commands to control a robot remotely via the Gobot api.
2013-10-23 07:45:31 +08:00
type Robot struct {
2018-08-30 12:30:38 +08:00
Name string
Work func()
connections *Connections
devices *Devices
trap func(chan os.Signal)
AutoRun bool
running atomic.Value
done chan bool
workRegistry *RobotWorkRegistry
WorkEveryWaitGroup *sync.WaitGroup
WorkAfterWaitGroup *sync.WaitGroup
Commander
Eventer
2013-10-23 07:45:31 +08:00
}
// Robots is a collection of Robot
type Robots []*Robot
2014-06-24 11:33:59 +08:00
// Len returns the amount of Robots in the collection.
func (r *Robots) Len() int {
2014-07-10 00:38:43 +08:00
return len(*r)
2014-06-24 11:33:59 +08:00
}
2014-04-30 04:20:32 +08:00
// Start calls the Start method of each Robot in the collection. We return on first error.
func (r *Robots) Start(args ...interface{}) error {
autoRun := true
if args[0] != nil {
var ok bool
if autoRun, ok = args[0].(bool); !ok {
// we treat this as false
autoRun = false
}
}
2014-07-10 00:38:43 +08:00
for _, robot := range *r {
if err := robot.Start(autoRun); err != nil {
return err
2014-11-13 03:21:50 +08:00
}
2014-04-27 00:44:26 +08:00
}
return nil
2014-04-30 04:20:32 +08:00
}
2014-04-27 00:44:26 +08:00
// Stop calls the Stop method of each Robot in the collection. We try to stop all robots and
// collect the errors.
func (r *Robots) Stop() error {
var err error
for _, robot := range *r {
if e := robot.Stop(); e != nil {
err = multierror.Append(err, e)
}
}
return err
}
// Each enumerates through the Robots and calls specified callback function.
func (r *Robots) Each(f func(*Robot)) {
2014-07-10 00:38:43 +08:00
for _, robot := range *r {
2014-04-30 04:20:32 +08:00
f(robot)
}
2013-12-19 15:50:42 +08:00
}
// NewRobot returns a new Robot. It supports the following optional params:
2014-11-14 03:06:57 +08:00
//
// name: string with the name of the Robot. A name will be automatically generated if no name is supplied.
// []Connection: Connections which are automatically started and stopped with the robot
// []Device: Devices which are automatically started and stopped with the robot
// func(): The work routine the robot will execute once all devices and connections have been initialized and started
func NewRobot(v ...interface{}) *Robot {
2014-04-30 04:20:32 +08:00
r := &Robot{
Name: fmt.Sprintf("%X", Rand(int(^uint(0)>>1))),
connections: &Connections{},
devices: &Devices{},
done: make(chan bool, 1),
trap: func(c chan os.Signal) {
signal.Notify(c, os.Interrupt)
},
AutoRun: true,
Work: nil,
Eventer: NewEventer(),
Commander: NewCommander(),
2014-04-30 04:20:32 +08:00
}
for i := range v {
switch val := v[i].(type) {
case string:
r.Name = val
case []Connection:
log.Println("Initializing connections...")
for _, connection := range val {
2014-07-10 00:38:43 +08:00
c := r.AddConnection(connection)
log.Println("Initializing connection", c.Name(), "...")
}
case []Device:
log.Println("Initializing devices...")
for _, device := range val {
2014-07-10 00:38:43 +08:00
d := r.AddDevice(device)
log.Println("Initializing device", d.Name(), "...")
}
case func():
r.Work = val
2014-06-13 12:32:38 +08:00
}
}
2018-08-30 03:15:04 +08:00
r.workRegistry = &RobotWorkRegistry{
r: make(map[string]*RobotWork),
}
2018-08-30 12:30:38 +08:00
r.WorkAfterWaitGroup = &sync.WaitGroup{}
r.WorkEveryWaitGroup = &sync.WaitGroup{}
2018-08-30 03:15:04 +08:00
r.running.Store(false)
log.Println("Robot", r.Name, "initialized.")
2014-04-30 04:20:32 +08:00
return r
}
// Start a Robot's Connections, Devices, and work. We stop initialization of
// connections and devices on first error.
func (r *Robot) Start(args ...interface{}) error {
if len(args) > 0 && args[0] != nil {
var ok bool
if r.AutoRun, ok = args[0].(bool); !ok {
// we treat this as false
r.AutoRun = false
}
}
2014-04-30 23:10:44 +08:00
log.Println("Starting Robot", r.Name, "...")
if err := r.Connections().Start(); err != nil {
log.Println(err)
return err
2013-12-31 08:51:21 +08:00
}
if err := r.Devices().Start(); err != nil {
log.Println(err)
return err
2013-12-31 08:51:21 +08:00
}
if r.Work == nil {
r.Work = func() {}
2013-12-04 07:51:17 +08:00
}
log.Println("Starting work...")
go func() {
r.Work()
<-r.done
}()
r.running.Store(true)
if !r.AutoRun {
return nil
}
c := make(chan os.Signal, 1)
r.trap(c)
// waiting for interrupt coming on the channel
<-c
// Stop calls the Stop method on itself, if we are "auto-running".
return r.Stop()
2013-10-23 07:45:31 +08:00
}
// Stop stops a Robot's connections and devices. We try to stop all items and
// collect all errors.
func (r *Robot) Stop() error {
var err error
log.Println("Stopping Robot", r.Name, "...")
if e := r.Devices().Halt(); e != nil {
err = multierror.Append(err, e)
}
if e := r.Connections().Finalize(); e != nil {
err = multierror.Append(err, e)
}
r.done <- true
r.running.Store(false)
return err
}
// Running returns if the Robot is currently started or not
func (r *Robot) Running() bool {
return r.running.Load().(bool) //nolint:forcetypeassert // no error return value, so there is no better way
}
// Devices returns all devices associated with this Robot.
func (r *Robot) Devices() *Devices {
2014-06-24 11:33:59 +08:00
return r.devices
2013-11-24 02:36:08 +08:00
}
// AddDevice adds a new Device to the robots collection of devices. Returns the
2014-11-14 03:06:57 +08:00
// added device.
2014-07-08 12:45:36 +08:00
func (r *Robot) AddDevice(d Device) Device {
2014-07-10 00:38:43 +08:00
*r.devices = append(*r.Devices(), d)
return d
2014-07-08 12:45:36 +08:00
}
// Device returns a device given a name. Returns nil if the Device does not exist.
2014-07-03 09:08:44 +08:00
func (r *Robot) Device(name string) Device {
2014-04-27 00:13:33 +08:00
if r == nil {
return nil
}
2014-07-10 00:38:43 +08:00
for _, device := range *r.devices {
if device.Name() == name {
2014-01-03 07:12:41 +08:00
return device
}
}
return nil
}
2014-11-14 03:06:57 +08:00
// Connections returns all connections associated with this robot.
func (r *Robot) Connections() *Connections {
2014-06-24 11:33:59 +08:00
return r.connections
2014-01-03 07:12:41 +08:00
}
2014-11-14 03:06:57 +08:00
// AddConnection adds a new connection to the robots collection of connections.
// Returns the added connection.
2014-07-08 12:45:36 +08:00
func (r *Robot) AddConnection(c Connection) Connection {
2014-07-10 00:38:43 +08:00
*r.connections = append(*r.Connections(), c)
return c
2014-07-08 12:45:36 +08:00
}
// Connection returns a connection given a name. Returns nil if the Connection
// does not exist.
2014-07-03 09:08:44 +08:00
func (r *Robot) Connection(name string) Connection {
2014-04-27 00:13:33 +08:00
if r == nil {
return nil
}
2014-07-10 00:38:43 +08:00
for _, connection := range *r.connections {
if connection.Name() == name {
2014-01-03 07:12:41 +08:00
return connection
2013-11-24 01:12:57 +08:00
}
}
return nil
}