1
0
mirror of https://github.com/hybridgroup/gobot.git synced 2025-04-24 13:48:49 +08:00

220 lines
5.4 KiB
Go
Raw Normal View History

2014-04-30 08:10:44 -07:00
package gobot
2013-10-22 16:45:31 -07:00
import (
2013-11-13 20:44:54 -08:00
"fmt"
2013-12-30 22:04:23 -08:00
"log"
2013-10-22 16:45:31 -07:00
)
// JSONRobot a JSON representation of a Robot.
2014-06-10 15:16:11 -07:00
type JSONRobot struct {
2014-05-15 11:50:45 -07:00
Name string `json:"name"`
Commands []string `json:"commands"`
2014-06-10 15:16:11 -07:00
Connections []*JSONConnection `json:"connections"`
Devices []*JSONDevice `json:"devices"`
2014-05-15 11:50:45 -07: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
}
2014-11-13 11:06:57 -08:00
// Robot is a named entitity that manages a collection of connections and devices.
// It containes it's own work routine and a collection of
// custom commands to control a robot remotely via the Gobot api.
2013-10-22 16:45:31 -07:00
type Robot struct {
Name string
Work func()
connections *Connections
devices *Devices
Commander
Eventer
2013-10-22 16:45:31 -07:00
}
// Robots is a collection of Robot
type Robots []*Robot
2014-06-23 20:33:59 -07:00
// Len returns the amount of Robots in the collection.
func (r *Robots) Len() int {
2014-07-09 09:38:43 -07:00
return len(*r)
2014-06-23 20:33:59 -07:00
}
2014-04-29 13:20:32 -07:00
// Start calls the Start method of each Robot in the collection
func (r *Robots) Start() (errs []error) {
2014-07-09 09:38:43 -07:00
for _, robot := range *r {
if errs = robot.Start(); len(errs) > 0 {
for i, err := range errs {
errs[i] = fmt.Errorf("Robot %q: %v", robot.Name, err)
}
2014-11-17 16:23:19 -08:00
return
2014-11-12 11:21:50 -08:00
}
2014-04-26 09:44:26 -07:00
}
2014-11-17 16:23:19 -08:00
return
2014-04-29 13:20:32 -07:00
}
2014-04-26 09:44:26 -07:00
// Stop calls the Stop method of each Robot in the collection
func (r *Robots) Stop() (errs []error) {
for _, robot := range *r {
if errs = robot.Stop(); len(errs) > 0 {
for i, err := range errs {
errs[i] = fmt.Errorf("Robot %q: %v", robot.Name, err)
}
return
}
}
return
}
// Each enumerates through the Robots and calls specified callback function.
func (r *Robots) Each(f func(*Robot)) {
2014-07-09 09:38:43 -07:00
for _, robot := range *r {
2014-04-29 13:20:32 -07:00
f(robot)
}
2013-12-18 23:50:42 -08:00
}
2014-11-13 11:06:57 -08:00
// NewRobot returns a new Robot given a name and optionally accepts:
//
// []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
// A name will be automaically generated if no name is supplied.
2014-06-12 21:32:38 -07:00
func NewRobot(name string, v ...interface{}) *Robot {
if name == "" {
name = fmt.Sprintf("%X", Rand(int(^uint(0)>>1)))
}
2014-04-29 13:20:32 -07:00
r := &Robot{
Name: name,
connections: &Connections{},
devices: &Devices{},
Work: nil,
Eventer: NewEventer(),
Commander: NewCommander(),
2014-04-29 13:20:32 -07:00
}
2014-04-30 08:10:44 -07:00
log.Println("Initializing Robot", r.Name, "...")
for i := range v {
switch v[i].(type) {
case []Connection:
log.Println("Initializing connections...")
for _, connection := range v[i].([]Connection) {
2014-07-09 09:38:43 -07:00
c := r.AddConnection(connection)
log.Println("Initializing connection", c.Name(), "...")
}
case []Device:
log.Println("Initializing devices...")
for _, device := range v[i].([]Device) {
2014-07-09 09:38:43 -07:00
d := r.AddDevice(device)
log.Println("Initializing device", d.Name(), "...")
}
case func():
r.Work = v[i].(func())
2014-06-12 21:32:38 -07:00
}
}
2014-04-29 13:20:32 -07:00
return r
}
// Start a Robot's Connections, Devices, and work.
func (r *Robot) Start() (errs []error) {
2014-04-30 08:10:44 -07:00
log.Println("Starting Robot", r.Name, "...")
if cerrs := r.Connections().Start(); len(cerrs) > 0 {
errs = append(errs, cerrs...)
2014-11-17 16:23:19 -08:00
return
2013-12-30 16:51:21 -08:00
}
if derrs := r.Devices().Start(); len(derrs) > 0 {
errs = append(errs, derrs...)
return
2013-12-30 16:51:21 -08:00
}
2013-12-03 15:51:17 -08:00
if r.Work != nil {
2014-04-30 08:10:44 -07:00
log.Println("Starting work...")
2013-12-03 15:51:17 -08:00
r.Work()
}
return
2013-10-22 16:45:31 -07:00
}
// Stop stops a Robot's connections and Devices
func (r *Robot) Stop() (errs []error) {
log.Println("Stopping Robot", r.Name, "...")
if heers := r.Devices().Halt(); len(heers) > 0 {
for _, err := range heers {
errs = append(errs, err)
}
}
if ceers := r.Connections().Finalize(); len(ceers) > 0 {
for _, err := range ceers {
errs = append(errs, err)
}
}
return errs
}
// Devices returns all devices associated with this Robot.
func (r *Robot) Devices() *Devices {
2014-06-23 20:33:59 -07:00
return r.devices
2013-11-23 10:36:08 -08:00
}
// AddDevice adds a new Device to the robots collection of devices. Returns the
2014-11-13 11:06:57 -08:00
// added device.
2014-07-07 21:45:36 -07:00
func (r *Robot) AddDevice(d Device) Device {
2014-07-09 09:38:43 -07:00
*r.devices = append(*r.Devices(), d)
return d
2014-07-07 21:45:36 -07:00
}
// Device returns a device given a name. Returns nil if the Device does not exist.
2014-07-02 18:08:44 -07:00
func (r *Robot) Device(name string) Device {
2014-04-26 10:13:33 -06:00
if r == nil {
return nil
}
2014-07-09 09:38:43 -07:00
for _, device := range *r.devices {
if device.Name() == name {
2014-01-02 15:12:41 -08:00
return device
}
}
return nil
}
2014-11-13 11:06:57 -08:00
// Connections returns all connections associated with this robot.
func (r *Robot) Connections() *Connections {
2014-06-23 20:33:59 -07:00
return r.connections
2014-01-02 15:12:41 -08:00
}
2014-11-13 11:06:57 -08:00
// AddConnection adds a new connection to the robots collection of connections.
// Returns the added connection.
2014-07-07 21:45:36 -07:00
func (r *Robot) AddConnection(c Connection) Connection {
2014-07-09 09:38:43 -07:00
*r.connections = append(*r.Connections(), c)
return c
2014-07-07 21:45:36 -07:00
}
// Connection returns a connection given a name. Returns nil if the Connection
// does not exist.
2014-07-02 18:08:44 -07:00
func (r *Robot) Connection(name string) Connection {
2014-04-26 10:13:33 -06:00
if r == nil {
return nil
}
2014-07-09 09:38:43 -07:00
for _, connection := range *r.connections {
if connection.Name() == name {
2014-01-02 15:12:41 -08:00
return connection
2013-11-23 09:12:57 -08:00
}
}
return nil
}