cylon/lib/robot.js

435 lines
9.9 KiB
JavaScript
Raw Normal View History

2015-04-15 09:16:03 +08:00
/**
* Cylon.js - Robot class
2013-10-25 05:25:42 +08:00
* cylonjs.com
*
2015-01-08 04:58:50 +08:00
* Copyright (c) 2013-2015 The Hybrid Group
2013-10-25 05:25:42 +08:00
* Licensed under the Apache 2.0 license.
*/
2014-12-16 03:15:29 +08:00
"use strict";
2013-10-25 05:25:42 +08:00
var initializer = require("./initializer"),
2014-12-16 03:15:29 +08:00
Logger = require("./logger"),
Utils = require("./utils"),
Config = require("./config"),
_ = require("./utils/helpers");
2014-05-08 23:57:20 +08:00
var validator = require("./validator");
2014-03-04 06:33:29 +08:00
var Async = require("async"),
2014-12-16 03:15:29 +08:00
EventEmitter = require("events").EventEmitter;
// used when creating default robot names
var ROBOT_ID = 1;
2015-06-09 13:57:49 +08:00
/**
* Creates a new Robot instance based on provided options
*
* @constructor
* @param {Object} opts object with Robot options
* @param {String} [name] the name the robot should have
* @param {Object} [connections] object containing connection info for the Robot
* @param {Object} [devices] object containing device information for the Robot
* @param {Function} [work] a function the Robot will run when started
* @returns {Robot} new Robot instance
*/
2014-06-10 04:39:16 +08:00
var Robot = module.exports = function Robot(opts) {
Utils.classCallCheck(this, Robot);
2014-07-13 01:48:54 +08:00
opts = opts || {};
2014-05-08 23:57:20 +08:00
validator.validate(opts);
2014-05-08 23:57:20 +08:00
var methods = [
"toString",
"halt",
"startDevices",
"startConnections",
"start",
"initRobot",
2014-05-08 23:57:20 +08:00
"initDevices",
"initConnections",
"log"
2014-05-08 23:57:20 +08:00
];
2015-02-20 09:23:41 +08:00
methods.forEach(function(method) {
this[method] = this[method].bind(this);
}, this);
2014-05-08 23:57:20 +08:00
this.initRobot(opts);
this.initConnections(opts);
this.initDevices(opts);
2014-05-08 23:57:20 +08:00
_.each(opts, function(opt, name) {
2014-12-18 06:42:34 +08:00
if (this[name] !== undefined) {
return;
}
2014-08-06 09:41:57 +08:00
if (_.isFunction(opt)) {
this[name] = opt.bind(this);
if (opts.commands == null) {
this.commands[name] = opt.bind(this);
}
} else {
this[name] = opt;
2014-08-06 09:41:57 +08:00
}
}, this);
2014-08-06 09:41:57 +08:00
if (opts.commands) {
2015-02-20 09:23:41 +08:00
var cmds;
if (_.isFunction(opts.commands)) {
2015-02-20 09:23:41 +08:00
cmds = opts.commands.call(this);
} else {
cmds = opts.commands;
}
if (_.isObject(cmds)) {
this.commands = cmds;
} else {
var err = "#commands must be an object ";
err += "or a function that returns an object";
throw new Error(err);
}
2014-05-08 23:57:20 +08:00
}
2014-08-06 09:41:57 +08:00
2014-12-16 03:15:29 +08:00
var mode = Utils.fetch(Config, "mode", "manual");
2014-09-09 04:41:59 +08:00
2014-12-16 03:15:29 +08:00
if (mode === "auto") {
// run on the next tick, to allow for "work" event handlers to be set up
2014-09-09 05:33:07 +08:00
setTimeout(this.start, 0);
2014-09-09 04:41:59 +08:00
}
2014-05-08 23:57:20 +08:00
};
Utils.subclass(Robot, EventEmitter);
2014-05-08 23:57:20 +08:00
2015-06-09 13:57:49 +08:00
/**
* Condenses information on a Robot to a JSON-serializable format
*
* @return {Object} serializable information on the Robot
*/
Robot.prototype.toJSON = function() {
2014-05-08 23:57:20 +08:00
return {
name: this.name,
connections: _.invoke(this.connections, "toJSON"),
devices: _.invoke(this.devices, "toJSON"),
2015-02-20 09:23:41 +08:00
commands: Object.keys(this.commands),
events: _.isArray(this.events) ? this.events : []
2014-05-08 23:57:20 +08:00
};
};
2015-06-09 13:57:49 +08:00
/**
* Adds a new Connection to the Robot with the provided name and details.
*
* @param {String} name string name for the Connection to use
* @param {Object} conn options for the Connection initializer
* @return {Object} the robot
*/
Robot.prototype.connection = function(name, conn) {
conn.robot = this;
conn.name = name;
if (this.connections[conn.name]) {
2014-12-16 03:15:29 +08:00
var original = conn.name,
str;
conn.name = Utils.makeUnique(original, Object.keys(this.connections));
2014-12-16 03:15:29 +08:00
str = "Connection names must be unique.";
str += "Renaming '" + original + "' to '" + conn.name + "'";
this.log("warn", str);
}
this.connections[conn.name] = initializer("adaptor", conn);
return this;
};
2015-06-09 13:57:49 +08:00
/**
* Initializes all values for a new Robot.
*
* @param {Object} opts object passed to Robot constructor
* @return {void}
*/
Robot.prototype.initRobot = function(opts) {
this.name = opts.name || "Robot " + ROBOT_ID++;
this.connections = {};
this.devices = {};
this.adaptors = {};
this.drivers = {};
this.commands = {};
this.running = false;
this.work = opts.work || opts.play;
if (!this.work) {
this.work = function() { this.log("debug", "No work yet."); };
}
};
2015-06-09 13:57:49 +08:00
/**
* Initializes a Robot's connections
*
* @param {Object} opts object passed to constructor
* @return {Array} the Robot's connections
*/
Robot.prototype.initConnections = function(opts) {
this.log("info", "Initializing connections.");
2014-05-08 23:57:20 +08:00
if (opts.connections == null) {
return this.connections;
}
_.each(opts.connections, function(conn, key) {
var name = _.isString(key) ? key : conn.name;
if (conn.devices) {
opts.devices = opts.devices || {};
_.each(conn.devices, function(device, d) {
device.connection = name;
opts.devices[d] = device;
});
delete conn.devices;
}
this.connection(name, conn);
}, this);
2014-05-08 23:57:20 +08:00
return this.connections;
};
2015-06-09 13:57:49 +08:00
/**
* Adds a new Device to the Robot with the provided name and details.
*
* @param {String} name string name for the Device to use
* @param {Object} device options for the Device initializer
* @return {Object} the robot
*/
Robot.prototype.device = function(name, device) {
2014-12-16 03:15:29 +08:00
var str;
device.robot = this;
device.name = name;
if (this.devices[device.name]) {
var original = device.name;
device.name = Utils.makeUnique(original, Object.keys(this.devices));
2014-12-16 03:15:29 +08:00
str = "Device names must be unique.";
str += "Renaming '" + original + "' to '" + device.name + "'";
this.log("warn", str);
}
if (_.isString(device.connection)) {
if (this.connections[device.connection] == null) {
2014-12-16 03:15:29 +08:00
str = "No connection found with the name " + device.connection + ".\n";
this.log("fatal", str);
2014-12-16 03:15:29 +08:00
process.emit("SIGINT");
}
2014-12-18 06:42:34 +08:00
device.connection = this.connections[device.connection];
} else {
2015-02-20 09:23:41 +08:00
for (var c in this.connections) {
device.connection = this.connections[c];
break;
}
}
this.devices[device.name] = initializer("driver", device);
return this;
2014-12-16 03:15:29 +08:00
};
2015-06-09 13:57:49 +08:00
/**
* Initializes a Robot's devices
*
* @param {Object} opts object passed to constructor
* @return {Array} the Robot's devices
*/
Robot.prototype.initDevices = function(opts) {
this.log("info", "Initializing devices.");
2014-05-08 23:57:20 +08:00
if (opts.devices == null) {
return this.devices;
}
// check that there are connections to use
if (!Object.keys(this.connections).length) {
2014-12-16 03:15:29 +08:00
throw new Error("No connections specified");
}
_.each(opts.devices, function(device, key) {
var name = _.isString(key) ? key : device.name;
this.device(name, device);
}, this);
2014-05-08 23:57:20 +08:00
return this.devices;
};
2015-06-09 13:57:49 +08:00
/**
* Starts the Robot's connections, then devices, then work.
*
* @param {Function} callback function to be triggered when the Robot has
* started working
* @return {Object} the Robot
*/
2014-10-04 15:56:08 +08:00
Robot.prototype.start = function(callback) {
if (this.running) {
2014-10-01 03:22:00 +08:00
return this;
}
2014-12-16 03:15:29 +08:00
var mode = Utils.fetch(Config, "workMode", "async");
2014-06-10 04:55:44 +08:00
2014-12-18 06:42:34 +08:00
var start = function() {
if (mode === "async") {
this.startWork();
}
}.bind(this);
2014-06-10 04:55:44 +08:00
Async.series([
this.startConnections,
this.startDevices,
2014-12-18 06:42:34 +08:00
start
2014-10-04 15:56:08 +08:00
], function(err, results) {
2015-04-15 12:49:12 +08:00
if (err) {
this.log("fatal", "An error occured while trying to start the robot:");
this.log("fatal", err);
2014-12-18 06:42:34 +08:00
this.halt(function() {
if (_.isFunction(this.error)) {
this.error.call(this, err);
}
if (this.listeners("error").length) {
this.emit("error", err);
}
}.bind(this));
2014-06-10 04:55:44 +08:00
}
2014-12-18 06:42:34 +08:00
if (_.isFunction(callback)) {
2014-10-04 15:56:08 +08:00
callback(err, results);
}
2014-10-01 07:13:20 +08:00
}.bind(this));
2014-09-09 04:41:59 +08:00
return this;
2014-05-08 23:57:20 +08:00
};
2015-06-09 13:57:49 +08:00
/**
* Starts the Robot's work function
*
* @return {void}
*/
2014-10-04 14:07:07 +08:00
Robot.prototype.startWork = function() {
this.log("info", "Working.");
2014-10-04 14:07:07 +08:00
2014-12-16 03:15:29 +08:00
this.emit("ready", this);
2014-10-04 14:07:07 +08:00
this.work.call(this, this);
this.running = true;
};
2015-06-09 13:57:49 +08:00
/**
* Starts the Robot's connections
*
* @param {Function} callback function to be triggered after the connections are
* started
* @return {void}
*/
2014-05-08 23:57:20 +08:00
Robot.prototype.startConnections = function(callback) {
this.log("info", "Starting connections.");
2014-05-08 23:57:20 +08:00
var starters = _.map(this.connections, function(conn, name) {
2014-12-18 06:42:34 +08:00
this[name] = conn;
return function(cb) {
2014-12-18 06:42:34 +08:00
var str = "Starting connection '" + name + "'";
if (conn.host) {
str += " on host " + conn.host;
} else if (conn.port) {
str += " on port " + conn.port;
}
this.log("debug", str + ".");
return conn.connect.call(conn, cb);
}.bind(this);
}, this);
2014-05-08 23:57:20 +08:00
return Async.parallel(starters, callback);
};
2015-06-09 13:57:49 +08:00
/**
* Starts the Robot's devices
*
* @param {Function} callback function to be triggered after the devices are
* started
* @return {void}
*/
2014-05-08 23:57:20 +08:00
Robot.prototype.startDevices = function(callback) {
var log = this.log;
log("info", "Starting devices.");
2014-05-08 23:57:20 +08:00
var starters = _.map(this.devices, function(device, name) {
2014-12-18 06:42:34 +08:00
this[name] = device;
return function(cb) {
2014-12-18 06:42:34 +08:00
var str = "Starting device '" + name + "'";
if (device.pin) {
str += " on pin " + device.pin;
}
log("debug", str + ".");
return device.start.call(device, cb);
};
}, this);
2014-05-08 23:57:20 +08:00
return Async.parallel(starters, callback);
};
2015-06-09 13:57:49 +08:00
/**
* Halts the Robot, attempting to gracefully stop devices and connections.
*
* @param {Function} callback to be triggered when the Robot has stopped
* @return {void}
*/
Robot.prototype.halt = function(callback) {
callback = callback || function() {};
if (!this.isRunning) {
return callback();
}
var devices = _.pluck(this.devices, "halt"),
connections = _.pluck(this.connections, "disconnect");
try {
Async.parallel(devices, function() {
Async.parallel(connections, callback);
});
} catch (e) {
var msg = "An error occured while attempting to safely halt the robot";
this.log("error", msg);
this.log("error", e.message);
}
this.running = false;
2014-05-08 23:57:20 +08:00
};
2015-06-09 13:57:49 +08:00
/**
* Generates a String representation of a Robot
*
* @return {String} representation of a Robot
*/
2014-05-08 23:57:20 +08:00
Robot.prototype.toString = function() {
return "[Robot name='" + this.name + "']";
};
2015-01-18 13:29:07 +08:00
Robot.prototype.log = function(level) {
var args = Array.prototype.slice.call(arguments, 1);
args.unshift("[" + this.name + "] -");
Logger[level].apply(null, args);
};