220 lines
5.8 KiB
JavaScript
220 lines
5.8 KiB
JavaScript
/*
|
|
* cylon
|
|
* cylonjs.com
|
|
*
|
|
* Copyright (c) 2013 The Hybrid Group
|
|
* Licensed under the Apache 2.0 license.
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
require('./config');
|
|
require('./utils');
|
|
require('./logger');
|
|
require('./driver');
|
|
require('./adaptor');
|
|
|
|
Logger.setup();
|
|
|
|
// Cylon is the global namespace for the project, and also in charge of
|
|
// maintaining the Master singleton that controls all the robots.
|
|
var Cylon = (function() {
|
|
function Cylon() {}
|
|
|
|
var instance = null;
|
|
|
|
// Public: Fetches singleton instance of Master, or creates a new one if it
|
|
// doesn't already exist
|
|
//
|
|
// Returns a Master instance
|
|
Cylon.getInstance = function() {
|
|
return new Master();
|
|
};
|
|
|
|
// The Master class is our puppeteer that manages all the robots, as well as
|
|
// starting them and the API.
|
|
var Master = (function() {
|
|
// Public: Creates a new Master
|
|
//
|
|
// Returns a Master instance
|
|
function Master() {
|
|
this.self = this;
|
|
|
|
this.api_instance = null;
|
|
this.api_config = { host: '127.0.0.1', port: '3000' };
|
|
this.robots = [];
|
|
|
|
this.robot = bind(this.robot, this);
|
|
|
|
if (process.platform === "win32") {
|
|
var readline = require("readline"),
|
|
io = { input: process.stdin, output: process.stdout };
|
|
|
|
readline.createInterface(io).on("SIGINT", function() {
|
|
process.emit("SIGINT");
|
|
});
|
|
}
|
|
|
|
process.on("SIGINT", function() {
|
|
Cylon.getInstance().halt();
|
|
process.kill();
|
|
});
|
|
}
|
|
|
|
// Public: Creates a new Robot
|
|
//
|
|
// opts - hash of Robot attributes
|
|
//
|
|
// Returns a shiny new Robot
|
|
// Examples:
|
|
// Cylon.robot
|
|
// connection: { name: 'arduino', adaptor: 'firmata' }
|
|
// device: { name: 'led', driver: 'led', pin: 13 }
|
|
//
|
|
// work: (me) ->
|
|
// me.led.toggle()
|
|
Master.prototype.robot = function(opts) {
|
|
var Robot = require("./robot");
|
|
opts.master = this;
|
|
|
|
var robot = new Robot(opts);
|
|
|
|
this.robots.push(robot);
|
|
return robot;
|
|
};
|
|
|
|
// Public: Configures the API host and port based on passed options
|
|
//
|
|
// opts - object containing API options
|
|
// host - host address API should serve from
|
|
// port - port API should listen for requests on
|
|
//
|
|
// Returns the API configuration
|
|
Master.prototype.api = function(opts) {
|
|
if (opts == null) { opts = {}; }
|
|
|
|
var host = opts.host || this.api_config.host,
|
|
port = opts.port || this.api_config.port;
|
|
|
|
this.api_config = { host: host, port: port };
|
|
|
|
return this.api_config;
|
|
};
|
|
|
|
// Public: Finds a particular robot by name
|
|
//
|
|
// name - name of the robot to find
|
|
// callback - optional callback to be executed
|
|
//
|
|
// Returns the found Robot or result of the callback if it's supplied
|
|
Master.prototype.findRobot = function(name, callback) {
|
|
var error,
|
|
robot = null;
|
|
|
|
for (var i = 0; i < this.robots.length; i++) {
|
|
var bot = this.robots[i];
|
|
if (bot.name === name) { robot = bot; }
|
|
}
|
|
|
|
if (robot == null) {
|
|
error = { error: "No Robot found with the name " + name };
|
|
}
|
|
|
|
return callback ? callback(error, robot) : robot;
|
|
};
|
|
|
|
// Public: Finds a particular Robot's device by name
|
|
//
|
|
// robotid - name of the robot to find
|
|
// deviceid - name of the device to find
|
|
// callback - optional callback to be executed
|
|
//
|
|
// Returns the found Device or result of the callback if it's supplied
|
|
Master.prototype.findRobotDevice = function(robotid, deviceid, callback) {
|
|
return this.findRobot(robotid, function(err, robot) {
|
|
var error,
|
|
device = null;
|
|
|
|
if (err) { return callback ? callback(err, robot) : robot }
|
|
|
|
if (robot.devices[deviceid]) { device = robot.devices[deviceid]; }
|
|
|
|
if (device == null) {
|
|
error = { error: "No device found with the name " + deviceid + "." };
|
|
}
|
|
|
|
return callback ? callback(error, device) : device;
|
|
});
|
|
};
|
|
|
|
// Public: Finds a particular Robot's connection by name
|
|
//
|
|
// robotid - name of the robot to find
|
|
// connid - name of the device to find
|
|
// callback - optional callback to be executed
|
|
//
|
|
// Returns the found Connection or result of the callback if it's supplied
|
|
Master.prototype.findRobotConnection = function(robotid, connid, callback) {
|
|
return this.findRobot(robotid, function(err, robot) {
|
|
var error,
|
|
connection = null;
|
|
|
|
if (err) { return callback ? callback(err, robot) : robot }
|
|
|
|
if (robot.connections[connid]) { connection = robot.connections[connid]; }
|
|
|
|
if (connection == null) {
|
|
error = { error: "No connection found with the name " + connid + "." };
|
|
}
|
|
|
|
return callback ? callback(error, connection) : connection;
|
|
});
|
|
};
|
|
|
|
// Public: Starts up the API and the robots
|
|
//
|
|
// Returns nothing
|
|
Master.prototype.start = function() {
|
|
this.startAPI();
|
|
|
|
for (var i = 0; i < this.robots.length; i++) {
|
|
var robot = this.robots[i];
|
|
robot.start();
|
|
}
|
|
};
|
|
|
|
// Public: Halts the API and the robots
|
|
//
|
|
// Returns nothing
|
|
Master.prototype.halt = function() {
|
|
for (var i = 0; i < this.robots.length; i++) {
|
|
var robot = this.robots[i];
|
|
robot.halt();
|
|
}
|
|
};
|
|
|
|
// Creates a new instance of the Cylon API server, or returns the
|
|
// already-existing one.
|
|
//
|
|
// Returns an Cylon.ApiServer instance
|
|
Master.prototype.startAPI = function() {
|
|
var Server = require('./api');
|
|
this.api_config.master = this.self;
|
|
|
|
if (this.api_instance === null) {
|
|
this.api_instance = new Server(this.api_config);
|
|
}
|
|
|
|
return this.api_instance;
|
|
};
|
|
|
|
return Master;
|
|
|
|
})();
|
|
|
|
return Cylon;
|
|
|
|
}).call(this);
|
|
|
|
module.exports = Cylon.getInstance();
|