cylon/lib/cylon.js

186 lines
4.1 KiB
JavaScript
Raw Normal View History

2013-10-25 05:25:42 +08:00
/*
2015-04-15 09:16:03 +08:00
* Cylon.js - Master Control Program
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-02-28 05:31:54 +08:00
"use strict";
2013-10-25 05:25:42 +08:00
2014-12-16 03:15:29 +08:00
var Async = require("async");
2014-12-16 03:15:29 +08:00
var Logger = require("./logger"),
Robot = require("./robot"),
Config = require("./config"),
Utils = require("./utils"),
_ = require("./utils/helpers");
var EventEmitter = require("events").EventEmitter;
var Cylon = module.exports = new EventEmitter();
Cylon.Logger = Logger;
Cylon.Driver = require("./driver");
Cylon.Adaptor = require("./adaptor");
Cylon.Utils = Utils;
Cylon.IO = {
DigitalPin: require("./io/digital-pin"),
Utils: require("./io/utils")
};
2015-01-08 07:59:35 +08:00
Cylon.apiInstances = [];
Cylon.robots = {};
Cylon.commands = {};
Cylon.events = [ "robot_added", "robot_removed" ];
// Public: Creates a new Robot
//
// opts - hash of Robot attributes
//
// Returns a shiny new Robot
// Examples:
// Cylon.robot
2014-12-16 03:15:29 +08:00
// connection: { name: "arduino", adaptor: "firmata" }
// device: { name: "led", driver: "led", pin: 13 }
//
// work: (me) ->
// me.led.toggle()
Cylon.robot = function robot(opts) {
opts = opts || {};
// check if a robot with the same name exists already
if (opts.name && this.robots[opts.name]) {
var original = opts.name;
opts.name = Utils.makeUnique(original, Object.keys(this.robots));
2014-12-16 03:15:29 +08:00
var str = "Robot names must be unique. Renaming '";
str += original + "' to '" + opts.name + "'";
Logger.warn(str);
}
2014-12-16 03:15:29 +08:00
var bot = new Robot(opts);
this.robots[bot.name] = bot;
this.emit("robot_added", bot.name);
2014-12-16 03:15:29 +08:00
return bot;
};
2015-01-08 07:59:35 +08:00
// Public: Initializes an API instance based on provided options.
//
2014-06-05 02:37:15 +08:00
// Returns nothing
2015-01-08 07:59:35 +08:00
Cylon.api = function api(Server, opts) {
// if only passed options (or nothing), assume HTTP server
if (Server == null || _.isObject(Server) && !_.isFunction(Server)) {
2015-01-08 07:59:35 +08:00
opts = Server;
Server = "http";
}
2015-01-08 07:59:35 +08:00
opts = opts || {};
if (_.isString(Server)) {
2015-01-08 07:59:35 +08:00
var req = "cylon-api-" + Server;
try {
Server = require(req);
} catch (e) {
2015-04-19 08:15:55 +08:00
if (e.code !== "MODULE_NOT_FOUND") {
2015-01-08 07:59:35 +08:00
throw e;
}
2015-04-19 08:15:55 +08:00
[
"Cannot find the " + req + " API module.",
"You may be able to install it: `npm install " + req + "`"
].forEach(_.arity(Logger.fatal, 1));
throw new Error("Missing API plugin - cannot proceed");
2015-01-08 07:59:35 +08:00
}
}
2014-02-28 05:31:54 +08:00
2015-01-08 07:59:35 +08:00
opts.mcp = this;
var instance = new Server(opts);
this.apiInstances.push(instance);
instance.start();
};
// Public: Starts up the API and the robots
//
// Returns nothing
Cylon.start = function start() {
var starters = _.pluck(this.robots, "start");
2014-12-18 06:42:34 +08:00
2014-12-16 03:15:29 +08:00
Async.parallel(starters, function() {
var mode = Utils.fetch(Config, "workMode", "async");
if (mode === "sync") {
_.invoke(this.robots, "startWork");
2014-12-16 03:15:29 +08:00
}
2014-10-04 15:56:08 +08:00
}.bind(this));
};
// Public: Sets the internal configuration, based on passed options
//
// opts - object containing configuration key/value pairs
//
2014-09-16 05:46:24 +08:00
// Returns the current config
Cylon.config = function(opts) {
2015-02-20 09:23:41 +08:00
var loggingChanged = (
opts.logging && Config.logging !== _.extend(Config.logging, opts.logging)
2015-02-20 09:23:41 +08:00
);
if (_.isObject(opts)) {
Config = _.extend(Config, opts);
}
2015-02-20 09:23:41 +08:00
if (loggingChanged) {
Logger.setup();
}
return Config;
};
// Public: Halts the API and the robots
//
// callback - callback to be triggered when Cylon is ready to shutdown
//
// Returns nothing
Cylon.halt = function halt(callback) {
2014-12-16 03:15:29 +08:00
callback = callback || function() {};
2014-12-18 06:42:34 +08:00
var fns = _.pluck(this.robots, "halt");
2014-12-18 06:42:34 +08:00
2014-12-16 03:15:29 +08:00
// if robots can"t shut down quickly enough, forcefully self-terminate
var timeout = Config.haltTimeout || 3000;
Utils.after(timeout, callback);
Async.parallel(fns, callback);
};
2014-07-16 03:27:16 +08:00
Cylon.toJSON = function() {
return {
robots: _.invoke(this.robots, "toJSON"),
2015-02-20 09:23:41 +08:00
commands: Object.keys(this.commands),
events: this.events
2014-12-16 03:15:29 +08:00
};
2014-07-16 03:27:16 +08:00
};
if (process.platform === "win32") {
2015-04-15 12:49:12 +08:00
var readline = require("readline");
var io = { input: process.stdin, output: process.stdout };
readline.createInterface(io).on("SIGINT", function() {
process.emit("SIGINT");
});
2014-06-17 04:09:13 +08:00
}
process.on("SIGINT", function() {
Cylon.halt(function() {
process.kill(process.pid);
});
});