cylon/lib/robot.js

510 lines
12 KiB
JavaScript
Raw Normal View History

2013-10-25 05:25:42 +08:00
/*
* robot
* cylonjs.com
*
2014-02-28 03:17:02 +08:00
* Copyright (c) 2013-2014 The Hybrid Group
2013-10-25 05:25:42 +08:00
* Licensed under the Apache 2.0 license.
*/
2014-02-28 03:17:02 +08:00
'use strict';
2013-10-25 05:25:42 +08:00
2014-05-08 23:57:20 +08:00
var Connection = require("./connection"),
2014-05-14 09:14:36 +08:00
Device = require("./device"),
2014-05-14 10:44:40 +08:00
Logger = require('./logger'),
Utils = require('./utils'),
Config = require('./config');
2014-05-08 23:57:20 +08:00
2014-03-04 06:33:29 +08:00
var Async = require("async"),
2014-05-08 23:57:20 +08:00
EventEmitter = require('events').EventEmitter;
// Require these modules here, so Browserify can grab them later if necessary.
require('./test/loopback');
require('./test/test-adaptor');
require('./test/test-driver');
require('./test/ping');
var missingModuleError = function(module) {
var string = "Cannot find the '" + module + "' module. ";
string += "Please install it with 'npm install " + module + "' and try again.";
console.log(string);
process.emit('SIGINT');
};
2014-05-08 23:57:20 +08:00
// Public: Creates a new Robot
//
// opts - object containing Robot options
// name - optional, string name of the robot
// connection/connections - object connections to connect to
// device/devices - object devices to connect to
// work - work to be performed when the Robot is started
//
// Returns a new Robot
// Example (CoffeeScript):
// Cylon.robot
// name: "Spherobot!"
//
// connection:
// name: 'sphero', adaptor: 'sphero', port: '/dev/rfcomm0'
//
// device:
// name: 'sphero', driver: 'sphero'
//
// work: (me) ->
// Utils.every 1.second(), ->
2014-05-08 23:57:20 +08:00
// me.sphero.roll 60, Math.floor(Math.random() * 360//
2014-06-10 04:39:16 +08:00
var Robot = module.exports = function Robot(opts) {
2014-07-13 01:48:54 +08:00
opts = opts || {};
2014-05-08 23:57:20 +08:00
var methods = [
"toString",
"registerDriver",
"requireDriver",
"registerAdaptor",
"requireAdaptor",
"halt",
"startDevices",
"startConnections",
"start",
"initDevices",
"initConnections"
];
2014-06-10 04:39:16 +08:00
methods.forEach(function(method) {
this[method] = this[method].bind(this);
}.bind(this));
2014-05-08 23:57:20 +08:00
2014-07-13 01:48:54 +08:00
this.name = opts.name || Robot.randomName();
2014-05-08 23:57:20 +08:00
this.connections = {};
this.devices = {};
this.adaptors = {};
this.drivers = {};
2014-08-06 09:41:57 +08:00
this.commands = {};
2014-05-08 23:57:20 +08:00
this.running = false;
2014-06-10 04:39:16 +08:00
this.work = opts.work || opts.play;
2014-05-08 23:57:20 +08:00
if (!this.work) {
2014-10-03 02:35:33 +08:00
this.work = function() { Logger.debug("No work yet."); };
}
2014-06-10 04:39:16 +08:00
this.registerDefaults();
2014-05-08 23:57:20 +08:00
this.initConnections(opts.connection || opts.connections);
this.initDevices(opts.device || opts.devices);
var hasDevices = !!Object.keys(this.devices).length,
hasConnections = !!Object.keys(this.connections).length;
if (hasDevices && !hasConnections) {
throw new Error("No connections specified");
}
2014-05-08 23:57:20 +08:00
for (var n in opts) {
var opt = opts[n];
2014-05-08 23:57:20 +08:00
if (this[n] !== undefined) {
continue;
}
2014-08-06 09:41:57 +08:00
this[n] = opt;
if (typeof opt === 'function' && opts.commands == null) {
this.commands[n] = opt;
2014-08-06 09:41:57 +08:00
}
}
if (opts.commands) {
var cmds = opts.commands;
if (typeof cmds === 'object') {
this.commands = cmds;
2014-08-06 09:41:57 +08:00
}
if (typeof cmds === 'function') {
var result = cmds.call(this, this);
if (typeof result === 'object' && !Array.isArray(result)) {
this.commands = result;
} else {
throw new Error("#commands function must return an object");
}
}
2014-05-08 23:57:20 +08:00
}
2014-08-06 09:41:57 +08:00
var mode = Utils.fetch(Config, 'mode', 'manual');
2014-09-09 04:41:59 +08:00
2014-09-23 05:20:26 +08:00
if (mode === 'auto') {
2014-09-09 05:33:07 +08:00
// run on the next tick, to allow for 'work' event handlers to be set up
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
2014-06-10 04:39:16 +08:00
// Public: Registers the default Drivers and Adaptors with Cylon.
//
// Returns nothing.
Robot.prototype.registerDefaults = function registerDefaults() {
this.registerAdaptor("./test/loopback", "loopback");
this.registerAdaptor("./test/test-adaptor", "test");
this.registerDriver("./test/ping", "ping");
this.registerDriver("./test/test-driver", "test");
};
2014-05-08 23:57:20 +08:00
// Public: Generates a random name for a Robot.
//
// Returns a string name
Robot.randomName = function() {
return "Robot " + (Math.floor(Math.random() * 100000));
};
// Public: Expresses the Robot in a JSON-serializable format
2014-05-08 23:57:20 +08:00
//
// Returns an Object containing Robot data
Robot.prototype.toJSON = function() {
2014-07-30 09:49:35 +08:00
var devices = [],
2014-10-03 02:35:33 +08:00
connections = [],
n;
2014-07-13 01:48:54 +08:00
2014-10-03 02:35:33 +08:00
for (n in this.connections) {
2014-07-30 09:49:35 +08:00
connections.push(this.connections[n]);
}
2014-07-13 01:48:54 +08:00
2014-10-03 02:35:33 +08:00
for (n in this.devices) {
2014-07-30 09:49:35 +08:00
devices.push(this.devices[n]);
}
2014-05-08 23:57:20 +08:00
return {
name: this.name,
connections: connections,
devices: devices,
2014-08-06 09:41:57 +08:00
commands: Object.keys(this.commands)
2014-05-08 23:57:20 +08:00
};
};
// Public: Initializes all connections for the robot
//
// connections - connections to initialize
//
// Returns initialized connections
Robot.prototype.initConnections = function(connections) {
2014-06-10 04:55:44 +08:00
Logger.info("Initializing connections.");
if (connections == null) {
return;
}
2014-05-08 23:57:20 +08:00
connections = [].concat(connections);
2014-07-13 01:48:54 +08:00
connections.forEach(function(conn) {
if (this.connections[conn.name]) {
var original = conn.name;
conn.name = Utils.makeUnique(original, Object.keys(this.connections));
Logger.warn("Connection names must be unique. Renaming '" + original + "' to '" + conn.name + "'");
}
2014-07-13 01:48:54 +08:00
Logger.info("Initializing connection '" + conn.name + "'.");
conn['robot'] = this;
this.connections[conn.name] = new Connection(conn);
}.bind(this));
2014-05-08 23:57:20 +08:00
return this.connections;
};
// Public: Initializes all devices for the robot
//
// devices - devices to initialize
//
// Returns initialized devices
Robot.prototype.initDevices = function(devices) {
2014-06-10 04:55:44 +08:00
Logger.info("Initializing devices.");
if (devices == null) {
return;
}
2014-05-08 23:57:20 +08:00
devices = [].concat(devices);
2014-07-13 01:48:54 +08:00
devices.forEach(function(device) {
if (this.devices[device.name]) {
var original = device.name;
device.name = Utils.makeUnique(original, Object.keys(this.devices));
Logger.warn("Device names must be unique. Renaming '" + original + "' to '" + device.name + "'");
}
2014-06-10 04:55:44 +08:00
Logger.info("Initializing device '" + device.name + "'.");
2014-05-08 23:57:20 +08:00
device['robot'] = this;
2014-07-13 01:48:54 +08:00
this.devices[device.name] = new Device(device);
}.bind(this));
2014-05-08 23:57:20 +08:00
return this.devices;
};
// Public: Starts the Robot working.
//
// Starts the connections, devices, and work.
//
// Returns the result of the work
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-10-04 14:07:07 +08:00
var mode = Utils.fetch(Config, 'workMode', 'async');
2014-06-10 04:55:44 +08:00
Async.series([
this.startConnections,
this.startDevices,
2014-10-04 14:07:07 +08:00
function(callback) {
if (mode === 'async') {
this.startWork();
}
callback(null, true);
}.bind(this)
2014-10-04 15:56:08 +08:00
], function(err, results) {
2014-06-10 04:55:44 +08:00
if (!!err) {
Logger.fatal("An error occured while trying to start the robot:");
Logger.fatal(err);
2014-10-03 02:35:33 +08:00
if (typeof(this.error) === 'function') {
2014-10-01 07:13:20 +08:00
this.error.call(this, err);
}
this.emit('error', err);
2014-06-10 04:55:44 +08:00
}
2014-10-04 15:56:08 +08:00
if (typeof(callback) === 'function') {
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
};
2014-10-04 14:07:07 +08:00
// Public: Starts the Robot's work and triggers a callback
//
// callback - callback function to be triggered
//
// Returns nothing
Robot.prototype.startWork = function() {
Logger.info('Working.');
this.emit('ready', this);
this.work.call(this, this);
this.running = true;
};
2014-05-08 23:57:20 +08:00
// Public: Starts the Robot's connections and triggers a callback
//
// callback - callback function to be triggered
//
// Returns nothing
Robot.prototype.startConnections = function(callback) {
var starters = {};
2014-06-10 04:55:44 +08:00
Logger.info("Starting connections.");
2014-05-08 23:57:20 +08:00
for (var n in this.connections) {
var connection = this.connections[n];
2014-06-10 04:24:42 +08:00
this[n] = connection;
2014-05-08 23:57:20 +08:00
starters[n] = connection.connect;
}
return Async.parallel(starters, callback);
};
// Public: Starts the Robot's devices and triggers a callback
//
// callback - callback function to be triggered
//
// Returns nothing
Robot.prototype.startDevices = function(callback) {
var starters = {};
2014-06-10 04:55:44 +08:00
Logger.info("Starting devices.");
2014-05-08 23:57:20 +08:00
for (var n in this.devices) {
var device = this.devices[n];
2014-06-10 04:24:42 +08:00
this[n] = device;
2014-05-08 23:57:20 +08:00
starters[n] = device.start;
}
return Async.parallel(starters, callback);
};
// Public: Halts the Robot.
//
// Halts the devices, disconnects the connections.
//
// callback - callback to be triggered when the Robot is stopped
//
2014-05-08 23:57:20 +08:00
// Returns nothing
Robot.prototype.halt = function(callback) {
callback = callback || function() {};
var fns = Object.keys(this.devices).map(function(d) {
var device = this.devices[d];
return function(callback) {
device.halt.call(device, callback);
};
}.bind(this));
2014-09-05 01:18:18 +08:00
Async.parallel(fns, function() {
var fns = Object.keys(this.connections).map(function(c) {
2014-09-05 01:18:18 +08:00
var connection = this.connections[c];
return function(callback) {
connection.disconnect.call(connection, callback);
};
}.bind(this));
2014-09-05 01:18:18 +08:00
Async.parallel(fns, callback);
}.bind(this));
this.running = false;
2014-05-08 23:57:20 +08:00
};
// Public: Initialize an adaptor and adds it to @robot.adaptors
//
// adaptorName - module name of adaptor to require
// connection - the Connection that requested the adaptor be required
//
// Returns the adaptor
Robot.prototype.initAdaptor = function(adaptorName, connection, opts) {
if (opts == null) {
opts = {};
}
2014-05-08 23:57:20 +08:00
2014-06-10 04:24:42 +08:00
var adaptor = this.requireAdaptor(adaptorName, opts).adaptor({
2014-05-08 23:57:20 +08:00
name: adaptorName,
connection: connection,
extraParams: opts
});
if (process.env.NODE_ENV === 'test' && !CYLON_TEST) {
2014-06-10 04:24:42 +08:00
var testAdaptor = this.requireAdaptor('test').adaptor({
2014-05-08 23:57:20 +08:00
name: adaptorName,
connection: connection,
extraParams: opts
});
2014-09-05 01:18:18 +08:00
2014-10-03 02:35:33 +08:00
Utils.proxyTestStubs(adaptor.commands, testAdaptor);
for (var prop in adaptor) {
if (typeof adaptor[prop] === 'function' && !testAdaptor[prop]) {
2014-10-03 02:35:33 +08:00
testAdaptor[prop] = function() { return true; };
}
}
return testAdaptor;
2014-05-08 23:57:20 +08:00
} else {
return adaptor;
}
};
// Public: Requires a hardware adaptor and adds it to @robot.adaptors
//
// adaptorName - module name of adaptor to require
//
// Returns the module for the adaptor
Robot.prototype.requireAdaptor = function(adaptorName, opts) {
2014-06-10 04:24:42 +08:00
if (this.adaptors[adaptorName] == null) {
var moduleName = opts.module || adaptorName;
2014-06-10 04:24:42 +08:00
this.registerAdaptor("cylon-" + moduleName, adaptorName);
this.adaptors[adaptorName].register(this);
2014-05-08 23:57:20 +08:00
}
2014-06-10 04:24:42 +08:00
return this.adaptors[adaptorName];
2014-05-08 23:57:20 +08:00
};
// Public: Registers an Adaptor with the Robot
//
// moduleName - name of the Node module to require
// adaptorName - name of the adaptor to register the moduleName under
//
// Returns the registered module name
Robot.prototype.registerAdaptor = function(moduleName, adaptorName) {
if (this.adaptors[adaptorName] == null) {
try {
return this.adaptors[adaptorName] = require(moduleName);
} catch (e) {
if (e.code === "MODULE_NOT_FOUND") {
missingModuleError(moduleName);
2014-02-28 03:17:02 +08:00
} else {
2014-05-08 23:57:20 +08:00
throw e;
2014-02-28 03:17:02 +08:00
}
2014-05-08 23:57:20 +08:00
}
}
};
// Public: Init a hardware driver
//
// driverName - driver name
// device - the Device that requested the driver be initialized
// opts - object containing options when initializing driver
//
// Returns the new driver
Robot.prototype.initDriver = function(driverName, device, opts) {
if (opts == null) {
opts = {};
}
2014-05-08 23:57:20 +08:00
2014-06-10 04:24:42 +08:00
var driver = this.requireDriver(driverName).driver({
2014-05-08 23:57:20 +08:00
name: driverName,
device: device,
extraParams: opts
});
if (process.env.NODE_ENV === 'test' && !CYLON_TEST) {
2014-06-10 04:24:42 +08:00
var testDriver = this.requireDriver('test').driver({
2014-05-08 23:57:20 +08:00
name: driverName,
device: device,
extraParams: opts
});
return Utils.proxyTestStubs(driver.commands, testDriver);
2014-05-08 23:57:20 +08:00
} else {
return driver;
}
};
// Public: Requires module for a driver and adds it to @robot.drivers
//
// driverName - module name of driver to require
//
// Returns the module for driver
Robot.prototype.requireDriver = function(driverName) {
2014-06-10 04:24:42 +08:00
if (this.drivers[driverName] == null) {
this.registerDriver("cylon-" + driverName, driverName);
this.drivers[driverName].register(this);
2014-05-08 23:57:20 +08:00
}
2014-06-10 04:24:42 +08:00
return this.drivers[driverName];
2014-05-08 23:57:20 +08:00
};
// Public: Registers an Driver with the Robot
//
// moduleName - name of the Node module to require
// driverName - name of the driver to register the moduleName under
//
2014-07-16 02:54:16 +08:00
// Returns the registered module name
2014-05-08 23:57:20 +08:00
Robot.prototype.registerDriver = function(moduleName, driverName) {
if (this.drivers[driverName] == null) {
try {
return this.drivers[driverName] = require(moduleName);
} catch (e) {
if (e.code === "MODULE_NOT_FOUND") {
missingModuleError(moduleName);
2014-02-28 03:17:02 +08:00
} else {
2014-05-08 23:57:20 +08:00
throw e;
2014-02-28 03:17:02 +08:00
}
2014-05-08 23:57:20 +08:00
}
}
};
2014-05-08 23:57:20 +08:00
// Public: Returns basic info about the robot as a String
//
// Returns a String
Robot.prototype.toString = function() {
return "[Robot name='" + this.name + "']";
};