First commit
authorBenjamin Auder <benjamin.auder@somewhere>
Sat, 27 Jan 2018 17:46:20 +0000 (18:46 +0100)
committerBenjamin Auder <benjamin.auder@somewhere>
Sat, 27 Jan 2018 17:46:20 +0000 (18:46 +0100)
72 files changed:
.gitattributes [new file with mode: 0644]
.gitfat [new file with mode: 0644]
.gitignore [new file with mode: 0644]
LICENSE [new file with mode: 0644]
README.md [new file with mode: 0644]
TODO_assessment_template [new file with mode: 0644]
app.js [new file with mode: 0644]
bin/www [new file with mode: 0755]
config/parameters.js.dist [new file with mode: 0644]
entities/assessment.js [new file with mode: 0644]
entities/course.js [new file with mode: 0644]
entities/user.js [new file with mode: 0644]
gulpfile.js [new file with mode: 0644]
models/assessment.js [new file with mode: 0644]
models/course.js [new file with mode: 0644]
models/user.js [new file with mode: 0644]
package-lock.json [new file with mode: 0644]
package.json [new file with mode: 0644]
public/favicon/LICENSE [new file with mode: 0644]
public/favicon/android-chrome-192x192.png [new file with mode: 0644]
public/favicon/android-chrome-512x512.png [new file with mode: 0644]
public/favicon/apple-touch-icon.png [new file with mode: 0644]
public/favicon/browserconfig.xml [new file with mode: 0644]
public/favicon/favicon-16x16.png [new file with mode: 0644]
public/favicon/favicon-32x32.png [new file with mode: 0644]
public/favicon/favicon.ico [new file with mode: 0644]
public/favicon/manifest.json [new file with mode: 0644]
public/favicon/mstile-150x150.png [new file with mode: 0644]
public/favicon/safari-pinned-tab.svg [new file with mode: 0644]
public/javascripts/assessment.js [new file with mode: 0644]
public/javascripts/course.js [new file with mode: 0644]
public/javascripts/courseList.js [new file with mode: 0644]
public/javascripts/login.js [new file with mode: 0644]
public/javascripts/monitor.js [new file with mode: 0644]
public/javascripts/utils/dialog.js [new file with mode: 0644]
public/javascripts/utils/sha1.js [new file with mode: 0644]
public/javascripts/utils/socketMessages.js [new file with mode: 0644]
public/javascripts/utils/validation.js [new file with mode: 0644]
public/stylesheets/assessment.css [new file with mode: 0644]
public/stylesheets/course.css [new file with mode: 0644]
public/stylesheets/courseList.css [new file with mode: 0644]
public/stylesheets/index.css [new file with mode: 0644]
public/stylesheets/layout.css [new file with mode: 0644]
public/stylesheets/login.css [new file with mode: 0644]
public/stylesheets/monitor.css [new file with mode: 0644]
public/vendor/prism/prism-components.zip [new file with mode: 0644]
public/vendor/prism/prism.css [new file with mode: 0644]
public/vendor/prism/prism.js [new file with mode: 0644]
routes/all.js [new file with mode: 0644]
routes/assessments.js [new file with mode: 0644]
routes/courses.js [new file with mode: 0644]
routes/pages.js [new file with mode: 0644]
routes/users.js [new file with mode: 0644]
setup/README [new file with mode: 0644]
setup/database.js [new file with mode: 0644]
setup/students.sample.csv [new file with mode: 0644]
sockets.js [new file with mode: 0644]
utils/access.js [new file with mode: 0644]
utils/database.js [new file with mode: 0644]
utils/mailer.js [new file with mode: 0644]
utils/tokenGenerator.js [new file with mode: 0644]
views/assessment.pug [new file with mode: 0644]
views/course-list.pug [new file with mode: 0644]
views/course.pug [new file with mode: 0644]
views/enable-js.pug [new file with mode: 0644]
views/error.pug [new file with mode: 0644]
views/index.pug [new file with mode: 0644]
views/layout.pug [new file with mode: 0644]
views/login.pug [new file with mode: 0644]
views/monitor.pug [new file with mode: 0644]
views/no-devtools.pug [new file with mode: 0644]
views/withQuestions.pug [new file with mode: 0644]

diff --git a/.gitattributes b/.gitattributes
new file mode 100644 (file)
index 0000000..53b9f2c
--- /dev/null
@@ -0,0 +1,5 @@
+*.pdf filter=fat
+*.zip filter=fat
+*.tar.xz filter=fat
+*.png filter=fat
+*.ico filter=fat
diff --git a/.gitfat b/.gitfat
new file mode 100644 (file)
index 0000000..f20d0fb
--- /dev/null
+++ b/.gitfat
@@ -0,0 +1,2 @@
+[rsync]
+remote = gitfat@auder.net:~/files/qomet
diff --git a/.gitignore b/.gitignore
new file mode 100644 (file)
index 0000000..d29bf32
--- /dev/null
@@ -0,0 +1,4 @@
+/node_modules/
+*.swp
+/config/parameters.js
+/public/vendor/prism/components/
diff --git a/LICENSE b/LICENSE
new file mode 100644 (file)
index 0000000..ee75e4c
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,15 @@
+ISC License
+
+Copyright (c) 2018, Benjamin Auder
+
+Permission to use, copy, modify, and/or distribute this software for any
+purpose with or without fee is hereby granted, provided that the above
+copyright notice and this permission notice appear in all copies.
+
+THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
diff --git a/README.md b/README.md
new file mode 100644 (file)
index 0000000..d79ff0d
--- /dev/null
+++ b/README.md
@@ -0,0 +1,66 @@
+# qomet
+
+### Questions Ouvertes ou à options Multiples pour l'Évaluation des éTudiants
+
+Or "... pour Examens sur inTernet", in french.
+
+In english, just revert the acronym:
+"sTudents Evaluation with Multiple chOices or Open Questions (or ...inTernet Exams with).
+
+## Features
+
+Allow teachers to create courses, containing assessments. Each of them can be public, or
+restricted to a classroom (identification by student ID).
+Individual answers to an exam are monitored in real time, and feedback is sent
+to each participant in the end (answers, computing grade).
+Once a series of exam is over, the teacher can get all grades in CSV format from course page.
+
+*Note:* for now the monitoring + socket part is still unimplemented,
+and exams composition is limited to single question exercises.
+
+## Installation
+
+See setup/README
+
+## Usage
+
+TODO: write tutorial, maybe a demo video.
+
+*Note about exams:*
+Once an assessment is started, it's impossible to quit and restart using another browser,
+because a password stored in cookies need to be sent with every request.
+So under normal circumstances it's also impossible for a student to continue the exam of another.
+(The password is destroyed when exam ends or when the teacher decide to finish assessment).
+
+## Limitations
+
+Version "standard classroom": some potential cheating ways,
+ - headless browsers with renamed http-user-agent; difficult to counter with 100% confidence
+ - block JS script using e.g. μblock, then re-inject the script cleaned of listeners
+ - intercept HTTP response to "start quiz" signal, re-compose the page without listeners and run
+
+The only way to garanty 0 internet cheat is to use some SELinux configuration in kiosk mode
+with just one safe web browser enabled, e.g. [surf](https://surf.suckless.org/).
+Not that more traditional ways of cheating may still be used (phones, talking, signs, memos...)
+
+## Alternative softwares
+
+ * [moodle](https://moodle.org)<br/>
+  Full-featured (open source!) project to manage learning activities.
+  Too big for my purpose; however qomet might be re-thought as a moodle plugin
+  (although [at least one](https://moodle.org/plugins/mod_exam) already exists for this task).
+
+ * [evalbox](https://evalbox.com/)<br/>
+  The closest to my goals, but only for simple quizzes, and not actively developed anymore.
+
+ * [wims](http://wims.unice.fr/~wims/)<br/>
+  Full-featured (and open source) training center for students, with various types of exercises,
+  possibly in exam mode too.
+  The spirit, however, is more "enhanced homework" than "internet exams".
+
+ * [socrative](https://socrative.com/)<br/>
+  Nice looking realtime feedback (lacking in evalbox), but thought for interactive classes.
+  In this perspective, I also found [educaplay](https://www.educaplay.com) appealing.
+
+ * [testmoz](https://testmoz.com/)<br/>
+  Old-fashioned look, lacking some features. Still interesting to set-up a quick test.
diff --git a/TODO_assessment_template b/TODO_assessment_template
new file mode 100644 (file)
index 0000000..f44a163
--- /dev/null
@@ -0,0 +1,36 @@
+TODO: format général TXT: (compilé en JSON)
+
+10 (time)
+1 (fixed)
+Introduction (multiline, from third line ; \n --> <br>)
+
+[Intro q1, multiline]
+
+q1 txt
+
+answer q1 (multiline txt)
+
+q2 intro (multiline)
+
+  q2.1 [intro optional]
+
+       q2.1 txt
+
+       q2.1 options:
+       + good
+       - bad
+       - bad ...etc
+
+       answer (integer array, one line)
+
+  q2.2 [intro optional]
+
+       q2.2 txt
+
+       answer (html multiline)
+
+Conclusion (last block)
+
+=====
+
+Seems that GUI would be easier, then summary in YAML file + parse from YAML
diff --git a/app.js b/app.js
new file mode 100644 (file)
index 0000000..7dc3b41
--- /dev/null
+++ b/app.js
@@ -0,0 +1,58 @@
+const express = require('express');
+const app = express();
+const path = require('path');
+const cookieParser = require('cookie-parser');
+const favicon = require('serve-favicon');
+const logger = require('morgan');
+const bodyParser = require('body-parser');
+const params = require(path.join(__dirname, "config", "parameters"));
+
+app.set('views', path.join(__dirname, 'views'));
+app.set('view engine', 'pug');
+app.use(favicon(path.join(__dirname, "public", "favicon", "favicon.ico")));
+if (app.get('env') === 'development')
+{
+       // Full logging in development mode
+       app.use(logger('dev'));
+}
+else
+{
+       app.set('trust proxy', true); //http://dev.rdybarra.com/2016/06/23/Production-Logging-With-Morgan-In-Express/
+       // In prod, only log error responses (https://github.com/expressjs/morgan)
+       app.use(logger('combined', {
+               skip: function (req, res) { return res.statusCode < 400 }
+       }));
+}
+app.use(bodyParser.json());
+app.use(bodyParser.urlencoded({ extended: false }));
+app.use(cookieParser());
+app.use(express.static(path.join(__dirname, 'public')));
+
+// Before any request, check cookies
+app.use(function(req, res, next) {
+       res.locals.loggedIn = !!req.cookies.token;
+       res.locals.myInitials = req.cookies.myInitials; //may be undefined
+       next();
+});
+
+// Routing
+let routes = require(path.join(__dirname, "routes", "all"));
+app.use("/", routes);
+
+// catch 404 and forward to error handler
+app.use(function(req, res, next) {
+  let err = new Error('Not Found');
+  err.status = 404;
+  next(err);
+});
+
+// Error handler
+app.use(function(err, req, res, next) {
+  // Set locals, only providing error in development
+  res.locals.message = err.message;
+  res.locals.error = req.app.get('env') === 'development' ? err : {};
+  res.status(err.status || 500);
+  res.render('error');
+});
+
+module.exports = app;
diff --git a/bin/www b/bin/www
new file mode 100755 (executable)
index 0000000..c964949
--- /dev/null
+++ b/bin/www
@@ -0,0 +1,107 @@
+#!/usr/bin/env node
+
+/**
+ * Module dependencies.
+ */
+
+var app = require('../app');
+var http = require('http');
+
+/**
+ * Get port from environment and store in Express.
+ */
+
+let port = normalizePort(process.env.PORT || '3000');
+app.set('port', port);
+
+/**
+ * Create HTTP server.
+ */
+
+let server = http.createServer(app);
+
+/*
+ * CRON tasks
+ */
+
+var cron = require('node-cron');
+var UserModel = require("../models/user");
+cron.schedule('0 0 0 * * *', function() {
+       // Remove unlogged users every 24h
+       UserModel.cleanUsersDb();
+});
+
+/**
+ * Listen on provided port, on all network interfaces.
+ */
+
+server.listen(port);
+server.on('error', onError);
+server.on('listening', onListening);
+let io = require('socket.io').listen(server); //sockets too
+//https://stackoverflow.com/a/24610678/4640434
+
+/*
+ * Sockets handling
+ */
+
+require('../sockets')(io);
+
+/**
+ * Normalize a port into a number, string, or false.
+ */
+
+function normalizePort(val)
+{
+  let port = parseInt(val, 10);
+
+  if (isNaN(port)) // named pipe
+    return val;
+
+  if (port >= 0) // port number
+    return port;
+
+  return false;
+}
+
+/**
+ * Event listener for HTTP server "error" event.
+ */
+
+function onError(error)
+{
+  if (error.syscall !== 'listen')
+    throw error;
+
+  let bind = typeof port === 'string'
+    ? 'Pipe ' + port
+    : 'Port ' + port;
+
+  // handle specific listen errors with friendly messages
+  switch (error.code)
+       {
+    case 'EACCES':
+      console.error(bind + ' requires elevated privileges');
+      process.exit(1);
+      break;
+    case 'EADDRINUSE':
+      console.error(bind + ' is already in use');
+      process.exit(1);
+      break;
+    default:
+      throw error;
+  }
+}
+
+/**
+ * Event listener for HTTP server "listening" event.
+ */
+
+function onListening()
+{
+  let addr = server.address();
+  let bind = typeof addr === 'string'
+    ? 'pipe ' + addr
+    : 'port ' + addr.port;
+  console.log('Listening on ' + bind);
+}
diff --git a/config/parameters.js.dist b/config/parameters.js.dist
new file mode 100644 (file)
index 0000000..ae0073d
--- /dev/null
@@ -0,0 +1,36 @@
+var Parameters = {};
+
+// For mail sending. WARNING: *no trailng slash*
+Parameters.siteURL = "http://localhost";
+
+// Lifespan of a (login) cookie
+Parameters.cookieExpire = 183*24*3600*1000; //6 months in milliseconds
+
+// Characters in a login token, and period of validity (in milliseconds)
+Parameters.token = {
+       length: 16,
+       expire: 1000*60*30, //30 minutes in milliseconds
+};
+
+// Whitelist of emails (full addresses or suffixes). Leave blank to accept all
+Parameters.whitelist = [
+       "some.email@some.domain.com",
+       "another.domain.org",
+];
+
+// msmtp account name adn address from, to send (login) emails
+Parameters.mail = {
+       account: "msmtpAccount",
+       from: "addressFrom",
+};
+
+// Database settings: see https://docs.mongodb.com/manual/reference/connection-string/
+Parameters.db = {
+       user: "username",
+       password: "password",
+       host: "localhost",
+       port: "27017",
+       name: "dbname",
+};
+
+module.exports = Parameters;
diff --git a/entities/assessment.js b/entities/assessment.js
new file mode 100644 (file)
index 0000000..2104193
--- /dev/null
@@ -0,0 +1,184 @@
+const db = require("../utils/database");
+
+const AssessmentEntity =
+{
+       /*
+        * Structure:
+        *   _id: BSON id
+        *   cid: course ID
+        *   name: varchar
+        *   active: boolean true/false
+        *   mode: secure | exam | open (decreasing security)
+        *   fixed: bool (questions in fixed order; default: false)
+        *   display: "one" or "all" (generally "all" for open questions, but...)
+        *   time: 0, //<=0 means "untimed"; otherwise, time in seconds
+        *   introduction: "",
+        *   conclusion: "https://www.youtube.com/watch?v=6-9AjToJYuw",
+        *   coefficient: number, default 1
+        *   questions: array of
+        *     index: for paper test, like 2.1.a (?!); and quiz: 0, 1, 2, 3...
+        *     wording: varchar (HTML)
+        *     options: array of varchar --> if present, question type == quiz!
+        *     fixed: bool, options in fixed order (default: false)
+        *     answer: array of integers (for quiz) or html text (for paper); striped in exam mode
+        *     active: boolean, is question in current assessment? --> striped if inactive!
+        *     points: points for this question (default 1)
+        *   papers : array of
+        *     number: student number
+        *     inputs: array of indexed arrays of integers (or html text if not quiz)
+        *     startTime, endTime
+        *     password: random string identifying student for exam session TEMPORARY
+        */
+
+       getById: function(aid, callback)
+       {
+               db.assessments.findOne(
+                       { _id: aid },
+                       callback
+               );
+       },
+
+       getByPath: function(cid, name, callback)
+       {
+               db.assessments.findOne(
+                       {
+                               cid: cid,
+                               name: name,
+                       },
+                       callback
+               );
+       },
+
+       insert: function(cid, name, callback)
+       {
+               db.assessments.insert(
+                       {
+                               name: name,
+                               cid: cid,
+                               active: false,
+                               mode: "exam",
+                               fixed: false,
+                               display: "one",
+                               time: 0,
+                               introduction: "",
+                               conclusion: "",
+                               coefficient: 1,
+                               questions: [ ],
+                               papers: [ ],
+                       },
+                       callback
+               );
+       },
+
+       getByCourse: function(cid, callback)
+       {
+               db.assessments.find(
+                       { cid: cid },
+                       callback
+               );
+       },
+
+       // arg: full assessment without _id field
+       replace: function(aid, assessment, cb)
+       {
+               // Should be: (but unsupported by mongojs)
+//             db.assessments.replaceOne(
+//                     { _id: aid },
+//                     assessment,
+//                     cb
+//             );
+               // Temporary workaround:
+               db.assessments.update(
+                       { _id: aid },
+                       { $set: assessment },
+                       cb
+               );
+       },
+
+       getQuestions: function(aid, callback)
+       {
+               db.assessments.findOne(
+                       { _id: aid },
+                       { questions: 1},
+                       (err,res) => {
+                               callback(err, !!res ? res.questions : null);
+                       }
+               );
+       },
+
+       startSession: function(aid, number, password, callback)
+       {
+               // TODO: security, do not re-do tasks if already done
+               db.assessments.update(
+                       { _id: aid },
+                       { $push: { papers: {
+                               number: number,
+                               startTime: Date.now(),
+                               endTime: undefined,
+                               password: password,
+                               inputs: [ ], //TODO: this is stage 1, stack indexed answers.
+                               // then build JSON tree for easier access / correct
+                       }}},
+                       callback
+               );
+       },
+
+       // https://stackoverflow.com/questions/27874469/mongodb-push-in-nested-array
+       setInput: function(aid, number, password, input, callback) //input: index + arrayOfInt (or txt)
+       {
+               db.assessments.update(
+                       {
+                               _id: aid,
+                               "papers.number": number,
+                               "papers.password": password,
+                       },
+                       { $push: { "papers.$.inputs": input } },
+                       callback
+               );
+       },
+
+       endAssessment: function(aid, number, password, callback)
+       {
+               db.assessments.update(
+                       {
+                               _id: aid,
+                               "papers.number": number,
+                               "papers.password": password,
+                       },
+                       { $set: {
+                               "papers.$.endTime": Date.now(),
+                               "papers.$.password": "",
+                       } },
+                       callback
+               );
+       },
+
+       getConclusion: function(aid, callback)
+       {
+               db.assessments.findOne(
+                       { _id: aid },
+                       { conclusion: 1},
+                       (err,res) => {
+                               callback(err, !!res ? res.conclusion : null);
+                       }
+               );
+       },
+
+       remove: function(aid, cb)
+       {
+               db.assessments.remove(
+                       { _id: aid },
+                       cb
+               );
+       },
+
+       removeGroup: function(cid, cb)
+       {
+               db.assessments.remove(
+                       { cid: cid },
+                       cb
+               );
+       },
+}
+
+module.exports = AssessmentEntity;
diff --git a/entities/course.js b/entities/course.js
new file mode 100644 (file)
index 0000000..66fcaee
--- /dev/null
@@ -0,0 +1,100 @@
+const db = require("../utils/database");
+
+const CourseEntity =
+{
+       /*
+        * Structure:
+        *   _id: BSON id
+        *   uid: prof ID
+        *   code: varchar
+        *   description: varchar
+        *   password: monitoring password hash
+        *   students: array of
+        *     number: student number
+        *     forename: varchar
+        *     name: varchar
+        *     group: integer
+        */
+
+       getByUser: function(uid, callback)
+       {
+               db.courses.find(
+                       { uid: uid },
+                       callback
+               );
+       },
+
+       getById: function(cid, callback)
+       {
+               db.courses.findOne(
+                       { _id: cid },
+                       callback
+               );
+       },
+
+       getByPath: function(uid, code, callback)
+       {
+               db.courses.findOne(
+                       {
+                               $and: [
+                                       { uid: uid },
+                                       { code: code },
+                               ]
+                       },
+                       callback
+               );
+       },
+
+       insert: function(uid, code, description, cb)
+       {
+               db.courses.insert(
+                       {
+                               uid: uid,
+                               code: code,
+                               description: description,
+                               students: [ ],
+                       },
+                       cb);
+       },
+
+       setStudents: function(cid, students, cb)
+       {
+               db.courses.update(
+                       { _id: cid },
+                       { $set: { students: students } },
+                       cb
+               );
+       },
+
+       // Note: return { students: { ... } }, pointing on the requested row
+       getStudent: function(cid, number, cb)
+       {
+               db.courses.findOne(
+                       { _id: cid },
+                       {
+                               _id: 0,
+                               students: { $elemMatch: {number: number} }
+                       },
+                       cb
+               );
+       },
+
+       setPassword: function(cid, pwd, cb)
+       {
+               db.courses.update(
+                       { _id: cid },
+                       { $set: { password: pwd } },
+                       cb
+               );
+       },
+
+       remove: function(cid, cb)
+       {
+               db.courses.remove(
+                       { _id: cid },
+                       cb
+               );
+       },
+}
+
+module.exports = CourseEntity;
diff --git a/entities/user.js b/entities/user.js
new file mode 100644 (file)
index 0000000..1026eaf
--- /dev/null
@@ -0,0 +1,146 @@
+const db = require("../utils/database");
+
+const UserEntity =
+{
+       /*
+        * Structure:
+        *   _id: BSON id
+        *   ** Strings, identification informations:
+        *   email
+        *   forename
+        *   name
+        *   initials : computed, Benjamin Auder --> ba ...etc
+        *   loginToken: {
+        *     value: string
+        *     timestamp: datetime (validity)
+        *     ip: address of requesting machine
+        *   }
+        *   sessionTokens (array): cookie identification
+        */
+
+       getInitialsByPrefix: function(prefix, cb)
+       {
+               db.users.find(
+                       { initials: new RegExp("^" + prefix) },
+                       { initials: 1, _id: 0 },
+                       cb
+               );
+       },
+
+       insert: function(newUser, cb)
+       {
+               db.users.insert(Object.assign({},
+                       newUser,
+                       {
+                               loginToken: { },
+                               sessionTokens: [ ],
+                       }),
+                       cb
+               );
+       },
+
+       getByLoginToken: function(token, cb)
+       {
+               db.users.findOne(
+                       { "loginToken.value": token },
+                       cb
+               );
+       },
+
+       getBySessionToken: function(token, cb)
+       {
+               db.users.findOne(
+                       { sessionTokens: token},
+                       cb
+               );
+       },
+
+       getById: function(uid, cb)
+       {
+               db.users.findOne(
+                       { _id: uid },
+                       cb
+               );
+       },
+
+       getByEmail: function(email, cb)
+       {
+               db.users.findOne(
+                       { email: email },
+                       cb
+               );
+       },
+
+       getByInitials: function(initials, cb)
+       {
+               db.users.findOne(
+                       { initials: initials },
+                       cb
+               );
+       },
+
+       getUnlogged: function(cb)
+       {
+               var tsNow = new Date().getTime();
+               // 86400000 = 24 hours in milliseconds
+               var day = 86400000;
+               db.users.find({}, (err,userArray) => {
+                       let unlogged = userArray.filter( u => {
+                               return u.sessionTokens.length==0 && u._id.getTimestamp().getTime() + day < tsNow;
+                       });
+                       cb(err, unlogged);
+               });
+       },
+
+       getAll: function(cb)
+       {
+               db.users.find({}, cb);
+       },
+
+       setLoginToken: function(token, uid, ip, cb)
+       {
+               db.users.update(
+                       { _id: uid },
+                       { $set: { loginToken: {
+                                       value: token,
+                                       timestamp: new Date().getTime(),
+                                       ip: ip,
+                               }}
+                       },
+                       cb
+               );
+       },
+
+       setSessionToken: function(token, uid, cb)
+       {
+               // Also empty the login token to invalidate future attempts
+               db.users.update(
+                       { _id: uid },
+                       {
+                               $set: { loginToken: {} },
+                               $push: { sessionTokens: {
+                                       $each: [token],
+                                       $slice: -7 //only allow 7 simultaneous connections per user (TODO?)
+                               }}
+                       },
+                       cb
+               );
+       },
+
+       removeToken: function(uid, token, cb)
+       {
+               db.users.update(
+                       { _id: uid },
+                       { $pull: {sessionTokens: token} },
+                       cb
+               );
+       },
+
+       // TODO: later, allow account removal
+       remove: function(uids)
+       {
+               db.users.remove({_id: uids});
+       },
+}
+
+module.exports = UserEntity;
diff --git a/gulpfile.js b/gulpfile.js
new file mode 100644 (file)
index 0000000..a1286c5
--- /dev/null
@@ -0,0 +1,19 @@
+var gulp = require('gulp');
+var nodemon = require('gulp-nodemon'); //reload server on changes
+
+var nodemonOptions = {
+       script: 'bin/www',
+       ext: 'js',
+       env: { 'NODE_ENV': 'development' },
+       verbose: true,
+       watch: ['./','config','utils','routes','models','entities']
+};
+
+// TODO: tasks for uglify / sass / use webpack
+
+gulp.task('server', function () {
+       nodemon(nodemonOptions)
+       .on('restart', function () {
+               console.log('restarted!')
+       });
+});
diff --git a/models/assessment.js b/models/assessment.js
new file mode 100644 (file)
index 0000000..3f89cc9
--- /dev/null
@@ -0,0 +1,85 @@
+const AssessmentEntity = require("../entities/assessment");
+const CourseEntity = require("../entities/course");
+const ObjectId = require("bson-objectid");
+const UserEntity = require("../entities/user");
+const TokenGen = require("../utils/tokenGenerator");
+
+const AssessmentModel =
+{
+       getByRefs: function(initials, code, name, cb)
+       {
+               UserEntity.getByInitials(initials, (err,user) => {
+                       if (!!err || !user)
+                               return cb(err || {errmsg: "User not found"});
+                       CourseEntity.getByPath(user._id, code, (err2,course) => {
+                               if (!!err2 || !course)
+                                       return cb(err2 || {errmsg: "Course not found"});
+                               AssessmentEntity.getByPath(course._id, name, (err3,assessment) => {
+                                       if (!!err3 || !assessment)
+                                               return cb(err3 || {errmsg: "Assessment not found"});
+                                       cb(null,assessment);
+                               });
+                       });
+               });
+       },
+
+       add: function(uid, cid, name, cb)
+       {
+               // 1) Check that course is owned by user of ID uid
+               CourseEntity.getById(cid, (err,course) => {
+                       if (!!err || !course)
+                               return cb({errmsg: "Course retrieval failure"});
+                       if (!course.uid.equals(uid))
+                               return cb({errmsg:"Not your course"},undefined);
+                       // 2) Insert new blank assessment
+                       AssessmentEntity.insert(cid, name, cb);
+               });
+       },
+
+       update: function(uid, assessment, cb)
+       {
+               const qid = ObjectId(assessment._id);
+               // 1) Check that assessment is owned by user of ID uid
+               AssessmentEntity.getById(qid, (err,assessmentOld) => {
+                       if (!!err || !assessmentOld)
+                               return cb({errmsg: "Assessment retrieval failure"});
+                       CourseEntity.getById(ObjectId(assessmentOld.cid), (err2,course) => {
+                               if (!!err2 || !course)
+                                       return cb({errmsg: "Course retrieval failure"});
+                               if (!course.uid.equals(uid))
+                                       return cb({errmsg:"Not your course"},undefined);
+                               // 2) Replace assessment
+                               delete assessment["_id"];
+                               assessment.cid = ObjectId(assessment.cid);
+                               AssessmentEntity.replace(qid, assessment, cb);
+                       });
+               });
+       },
+
+       // Set password in responses collection
+       startSession: function(aid, number, cb)
+       {
+               const password = TokenGen.generate(12); //arbitrary number, 12 seems enough...
+               AssessmentEntity.getQuestions(aid, (err,questions) => {
+                       AssessmentEntity.startSession(aid, number, password, (err2,ret) => {
+                               cb(err, {
+                                       questions: questions,
+                                       password: password,
+                               });
+                       });
+               });
+       },
+
+       endSession: function(aid, number, password, cb)
+       {
+               AssessmentEntity.endAssessment(aid, number, password, (err,ret) => {
+                       if (!!err || !ret)
+                               return cb(err,ret);
+                       AssessmentEntity.getConclusion(aid, (err2,conclusion) => {
+                               cb(err2, {conclusion:conclusion});
+                       });
+               });
+       },
+};
+
+module.exports = AssessmentModel;
diff --git a/models/course.js b/models/course.js
new file mode 100644 (file)
index 0000000..53419c1
--- /dev/null
@@ -0,0 +1,74 @@
+const CourseEntity = require("../entities/course");
+const UserEntity = require("../entities/user");
+const AssessmentEntity = require("../entities/assessment");
+
+const CourseModel =
+{
+       getByInitials: function(initials, callback)
+       {
+               UserEntity.getByInitials(initials, (err,user) => {
+                       if (!!err || !user)
+                               callback(err, []);
+                       else
+                       {
+                               CourseEntity.getByUser(user._id, (err2,courseArray) => {
+                                       callback(err2, courseArray);
+                               });
+                       }
+               });
+       },
+
+       getByRefs: function(initials, code, callback)
+       {
+               UserEntity.getByInitials(initials, (err,user) => {
+                       if (!!err || !user)
+                               callback(err, []);
+                       else
+                       {
+                               CourseEntity.getByPath(user._id, code, (err2,course) => {
+                                       callback(err2, course);
+                               });
+                       }
+               });
+       },
+
+       importStudents: function(uid, cid, students, cb)
+       {
+               // 1) check if uid == course uid
+               CourseEntity.getById(cid, (err,course) => {
+                       if (!!err || !course || !course.uid.equals(uid))
+                               return cb({errmsg:"Not your course"},{});
+                       // 2) Set students
+                       CourseEntity.setStudents(cid, students, cb);
+               });
+       },
+
+       setPassword: function(uid, cid, pwd, cb)
+       {
+               // 1) check if uid == course uid
+               CourseEntity.getById(cid, (err,course) => {
+                       if (!!err || !course || !course.uid.equals(uid))
+                               return cb({errmsg:"Not your course"},{});
+                       // 2) Insert new student (overwrite if number already exists)
+                       CourseEntity.setPassword(cid, pwd, cb);
+               });
+       },
+
+       remove: function(uid, cid, cb)
+       {
+               // 1) check if uid == course uid
+               CourseEntity.getById(cid, (err,course) => {
+                       if (!!err || !course || !course.uid.equals(uid))
+                               return cb({errmsg:"Not your course"},{});
+                       // 2) remove all associated assessments
+                       AssessmentEntity.removeGroup(cid, (err2,ret) => {
+                               if (!!err)
+                                       return cb(err,{});
+                               // 3) remove course (with its students)
+                               CourseEntity.remove(cid, cb);
+                       });
+               });
+       },
+}
+
+module.exports = CourseModel;
diff --git a/models/user.js b/models/user.js
new file mode 100644 (file)
index 0000000..f495bdc
--- /dev/null
@@ -0,0 +1,60 @@
+const UserEntity = require("../entities/user");
+const params = require("../config/parameters");
+
+const UserModel =
+{
+       create: function(newUser, callback)
+       {
+               // Determine initials from forename+name
+               let forenameParts = newUser.forename.split(/[ -]+/);
+               let nameParts = newUser.name.split(/[ -]+/);
+               let initials =
+                       forenameParts.map( n => { return n.charAt(0).toLowerCase(); }).join("") +
+                       nameParts.map( n => { return n.charAt(0).toLowerCase(); }).join("");
+               // First retrieve all users with similar prefix initials
+               UserEntity.getInitialsByPrefix(initials, (err,userArray) => {
+                       if (!!userArray && userArray.length == 1)
+                               initials = initials + "2"; //thus number == users count for this hash
+                       else if (!!userArray && userArray.length > 1)
+                       {
+                               // Pick the highest number after initials (if any), and increment
+                               let numbers = userArray.map( u => {
+                                       let digitMatch = u.initials.match(/[0-9]/);
+                                       if (!digitMatch)
+                                               return 1; //irrelevant
+                                       let firstDigit = digitMatch.index;
+                                       return parseInt(u.initials.slice(digitMatch.index));
+                               });
+                               initials = initials + (Math.max(...numbers)+1);
+                       }
+                       Object.assign(newUser, {initials: initials});
+                       UserEntity.insert(newUser, callback);
+               });
+       },
+
+       whitelistCheck: function(email)
+       {
+               if (params.whitelist.length == 0)
+                       return true; //no whitelist, everyone allowed
+               for (let w of params.whitelist)
+               {
+                       if ((w.indexOf('@') >= 0 && w==email) || !!email.match(new RegExp(w+"$")))
+                               return true;
+               }
+               return false;
+       },
+
+       logout: function(uid, token, cb)
+       {
+               UserEntity.removeToken(uid, token, cb);
+       },
+
+       cleanUsersDb: function()
+       {
+               UserEntity.getUnlogged( (err,unlogged) => {
+                       UserEntity.remove(unlogged);
+               });
+       },
+}
+
+module.exports = UserModel;
diff --git a/package-lock.json b/package-lock.json
new file mode 100644 (file)
index 0000000..7cd2e0e
--- /dev/null
@@ -0,0 +1,5833 @@
+{
+  "name": "qcm",
+  "version": "0.1.0",
+  "lockfileVersion": 1,
+  "requires": true,
+  "dependencies": {
+    "abbrev": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
+      "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==",
+      "dev": true
+    },
+    "accepts": {
+      "version": "1.3.4",
+      "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.4.tgz",
+      "integrity": "sha1-hiRnWMfdbSGmR0/whKR0DsBesh8=",
+      "requires": {
+        "mime-types": "2.1.17",
+        "negotiator": "0.6.1"
+      }
+    },
+    "acorn": {
+      "version": "3.3.0",
+      "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz",
+      "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo="
+    },
+    "acorn-globals": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-3.1.0.tgz",
+      "integrity": "sha1-/YJw9x+7SZawBPqIDuXUZXOnMb8=",
+      "requires": {
+        "acorn": "4.0.13"
+      },
+      "dependencies": {
+        "acorn": {
+          "version": "4.0.13",
+          "resolved": "https://registry.npmjs.org/acorn/-/acorn-4.0.13.tgz",
+          "integrity": "sha1-EFSVrlNh1pe9GVyCUZLhrX8lN4c="
+        }
+      }
+    },
+    "after": {
+      "version": "0.8.2",
+      "resolved": "https://registry.npmjs.org/after/-/after-0.8.2.tgz",
+      "integrity": "sha1-/ts5T58OAqqXaOcCvaI7UF+ufh8="
+    },
+    "align-text": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/align-text/-/align-text-0.1.4.tgz",
+      "integrity": "sha1-DNkKVhCT810KmSVsIrcGlDP60Rc=",
+      "requires": {
+        "kind-of": "3.2.2",
+        "longest": "1.0.1",
+        "repeat-string": "1.6.1"
+      }
+    },
+    "amdefine": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz",
+      "integrity": "sha1-SlKCrBZHKek2Gbz9OtFR+BfOkfU="
+    },
+    "ansi-align": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-2.0.0.tgz",
+      "integrity": "sha1-w2rsy6VjuJzrVW82kPCx2eNUf38=",
+      "dev": true,
+      "requires": {
+        "string-width": "2.1.1"
+      }
+    },
+    "ansi-gray": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/ansi-gray/-/ansi-gray-0.1.1.tgz",
+      "integrity": "sha1-KWLPVOyXksSFEKPetSRDaGHvclE=",
+      "dev": true,
+      "requires": {
+        "ansi-wrap": "0.1.0"
+      }
+    },
+    "ansi-regex": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+      "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=",
+      "dev": true
+    },
+    "ansi-styles": {
+      "version": "2.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz",
+      "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=",
+      "dev": true
+    },
+    "ansi-wrap": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/ansi-wrap/-/ansi-wrap-0.1.0.tgz",
+      "integrity": "sha1-qCJQ3bABXponyoLoLqYDu/pF768=",
+      "dev": true
+    },
+    "anymatch": {
+      "version": "1.3.2",
+      "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-1.3.2.tgz",
+      "integrity": "sha512-0XNayC8lTHQ2OI8aljNCN3sSx6hsr/1+rlcDAotXJR7C1oZZHCNsfpbKwMjRA3Uqb5tF1Rae2oloTr4xpq+WjA==",
+      "dev": true,
+      "requires": {
+        "micromatch": "2.3.11",
+        "normalize-path": "2.1.1"
+      },
+      "dependencies": {
+        "arr-diff": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-2.0.0.tgz",
+          "integrity": "sha1-jzuCf5Vai9ZpaX5KQlasPOrjVs8=",
+          "dev": true,
+          "requires": {
+            "arr-flatten": "1.1.0"
+          }
+        },
+        "array-unique": {
+          "version": "0.2.1",
+          "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.2.1.tgz",
+          "integrity": "sha1-odl8yvy8JiXMcPrc6zalDFiwGlM=",
+          "dev": true
+        },
+        "braces": {
+          "version": "1.8.5",
+          "resolved": "https://registry.npmjs.org/braces/-/braces-1.8.5.tgz",
+          "integrity": "sha1-uneWLhLf+WnWt2cR6RS3N4V79qc=",
+          "dev": true,
+          "requires": {
+            "expand-range": "1.8.2",
+            "preserve": "0.2.0",
+            "repeat-element": "1.1.2"
+          }
+        },
+        "expand-brackets": {
+          "version": "0.1.5",
+          "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-0.1.5.tgz",
+          "integrity": "sha1-3wcoTjQqgHzXM6xa9yQR5YHRF3s=",
+          "dev": true,
+          "requires": {
+            "is-posix-bracket": "0.1.1"
+          }
+        },
+        "extglob": {
+          "version": "0.3.2",
+          "resolved": "https://registry.npmjs.org/extglob/-/extglob-0.3.2.tgz",
+          "integrity": "sha1-Lhj/PS9JqydlzskCPwEdqo2DSaE=",
+          "dev": true,
+          "requires": {
+            "is-extglob": "1.0.0"
+          }
+        },
+        "is-extglob": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz",
+          "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=",
+          "dev": true
+        },
+        "is-glob": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz",
+          "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=",
+          "dev": true,
+          "requires": {
+            "is-extglob": "1.0.0"
+          }
+        },
+        "micromatch": {
+          "version": "2.3.11",
+          "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-2.3.11.tgz",
+          "integrity": "sha1-hmd8l9FyCzY0MdBNDRUpO9OMFWU=",
+          "dev": true,
+          "requires": {
+            "arr-diff": "2.0.0",
+            "array-unique": "0.2.1",
+            "braces": "1.8.5",
+            "expand-brackets": "0.1.5",
+            "extglob": "0.3.2",
+            "filename-regex": "2.0.1",
+            "is-extglob": "1.0.0",
+            "is-glob": "2.0.1",
+            "kind-of": "3.2.2",
+            "normalize-path": "2.1.1",
+            "object.omit": "2.0.1",
+            "parse-glob": "3.0.4",
+            "regex-cache": "0.4.4"
+          }
+        }
+      }
+    },
+    "archy": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz",
+      "integrity": "sha1-+cjBN1fMHde8N5rHeyxipcKGjEA=",
+      "dev": true
+    },
+    "arr-diff": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz",
+      "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=",
+      "dev": true
+    },
+    "arr-flatten": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz",
+      "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==",
+      "dev": true
+    },
+    "arr-union": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz",
+      "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=",
+      "dev": true
+    },
+    "array-differ": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/array-differ/-/array-differ-1.0.0.tgz",
+      "integrity": "sha1-7/UuN1gknTO+QCuLuOVkuytdQDE=",
+      "dev": true
+    },
+    "array-each": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/array-each/-/array-each-1.0.1.tgz",
+      "integrity": "sha1-p5SvDAWrF1KEbudTofIRoFugxE8=",
+      "dev": true
+    },
+    "array-flatten": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
+      "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
+    },
+    "array-slice": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/array-slice/-/array-slice-1.1.0.tgz",
+      "integrity": "sha512-B1qMD3RBP7O8o0H2KbrXDyB0IccejMF15+87Lvlor12ONPRHP6gTjXMNkt/d3ZuOGbAe66hFmaCfECI24Ufp6w==",
+      "dev": true
+    },
+    "array-uniq": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz",
+      "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY="
+    },
+    "array-unique": {
+      "version": "0.3.2",
+      "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz",
+      "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=",
+      "dev": true
+    },
+    "arraybuffer.slice": {
+      "version": "0.0.7",
+      "resolved": "https://registry.npmjs.org/arraybuffer.slice/-/arraybuffer.slice-0.0.7.tgz",
+      "integrity": "sha512-wGUIVQXuehL5TCqQun8OW81jGzAWycqzFF8lFp+GOM5BXLYj3bKNsYC4daB7n6XjCqxQA/qgTJ+8ANR3acjrog=="
+    },
+    "asap": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz",
+      "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY="
+    },
+    "assign-symbols": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz",
+      "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=",
+      "dev": true
+    },
+    "async-each": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.1.tgz",
+      "integrity": "sha1-GdOGodntxufByF04iu28xW0zYC0=",
+      "dev": true
+    },
+    "async-limiter": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz",
+      "integrity": "sha512-jp/uFnooOiO+L211eZOoSyzpOITMXx1rBITauYykG3BRYPu8h0UcxsPNB04RR5vo4Tyz3+ay17tR6JVf9qzYWg=="
+    },
+    "atob": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/atob/-/atob-2.0.3.tgz",
+      "integrity": "sha1-GcenYEc3dEaPILLS0DNyrX1Mv10=",
+      "dev": true
+    },
+    "backo2": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz",
+      "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc="
+    },
+    "balanced-match": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
+      "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
+      "dev": true
+    },
+    "base": {
+      "version": "0.11.2",
+      "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz",
+      "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==",
+      "dev": true,
+      "requires": {
+        "cache-base": "1.0.1",
+        "class-utils": "0.3.5",
+        "component-emitter": "1.2.1",
+        "define-property": "1.0.0",
+        "isobject": "3.0.1",
+        "mixin-deep": "1.3.0",
+        "pascalcase": "0.1.1"
+      }
+    },
+    "base64-arraybuffer": {
+      "version": "0.1.5",
+      "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.5.tgz",
+      "integrity": "sha1-c5JncZI7Whl0etZmqlzUv5xunOg="
+    },
+    "base64id": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/base64id/-/base64id-1.0.0.tgz",
+      "integrity": "sha1-R2iMuZu2gE8OBtPnY7HDLlfY5rY="
+    },
+    "basic-auth": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.0.tgz",
+      "integrity": "sha1-AV2z81PgLlY3d1X5YnQuiYHnu7o=",
+      "requires": {
+        "safe-buffer": "5.1.1"
+      }
+    },
+    "beeper": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/beeper/-/beeper-1.1.1.tgz",
+      "integrity": "sha1-5tXqjF2tABMEpwsiY4RH9pyy+Ak=",
+      "dev": true
+    },
+    "better-assert": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/better-assert/-/better-assert-1.0.2.tgz",
+      "integrity": "sha1-QIZrnhueC1W0gYlDEeaPr/rrxSI=",
+      "requires": {
+        "callsite": "1.0.0"
+      }
+    },
+    "binary-extensions": {
+      "version": "1.11.0",
+      "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.11.0.tgz",
+      "integrity": "sha1-RqoXUftqL5PuXmibsQh9SxTGwgU=",
+      "dev": true
+    },
+    "blob": {
+      "version": "0.0.4",
+      "resolved": "https://registry.npmjs.org/blob/-/blob-0.0.4.tgz",
+      "integrity": "sha1-vPEwUspURj8w+fx+lbmkdjCpSSE="
+    },
+    "body-parser": {
+      "version": "1.18.2",
+      "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.2.tgz",
+      "integrity": "sha1-h2eKGdhLR9hZuDGZvVm84iKxBFQ=",
+      "requires": {
+        "bytes": "3.0.0",
+        "content-type": "1.0.4",
+        "debug": "2.6.9",
+        "depd": "1.1.1",
+        "http-errors": "1.6.2",
+        "iconv-lite": "0.4.19",
+        "on-finished": "2.3.0",
+        "qs": "6.5.1",
+        "raw-body": "2.3.2",
+        "type-is": "1.6.15"
+      }
+    },
+    "boxen": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/boxen/-/boxen-1.3.0.tgz",
+      "integrity": "sha512-TNPjfTr432qx7yOjQyaXm3dSR0MH9vXp7eT1BFSl/C51g+EFnOR9hTg1IreahGBmDNCehscshe45f+C1TBZbLw==",
+      "dev": true,
+      "requires": {
+        "ansi-align": "2.0.0",
+        "camelcase": "4.1.0",
+        "chalk": "2.3.0",
+        "cli-boxes": "1.0.0",
+        "string-width": "2.1.1",
+        "term-size": "1.2.0",
+        "widest-line": "2.0.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.0",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.0.tgz",
+          "integrity": "sha512-NnSOmMEYtVR2JVMIGTzynRkkaxtiq1xnFBcdQD/DnNCYPoEPsVJhM98BDyaoNOQIi7p4okdi3E27eN7GQbsUug==",
+          "dev": true,
+          "requires": {
+            "color-convert": "1.9.1"
+          }
+        },
+        "camelcase": {
+          "version": "4.1.0",
+          "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz",
+          "integrity": "sha1-1UVjW+HjPFQmScaRc+Xeas+uNN0=",
+          "dev": true
+        },
+        "chalk": {
+          "version": "2.3.0",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.0.tgz",
+          "integrity": "sha512-Az5zJR2CBujap2rqXGaJKaPHyJ0IrUimvYNX+ncCy8PJP4ltOGTrHUIo097ZaL2zMeKYpiCdqDvS6zdrTFok3Q==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "3.2.0",
+            "escape-string-regexp": "1.0.5",
+            "supports-color": "4.5.0"
+          }
+        },
+        "supports-color": {
+          "version": "4.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-4.5.0.tgz",
+          "integrity": "sha1-vnoN5ITexcXN34s9WRJQRJEvY1s=",
+          "dev": true,
+          "requires": {
+            "has-flag": "2.0.0"
+          }
+        }
+      }
+    },
+    "brace-expansion": {
+      "version": "1.1.8",
+      "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.8.tgz",
+      "integrity": "sha1-wHshHHyVLsH479Uad+8NHTmQopI=",
+      "dev": true,
+      "requires": {
+        "balanced-match": "1.0.0",
+        "concat-map": "0.0.1"
+      }
+    },
+    "braces": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.0.tgz",
+      "integrity": "sha512-P4O8UQRdGiMLWSizsApmXVQDBS6KCt7dSexgLKBmH5Hr1CZq7vsnscFh8oR1sP1ab1Zj0uCHCEzZeV6SfUf3rA==",
+      "dev": true,
+      "requires": {
+        "arr-flatten": "1.1.0",
+        "array-unique": "0.3.2",
+        "define-property": "1.0.0",
+        "extend-shallow": "2.0.1",
+        "fill-range": "4.0.0",
+        "isobject": "3.0.1",
+        "repeat-element": "1.1.2",
+        "snapdragon": "0.8.1",
+        "snapdragon-node": "2.1.1",
+        "split-string": "3.1.0",
+        "to-regex": "3.0.1"
+      }
+    },
+    "bson": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/bson/-/bson-1.0.4.tgz",
+      "integrity": "sha1-k8ENOeqltYQVy8QFLz5T5WKwtyw="
+    },
+    "bson-objectid": {
+      "version": "1.2.2",
+      "resolved": "https://registry.npmjs.org/bson-objectid/-/bson-objectid-1.2.2.tgz",
+      "integrity": "sha512-GyjZ1yqTDXaK5HlcDe5NXwRlURZERSF2q0p4sQCQ0Cns2aXzc/5F6mgLPBnlAWOvq9awl6NNHZ8bqvNWvZkcMg=="
+    },
+    "buffer-shims": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/buffer-shims/-/buffer-shims-1.0.0.tgz",
+      "integrity": "sha1-mXjOMXOIxkmth5MCjDR37wRKi1E="
+    },
+    "bytes": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz",
+      "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg="
+    },
+    "cache-base": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz",
+      "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==",
+      "dev": true,
+      "requires": {
+        "collection-visit": "1.0.0",
+        "component-emitter": "1.2.1",
+        "get-value": "2.0.6",
+        "has-value": "1.0.0",
+        "isobject": "3.0.1",
+        "set-value": "2.0.0",
+        "to-object-path": "0.3.0",
+        "union-value": "1.0.0",
+        "unset-value": "1.0.0"
+      }
+    },
+    "callsite": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/callsite/-/callsite-1.0.0.tgz",
+      "integrity": "sha1-KAOY5dZkvXQDi28JBRU+borxvCA="
+    },
+    "camelcase": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-1.2.1.tgz",
+      "integrity": "sha1-m7UwTS4LVmmLLHWLCKPqqdqlijk="
+    },
+    "capture-stack-trace": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.0.tgz",
+      "integrity": "sha1-Sm+gc5nCa7pH8LJJa00PtAjFVQ0=",
+      "dev": true
+    },
+    "center-align": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/center-align/-/center-align-0.1.3.tgz",
+      "integrity": "sha1-qg0yYptu6XIgBBHL1EYckHvCt60=",
+      "requires": {
+        "align-text": "0.1.4",
+        "lazy-cache": "1.0.4"
+      }
+    },
+    "chalk": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz",
+      "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=",
+      "dev": true,
+      "requires": {
+        "ansi-styles": "2.2.1",
+        "escape-string-regexp": "1.0.5",
+        "has-ansi": "2.0.0",
+        "strip-ansi": "3.0.1",
+        "supports-color": "2.0.0"
+      }
+    },
+    "character-parser": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/character-parser/-/character-parser-2.2.0.tgz",
+      "integrity": "sha1-x84o821LzZdE5f/CxfzeHHMmH8A=",
+      "requires": {
+        "is-regex": "1.0.4"
+      }
+    },
+    "chokidar": {
+      "version": "1.7.0",
+      "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-1.7.0.tgz",
+      "integrity": "sha1-eY5ol3gVHIB2tLNg5e3SjNortGg=",
+      "dev": true,
+      "requires": {
+        "anymatch": "1.3.2",
+        "async-each": "1.0.1",
+        "fsevents": "1.1.3",
+        "glob-parent": "2.0.0",
+        "inherits": "2.0.3",
+        "is-binary-path": "1.0.1",
+        "is-glob": "2.0.1",
+        "path-is-absolute": "1.0.1",
+        "readdirp": "2.1.0"
+      },
+      "dependencies": {
+        "is-extglob": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz",
+          "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=",
+          "dev": true
+        },
+        "is-glob": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz",
+          "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=",
+          "dev": true,
+          "requires": {
+            "is-extglob": "1.0.0"
+          }
+        }
+      }
+    },
+    "class-utils": {
+      "version": "0.3.5",
+      "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.5.tgz",
+      "integrity": "sha1-F+eTEDdQ+WJ7IXbqNM/RtWWQPIA=",
+      "dev": true,
+      "requires": {
+        "arr-union": "3.1.0",
+        "define-property": "0.2.5",
+        "isobject": "3.0.1",
+        "lazy-cache": "2.0.2",
+        "static-extend": "0.1.2"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          }
+        },
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        },
+        "lazy-cache": {
+          "version": "2.0.2",
+          "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-2.0.2.tgz",
+          "integrity": "sha1-uRkKT5EzVGlIQIWfio9whNiCImQ=",
+          "dev": true,
+          "requires": {
+            "set-getter": "0.1.0"
+          }
+        }
+      }
+    },
+    "clean-css": {
+      "version": "3.4.28",
+      "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-3.4.28.tgz",
+      "integrity": "sha1-vxlF6C/ICPVWlebd6uwBQA79A/8=",
+      "requires": {
+        "commander": "2.8.1",
+        "source-map": "0.4.4"
+      }
+    },
+    "cli-boxes": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-1.0.0.tgz",
+      "integrity": "sha1-T6kXw+WclKAEzWH47lCdplFocUM=",
+      "dev": true
+    },
+    "cliui": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/cliui/-/cliui-2.1.0.tgz",
+      "integrity": "sha1-S0dXYP+AJkx2LDoXGQMukcf+oNE=",
+      "requires": {
+        "center-align": "0.1.3",
+        "right-align": "0.1.3",
+        "wordwrap": "0.0.2"
+      }
+    },
+    "clone": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.3.tgz",
+      "integrity": "sha1-KY1+IjFmD0DAA8LtMUDezz9TCF8=",
+      "dev": true
+    },
+    "clone-stats": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/clone-stats/-/clone-stats-0.0.1.tgz",
+      "integrity": "sha1-uI+UqCzzi4eR1YBG6kAprYjKmdE=",
+      "dev": true
+    },
+    "collection-visit": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz",
+      "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=",
+      "dev": true,
+      "requires": {
+        "map-visit": "1.0.0",
+        "object-visit": "1.0.1"
+      }
+    },
+    "color-convert": {
+      "version": "1.9.1",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.1.tgz",
+      "integrity": "sha512-mjGanIiwQJskCC18rPR6OmrZ6fm2Lc7PeGFYwCmy5J34wC6F1PzdGL6xeMfmgicfYcNLGuVFA3WzXtIDCQSZxQ==",
+      "requires": {
+        "color-name": "1.1.3"
+      }
+    },
+    "color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU="
+    },
+    "color-support": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-support/-/color-support-1.1.3.tgz",
+      "integrity": "sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==",
+      "dev": true
+    },
+    "colors": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/colors/-/colors-1.1.2.tgz",
+      "integrity": "sha1-FopHAXVran9RoSzgyXv6KMCE7WM=",
+      "dev": true
+    },
+    "commander": {
+      "version": "2.8.1",
+      "resolved": "https://registry.npmjs.org/commander/-/commander-2.8.1.tgz",
+      "integrity": "sha1-Br42f+v9oMMwqh4qBy09yXYkJdQ=",
+      "requires": {
+        "graceful-readlink": "1.0.1"
+      }
+    },
+    "component-bind": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/component-bind/-/component-bind-1.0.0.tgz",
+      "integrity": "sha1-AMYIq33Nk4l8AAllGx06jh5zu9E="
+    },
+    "component-emitter": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz",
+      "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY="
+    },
+    "component-inherit": {
+      "version": "0.0.3",
+      "resolved": "https://registry.npmjs.org/component-inherit/-/component-inherit-0.0.3.tgz",
+      "integrity": "sha1-ZF/ErfWLcrZJ1crmUTVhnbJv8UM="
+    },
+    "concat-map": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+      "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
+      "dev": true
+    },
+    "configstore": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/configstore/-/configstore-3.1.1.tgz",
+      "integrity": "sha512-5oNkD/L++l0O6xGXxb1EWS7SivtjfGQlRyxJsYgE0Z495/L81e2h4/d3r969hoPXuFItzNOKMtsXgYG4c7dYvw==",
+      "dev": true,
+      "requires": {
+        "dot-prop": "4.2.0",
+        "graceful-fs": "4.1.11",
+        "make-dir": "1.1.0",
+        "unique-string": "1.0.0",
+        "write-file-atomic": "2.3.0",
+        "xdg-basedir": "3.0.0"
+      },
+      "dependencies": {
+        "graceful-fs": {
+          "version": "4.1.11",
+          "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz",
+          "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=",
+          "dev": true
+        }
+      }
+    },
+    "constantinople": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/constantinople/-/constantinople-3.1.0.tgz",
+      "integrity": "sha1-dWnKqKo/jVk11i4fqW+fcCzYHHk=",
+      "requires": {
+        "acorn": "3.3.0",
+        "is-expression": "2.1.0"
+      }
+    },
+    "content-disposition": {
+      "version": "0.5.2",
+      "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz",
+      "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ="
+    },
+    "content-type": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
+      "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
+    },
+    "cookie": {
+      "version": "0.3.1",
+      "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz",
+      "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s="
+    },
+    "cookie-parser": {
+      "version": "1.4.3",
+      "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.3.tgz",
+      "integrity": "sha1-D+MfoZ0AC5X0qt8fU/3CuKIDuqU=",
+      "requires": {
+        "cookie": "0.3.1",
+        "cookie-signature": "1.0.6"
+      }
+    },
+    "cookie-signature": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
+      "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
+    },
+    "copy-descriptor": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz",
+      "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=",
+      "dev": true
+    },
+    "core-util-is": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
+      "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
+    },
+    "create-error-class": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz",
+      "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=",
+      "dev": true,
+      "requires": {
+        "capture-stack-trace": "1.0.0"
+      }
+    },
+    "cross-spawn": {
+      "version": "5.1.0",
+      "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz",
+      "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=",
+      "dev": true,
+      "requires": {
+        "lru-cache": "4.1.1",
+        "shebang-command": "1.2.0",
+        "which": "1.3.0"
+      },
+      "dependencies": {
+        "lru-cache": {
+          "version": "4.1.1",
+          "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.1.tgz",
+          "integrity": "sha512-q4spe4KTfsAS1SUHLO0wz8Qiyf1+vMIAgpRYioFYDMNqKfHQbg+AVDH3i4fvpl71/P1L0dBl+fQi+P37UYf0ew==",
+          "dev": true,
+          "requires": {
+            "pseudomap": "1.0.2",
+            "yallist": "2.1.2"
+          }
+        }
+      }
+    },
+    "crypto-random-string": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz",
+      "integrity": "sha1-ojD2T1aDEOFJgAmUB5DsmVRbyn4=",
+      "dev": true
+    },
+    "dateformat": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-2.2.0.tgz",
+      "integrity": "sha1-QGXiATz5+5Ft39gu+1Bq1MZ2kGI=",
+      "dev": true
+    },
+    "debug": {
+      "version": "2.6.9",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+      "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+      "requires": {
+        "ms": "2.0.0"
+      }
+    },
+    "decamelize": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
+      "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA="
+    },
+    "decode-uri-component": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz",
+      "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=",
+      "dev": true
+    },
+    "deep-extend": {
+      "version": "0.4.2",
+      "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.4.2.tgz",
+      "integrity": "sha1-SLaZwn4zS/ifEIkr5DL25MfTSn8=",
+      "dev": true
+    },
+    "defaults": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz",
+      "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=",
+      "dev": true,
+      "requires": {
+        "clone": "1.0.3"
+      }
+    },
+    "define-property": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz",
+      "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=",
+      "dev": true,
+      "requires": {
+        "is-descriptor": "1.0.2"
+      }
+    },
+    "depd": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz",
+      "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k="
+    },
+    "deprecated": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/deprecated/-/deprecated-0.0.1.tgz",
+      "integrity": "sha1-+cmvVGSvoeepcUWKi97yqpTVuxk=",
+      "dev": true
+    },
+    "destroy": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
+      "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
+    },
+    "detect-file": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/detect-file/-/detect-file-1.0.0.tgz",
+      "integrity": "sha1-8NZtA2cqglyxtzvbP+YjEMjlUrc=",
+      "dev": true
+    },
+    "doctypes": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/doctypes/-/doctypes-1.1.0.tgz",
+      "integrity": "sha1-6oCxBqh1OHdOijpKWv4pPeSJ4Kk="
+    },
+    "dom-serializer": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.1.0.tgz",
+      "integrity": "sha1-BzxpdUbOB4DOI75KKOKT5AvDDII=",
+      "requires": {
+        "domelementtype": "1.1.3",
+        "entities": "1.1.1"
+      },
+      "dependencies": {
+        "domelementtype": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.1.3.tgz",
+          "integrity": "sha1-vSh3PiZCiBrsUVRJJCmcXNgiGFs="
+        }
+      }
+    },
+    "domelementtype": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.0.tgz",
+      "integrity": "sha1-sXrtguirWeUt2cGbF1bg/BhyBMI="
+    },
+    "domhandler": {
+      "version": "2.4.1",
+      "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.4.1.tgz",
+      "integrity": "sha1-iS5HAAqZvlW783dP/qBWHYh5wlk=",
+      "requires": {
+        "domelementtype": "1.3.0"
+      }
+    },
+    "domutils": {
+      "version": "1.6.2",
+      "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.6.2.tgz",
+      "integrity": "sha1-GVjMC0yUJuntNn+xyOhUiRsPo/8=",
+      "requires": {
+        "dom-serializer": "0.1.0",
+        "domelementtype": "1.3.0"
+      }
+    },
+    "dot-prop": {
+      "version": "4.2.0",
+      "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-4.2.0.tgz",
+      "integrity": "sha512-tUMXrxlExSW6U2EXiiKGSBVdYgtV8qlHL+C10TsW4PURY/ic+eaysnSkwB4kA/mBlCyy/IKDJ+Lc3wbWeaXtuQ==",
+      "dev": true,
+      "requires": {
+        "is-obj": "1.0.1"
+      }
+    },
+    "duplexer": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.1.tgz",
+      "integrity": "sha1-rOb/gIwc5mtX0ev5eXessCM0z8E=",
+      "dev": true
+    },
+    "duplexer2": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.0.2.tgz",
+      "integrity": "sha1-xhTc9n4vsUmVqRcR5aYX6KYKMds=",
+      "dev": true,
+      "requires": {
+        "readable-stream": "1.1.14"
+      },
+      "dependencies": {
+        "isarray": {
+          "version": "0.0.1",
+          "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz",
+          "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=",
+          "dev": true
+        },
+        "readable-stream": {
+          "version": "1.1.14",
+          "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz",
+          "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=",
+          "dev": true,
+          "requires": {
+            "core-util-is": "1.0.2",
+            "inherits": "2.0.3",
+            "isarray": "0.0.1",
+            "string_decoder": "0.10.31"
+          }
+        },
+        "string_decoder": {
+          "version": "0.10.31",
+          "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz",
+          "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=",
+          "dev": true
+        }
+      }
+    },
+    "duplexer3": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz",
+      "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=",
+      "dev": true
+    },
+    "each-series": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/each-series/-/each-series-1.0.0.tgz",
+      "integrity": "sha1-+Ibmxm39sl7x/nNWQUbuXLR4r8s="
+    },
+    "ee-first": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
+      "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
+    },
+    "encodeurl": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.1.tgz",
+      "integrity": "sha1-eePVhlU0aQn+bw9Fpd5oEDspTSA="
+    },
+    "end-of-stream": {
+      "version": "0.1.5",
+      "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-0.1.5.tgz",
+      "integrity": "sha1-jhdyBsPICDfYVjLouTWd/osvbq8=",
+      "dev": true,
+      "requires": {
+        "once": "1.3.3"
+      },
+      "dependencies": {
+        "once": {
+          "version": "1.3.3",
+          "resolved": "https://registry.npmjs.org/once/-/once-1.3.3.tgz",
+          "integrity": "sha1-suJhVXzkwxTsgwTz+oJmPkKXyiA=",
+          "dev": true,
+          "requires": {
+            "wrappy": "1.0.2"
+          }
+        }
+      }
+    },
+    "engine.io": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-3.1.4.tgz",
+      "integrity": "sha1-PQIRtwpVLOhB/8fahiezAamkFi4=",
+      "requires": {
+        "accepts": "1.3.3",
+        "base64id": "1.0.0",
+        "cookie": "0.3.1",
+        "debug": "2.6.9",
+        "engine.io-parser": "2.1.2",
+        "uws": "0.14.5",
+        "ws": "3.3.3"
+      },
+      "dependencies": {
+        "accepts": {
+          "version": "1.3.3",
+          "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.3.tgz",
+          "integrity": "sha1-w8p0NJOGSMPg2cHjKN1otiLChMo=",
+          "requires": {
+            "mime-types": "2.1.17",
+            "negotiator": "0.6.1"
+          }
+        }
+      }
+    },
+    "engine.io-client": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-3.1.4.tgz",
+      "integrity": "sha1-T88TcLRxY70s6b4nM5ckMDUNTqE=",
+      "requires": {
+        "component-emitter": "1.2.1",
+        "component-inherit": "0.0.3",
+        "debug": "2.6.9",
+        "engine.io-parser": "2.1.2",
+        "has-cors": "1.1.0",
+        "indexof": "0.0.1",
+        "parseqs": "0.0.5",
+        "parseuri": "0.0.5",
+        "ws": "3.3.3",
+        "xmlhttprequest-ssl": "1.5.4",
+        "yeast": "0.1.2"
+      }
+    },
+    "engine.io-parser": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-2.1.2.tgz",
+      "integrity": "sha512-dInLFzr80RijZ1rGpx1+56/uFoH7/7InhH3kZt+Ms6hT8tNx3NGW/WNSA/f8As1WkOfkuyb3tnRyuXGxusclMw==",
+      "requires": {
+        "after": "0.8.2",
+        "arraybuffer.slice": "0.0.7",
+        "base64-arraybuffer": "0.1.5",
+        "blob": "0.0.4",
+        "has-binary2": "1.0.2"
+      }
+    },
+    "entities": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/entities/-/entities-1.1.1.tgz",
+      "integrity": "sha1-blwtClYhtdra7O+AuQ7ftc13cvA="
+    },
+    "es6-promise": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-3.2.1.tgz",
+      "integrity": "sha1-7FYjOGgDKQkgcXDDlEjiREndH8Q="
+    },
+    "escape-html": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
+      "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
+    },
+    "escape-string-regexp": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
+      "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ="
+    },
+    "etag": {
+      "version": "1.8.1",
+      "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
+      "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
+    },
+    "event-stream": {
+      "version": "3.3.4",
+      "resolved": "https://registry.npmjs.org/event-stream/-/event-stream-3.3.4.tgz",
+      "integrity": "sha1-SrTJoPWlTbkzi0w02Gv86PSzVXE=",
+      "dev": true,
+      "requires": {
+        "duplexer": "0.1.1",
+        "from": "0.1.7",
+        "map-stream": "0.1.0",
+        "pause-stream": "0.0.11",
+        "split": "0.3.3",
+        "stream-combiner": "0.0.4",
+        "through": "2.3.8"
+      }
+    },
+    "execa": {
+      "version": "0.7.0",
+      "resolved": "https://registry.npmjs.org/execa/-/execa-0.7.0.tgz",
+      "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=",
+      "dev": true,
+      "requires": {
+        "cross-spawn": "5.1.0",
+        "get-stream": "3.0.0",
+        "is-stream": "1.1.0",
+        "npm-run-path": "2.0.2",
+        "p-finally": "1.0.0",
+        "signal-exit": "3.0.2",
+        "strip-eof": "1.0.0"
+      }
+    },
+    "expand-brackets": {
+      "version": "2.1.4",
+      "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz",
+      "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=",
+      "dev": true,
+      "requires": {
+        "debug": "2.6.9",
+        "define-property": "0.2.5",
+        "extend-shallow": "2.0.1",
+        "posix-character-classes": "0.1.1",
+        "regex-not": "1.0.0",
+        "snapdragon": "0.8.1",
+        "to-regex": "3.0.1"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          }
+        },
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        }
+      }
+    },
+    "expand-range": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/expand-range/-/expand-range-1.8.2.tgz",
+      "integrity": "sha1-opnv/TNf4nIeuujiV+x5ZE/IUzc=",
+      "dev": true,
+      "requires": {
+        "fill-range": "2.2.3"
+      },
+      "dependencies": {
+        "fill-range": {
+          "version": "2.2.3",
+          "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-2.2.3.tgz",
+          "integrity": "sha1-ULd9/X5Gm8dJJHCWNpn+eoSFpyM=",
+          "dev": true,
+          "requires": {
+            "is-number": "2.1.0",
+            "isobject": "2.1.0",
+            "randomatic": "1.1.7",
+            "repeat-element": "1.1.2",
+            "repeat-string": "1.6.1"
+          }
+        },
+        "is-number": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/is-number/-/is-number-2.1.0.tgz",
+          "integrity": "sha1-Afy7s5NGOlSPL0ZszhbezknbkI8=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          }
+        },
+        "isobject": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz",
+          "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=",
+          "dev": true,
+          "requires": {
+            "isarray": "1.0.0"
+          }
+        }
+      }
+    },
+    "expand-tilde": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/expand-tilde/-/expand-tilde-2.0.2.tgz",
+      "integrity": "sha1-l+gBqgUt8CRU3kawK/YhZCzchQI=",
+      "dev": true,
+      "requires": {
+        "homedir-polyfill": "1.0.1"
+      }
+    },
+    "express": {
+      "version": "4.16.2",
+      "resolved": "https://registry.npmjs.org/express/-/express-4.16.2.tgz",
+      "integrity": "sha1-41xt/i1kt9ygpc1PIXgb4ymeB2w=",
+      "requires": {
+        "accepts": "1.3.4",
+        "array-flatten": "1.1.1",
+        "body-parser": "1.18.2",
+        "content-disposition": "0.5.2",
+        "content-type": "1.0.4",
+        "cookie": "0.3.1",
+        "cookie-signature": "1.0.6",
+        "debug": "2.6.9",
+        "depd": "1.1.1",
+        "encodeurl": "1.0.1",
+        "escape-html": "1.0.3",
+        "etag": "1.8.1",
+        "finalhandler": "1.1.0",
+        "fresh": "0.5.2",
+        "merge-descriptors": "1.0.1",
+        "methods": "1.1.2",
+        "on-finished": "2.3.0",
+        "parseurl": "1.3.2",
+        "path-to-regexp": "0.1.7",
+        "proxy-addr": "2.0.2",
+        "qs": "6.5.1",
+        "range-parser": "1.2.0",
+        "safe-buffer": "5.1.1",
+        "send": "0.16.1",
+        "serve-static": "1.13.1",
+        "setprototypeof": "1.1.0",
+        "statuses": "1.3.1",
+        "type-is": "1.6.15",
+        "utils-merge": "1.0.1",
+        "vary": "1.1.2"
+      }
+    },
+    "extend": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz",
+      "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=",
+      "dev": true
+    },
+    "extend-shallow": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz",
+      "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=",
+      "dev": true,
+      "requires": {
+        "is-extendable": "0.1.1"
+      }
+    },
+    "extglob": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.3.tgz",
+      "integrity": "sha512-AyptZexgu7qppEPq59DtN/XJGZDrLcVxSHai+4hdgMMS9EpF4GBvygcWWApno8lL9qSjVpYt7Raao28qzJX1ww==",
+      "dev": true,
+      "requires": {
+        "array-unique": "0.3.2",
+        "define-property": "1.0.0",
+        "expand-brackets": "2.1.4",
+        "extend-shallow": "2.0.1",
+        "fragment-cache": "0.2.1",
+        "regex-not": "1.0.0",
+        "snapdragon": "0.8.1",
+        "to-regex": "3.0.1"
+      }
+    },
+    "fancy-log": {
+      "version": "1.3.2",
+      "resolved": "https://registry.npmjs.org/fancy-log/-/fancy-log-1.3.2.tgz",
+      "integrity": "sha1-9BEl49hPLn2JpD0G2VjI94vha+E=",
+      "dev": true,
+      "requires": {
+        "ansi-gray": "0.1.1",
+        "color-support": "1.1.3",
+        "time-stamp": "1.1.0"
+      }
+    },
+    "filename-regex": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/filename-regex/-/filename-regex-2.0.1.tgz",
+      "integrity": "sha1-wcS5vuPglyXdsQa3XB4wH+LxiyY=",
+      "dev": true
+    },
+    "fill-range": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz",
+      "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=",
+      "dev": true,
+      "requires": {
+        "extend-shallow": "2.0.1",
+        "is-number": "3.0.0",
+        "repeat-string": "1.6.1",
+        "to-regex-range": "2.1.1"
+      }
+    },
+    "finalhandler": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.0.tgz",
+      "integrity": "sha1-zgtoVbRYU+eRsvzGgARtiCU91/U=",
+      "requires": {
+        "debug": "2.6.9",
+        "encodeurl": "1.0.1",
+        "escape-html": "1.0.3",
+        "on-finished": "2.3.0",
+        "parseurl": "1.3.2",
+        "statuses": "1.3.1",
+        "unpipe": "1.0.0"
+      }
+    },
+    "find-index": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/find-index/-/find-index-0.1.1.tgz",
+      "integrity": "sha1-Z101iyyjiS15Whq0cjL4tuLg3eQ=",
+      "dev": true
+    },
+    "findup-sync": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-2.0.0.tgz",
+      "integrity": "sha1-kyaxSIwi0aYIhlCoaQGy2akKLLw=",
+      "dev": true,
+      "requires": {
+        "detect-file": "1.0.0",
+        "is-glob": "3.1.0",
+        "micromatch": "3.1.4",
+        "resolve-dir": "1.0.1"
+      }
+    },
+    "fined": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/fined/-/fined-1.1.0.tgz",
+      "integrity": "sha1-s33IRLdqL15wgeiE98CuNE8VNHY=",
+      "dev": true,
+      "requires": {
+        "expand-tilde": "2.0.2",
+        "is-plain-object": "2.0.4",
+        "object.defaults": "1.1.0",
+        "object.pick": "1.3.0",
+        "parse-filepath": "1.0.2"
+      }
+    },
+    "first-chunk-stream": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/first-chunk-stream/-/first-chunk-stream-1.0.0.tgz",
+      "integrity": "sha1-Wb+1DNkF9g18OUzT2ayqtOatk04=",
+      "dev": true
+    },
+    "flagged-respawn": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/flagged-respawn/-/flagged-respawn-1.0.0.tgz",
+      "integrity": "sha1-Tnmumy6zi/hrO7Vr8+ClaqX8q9c=",
+      "dev": true
+    },
+    "for-in": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz",
+      "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=",
+      "dev": true
+    },
+    "for-own": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/for-own/-/for-own-1.0.0.tgz",
+      "integrity": "sha1-xjMy9BXO3EsE2/5wz4NklMU8tEs=",
+      "dev": true,
+      "requires": {
+        "for-in": "1.0.2"
+      }
+    },
+    "forwarded": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
+      "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
+    },
+    "fragment-cache": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz",
+      "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=",
+      "dev": true,
+      "requires": {
+        "map-cache": "0.2.2"
+      }
+    },
+    "fresh": {
+      "version": "0.5.2",
+      "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
+      "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
+    },
+    "from": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/from/-/from-0.1.7.tgz",
+      "integrity": "sha1-g8YK/Fi5xWmXAH7Rp2izqzA6RP4=",
+      "dev": true
+    },
+    "fsevents": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.1.3.tgz",
+      "integrity": "sha512-WIr7iDkdmdbxu/Gh6eKEZJL6KPE74/5MEsf2whTOFNxbIoIixogroLdKYqB6FDav4Wavh/lZdzzd3b2KxIXC5Q==",
+      "dev": true,
+      "optional": true,
+      "requires": {
+        "nan": "2.8.0",
+        "node-pre-gyp": "0.6.39"
+      },
+      "dependencies": {
+        "abbrev": {
+          "version": "1.1.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "ajv": {
+          "version": "4.11.8",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "co": "4.6.0",
+            "json-stable-stringify": "1.0.1"
+          }
+        },
+        "ansi-regex": {
+          "version": "2.1.1",
+          "bundled": true,
+          "dev": true
+        },
+        "aproba": {
+          "version": "1.1.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "are-we-there-yet": {
+          "version": "1.1.4",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "delegates": "1.0.0",
+            "readable-stream": "2.2.9"
+          }
+        },
+        "asn1": {
+          "version": "0.2.3",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "assert-plus": {
+          "version": "0.2.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "asynckit": {
+          "version": "0.4.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "aws-sign2": {
+          "version": "0.6.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "aws4": {
+          "version": "1.6.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "balanced-match": {
+          "version": "0.4.2",
+          "bundled": true,
+          "dev": true
+        },
+        "bcrypt-pbkdf": {
+          "version": "1.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "tweetnacl": "0.14.5"
+          }
+        },
+        "block-stream": {
+          "version": "0.0.9",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "inherits": "2.0.3"
+          }
+        },
+        "boom": {
+          "version": "2.10.1",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "hoek": "2.16.3"
+          }
+        },
+        "brace-expansion": {
+          "version": "1.1.7",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "balanced-match": "0.4.2",
+            "concat-map": "0.0.1"
+          }
+        },
+        "buffer-shims": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true
+        },
+        "caseless": {
+          "version": "0.12.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "co": {
+          "version": "4.6.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "code-point-at": {
+          "version": "1.1.0",
+          "bundled": true,
+          "dev": true
+        },
+        "combined-stream": {
+          "version": "1.0.5",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "delayed-stream": "1.0.0"
+          }
+        },
+        "concat-map": {
+          "version": "0.0.1",
+          "bundled": true,
+          "dev": true
+        },
+        "console-control-strings": {
+          "version": "1.1.0",
+          "bundled": true,
+          "dev": true
+        },
+        "core-util-is": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true
+        },
+        "cryptiles": {
+          "version": "2.0.5",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "boom": "2.10.1"
+          }
+        },
+        "dashdash": {
+          "version": "1.14.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "assert-plus": "1.0.0"
+          },
+          "dependencies": {
+            "assert-plus": {
+              "version": "1.0.0",
+              "bundled": true,
+              "dev": true,
+              "optional": true
+            }
+          }
+        },
+        "debug": {
+          "version": "2.6.8",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "ms": "2.0.0"
+          }
+        },
+        "deep-extend": {
+          "version": "0.4.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "delayed-stream": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true
+        },
+        "delegates": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "detect-libc": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "ecc-jsbn": {
+          "version": "0.1.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "jsbn": "0.1.1"
+          }
+        },
+        "extend": {
+          "version": "3.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "extsprintf": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true
+        },
+        "forever-agent": {
+          "version": "0.6.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "form-data": {
+          "version": "2.1.4",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "asynckit": "0.4.0",
+            "combined-stream": "1.0.5",
+            "mime-types": "2.1.15"
+          }
+        },
+        "fs.realpath": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true
+        },
+        "fstream": {
+          "version": "1.0.11",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "graceful-fs": "4.1.11",
+            "inherits": "2.0.3",
+            "mkdirp": "0.5.1",
+            "rimraf": "2.6.1"
+          }
+        },
+        "fstream-ignore": {
+          "version": "1.0.5",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "fstream": "1.0.11",
+            "inherits": "2.0.3",
+            "minimatch": "3.0.4"
+          }
+        },
+        "gauge": {
+          "version": "2.7.4",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "aproba": "1.1.1",
+            "console-control-strings": "1.1.0",
+            "has-unicode": "2.0.1",
+            "object-assign": "4.1.1",
+            "signal-exit": "3.0.2",
+            "string-width": "1.0.2",
+            "strip-ansi": "3.0.1",
+            "wide-align": "1.1.2"
+          }
+        },
+        "getpass": {
+          "version": "0.1.7",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "assert-plus": "1.0.0"
+          },
+          "dependencies": {
+            "assert-plus": {
+              "version": "1.0.0",
+              "bundled": true,
+              "dev": true,
+              "optional": true
+            }
+          }
+        },
+        "glob": {
+          "version": "7.1.2",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "fs.realpath": "1.0.0",
+            "inflight": "1.0.6",
+            "inherits": "2.0.3",
+            "minimatch": "3.0.4",
+            "once": "1.4.0",
+            "path-is-absolute": "1.0.1"
+          }
+        },
+        "graceful-fs": {
+          "version": "4.1.11",
+          "bundled": true,
+          "dev": true
+        },
+        "har-schema": {
+          "version": "1.0.5",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "har-validator": {
+          "version": "4.2.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "ajv": "4.11.8",
+            "har-schema": "1.0.5"
+          }
+        },
+        "has-unicode": {
+          "version": "2.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "hawk": {
+          "version": "3.1.3",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "boom": "2.10.1",
+            "cryptiles": "2.0.5",
+            "hoek": "2.16.3",
+            "sntp": "1.0.9"
+          }
+        },
+        "hoek": {
+          "version": "2.16.3",
+          "bundled": true,
+          "dev": true
+        },
+        "http-signature": {
+          "version": "1.1.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "assert-plus": "0.2.0",
+            "jsprim": "1.4.0",
+            "sshpk": "1.13.0"
+          }
+        },
+        "inflight": {
+          "version": "1.0.6",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "once": "1.4.0",
+            "wrappy": "1.0.2"
+          }
+        },
+        "inherits": {
+          "version": "2.0.3",
+          "bundled": true,
+          "dev": true
+        },
+        "ini": {
+          "version": "1.3.4",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "is-fullwidth-code-point": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "number-is-nan": "1.0.1"
+          }
+        },
+        "is-typedarray": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "isarray": {
+          "version": "1.0.0",
+          "bundled": true,
+          "dev": true
+        },
+        "isstream": {
+          "version": "0.1.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "jodid25519": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "jsbn": "0.1.1"
+          }
+        },
+        "jsbn": {
+          "version": "0.1.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "json-schema": {
+          "version": "0.2.3",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "json-stable-stringify": {
+          "version": "1.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "jsonify": "0.0.0"
+          }
+        },
+        "json-stringify-safe": {
+          "version": "5.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "jsonify": {
+          "version": "0.0.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "jsprim": {
+          "version": "1.4.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "assert-plus": "1.0.0",
+            "extsprintf": "1.0.2",
+            "json-schema": "0.2.3",
+            "verror": "1.3.6"
+          },
+          "dependencies": {
+            "assert-plus": {
+              "version": "1.0.0",
+              "bundled": true,
+              "dev": true,
+              "optional": true
+            }
+          }
+        },
+        "mime-db": {
+          "version": "1.27.0",
+          "bundled": true,
+          "dev": true
+        },
+        "mime-types": {
+          "version": "2.1.15",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "mime-db": "1.27.0"
+          }
+        },
+        "minimatch": {
+          "version": "3.0.4",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "brace-expansion": "1.1.7"
+          }
+        },
+        "minimist": {
+          "version": "0.0.8",
+          "bundled": true,
+          "dev": true
+        },
+        "mkdirp": {
+          "version": "0.5.1",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "minimist": "0.0.8"
+          }
+        },
+        "ms": {
+          "version": "2.0.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "node-pre-gyp": {
+          "version": "0.6.39",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "detect-libc": "1.0.2",
+            "hawk": "3.1.3",
+            "mkdirp": "0.5.1",
+            "nopt": "4.0.1",
+            "npmlog": "4.1.0",
+            "rc": "1.2.1",
+            "request": "2.81.0",
+            "rimraf": "2.6.1",
+            "semver": "5.3.0",
+            "tar": "2.2.1",
+            "tar-pack": "3.4.0"
+          }
+        },
+        "nopt": {
+          "version": "4.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "abbrev": "1.1.0",
+            "osenv": "0.1.4"
+          }
+        },
+        "npmlog": {
+          "version": "4.1.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "are-we-there-yet": "1.1.4",
+            "console-control-strings": "1.1.0",
+            "gauge": "2.7.4",
+            "set-blocking": "2.0.0"
+          }
+        },
+        "number-is-nan": {
+          "version": "1.0.1",
+          "bundled": true,
+          "dev": true
+        },
+        "oauth-sign": {
+          "version": "0.8.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "object-assign": {
+          "version": "4.1.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "once": {
+          "version": "1.4.0",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "wrappy": "1.0.2"
+          }
+        },
+        "os-homedir": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "os-tmpdir": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "osenv": {
+          "version": "0.1.4",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "os-homedir": "1.0.2",
+            "os-tmpdir": "1.0.2"
+          }
+        },
+        "path-is-absolute": {
+          "version": "1.0.1",
+          "bundled": true,
+          "dev": true
+        },
+        "performance-now": {
+          "version": "0.2.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "process-nextick-args": {
+          "version": "1.0.7",
+          "bundled": true,
+          "dev": true
+        },
+        "punycode": {
+          "version": "1.4.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "qs": {
+          "version": "6.4.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "rc": {
+          "version": "1.2.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "deep-extend": "0.4.2",
+            "ini": "1.3.4",
+            "minimist": "1.2.0",
+            "strip-json-comments": "2.0.1"
+          },
+          "dependencies": {
+            "minimist": {
+              "version": "1.2.0",
+              "bundled": true,
+              "dev": true,
+              "optional": true
+            }
+          }
+        },
+        "readable-stream": {
+          "version": "2.2.9",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "buffer-shims": "1.0.0",
+            "core-util-is": "1.0.2",
+            "inherits": "2.0.3",
+            "isarray": "1.0.0",
+            "process-nextick-args": "1.0.7",
+            "string_decoder": "1.0.1",
+            "util-deprecate": "1.0.2"
+          }
+        },
+        "request": {
+          "version": "2.81.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "aws-sign2": "0.6.0",
+            "aws4": "1.6.0",
+            "caseless": "0.12.0",
+            "combined-stream": "1.0.5",
+            "extend": "3.0.1",
+            "forever-agent": "0.6.1",
+            "form-data": "2.1.4",
+            "har-validator": "4.2.1",
+            "hawk": "3.1.3",
+            "http-signature": "1.1.1",
+            "is-typedarray": "1.0.0",
+            "isstream": "0.1.2",
+            "json-stringify-safe": "5.0.1",
+            "mime-types": "2.1.15",
+            "oauth-sign": "0.8.2",
+            "performance-now": "0.2.0",
+            "qs": "6.4.0",
+            "safe-buffer": "5.0.1",
+            "stringstream": "0.0.5",
+            "tough-cookie": "2.3.2",
+            "tunnel-agent": "0.6.0",
+            "uuid": "3.0.1"
+          }
+        },
+        "rimraf": {
+          "version": "2.6.1",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "glob": "7.1.2"
+          }
+        },
+        "safe-buffer": {
+          "version": "5.0.1",
+          "bundled": true,
+          "dev": true
+        },
+        "semver": {
+          "version": "5.3.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "set-blocking": {
+          "version": "2.0.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "signal-exit": {
+          "version": "3.0.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "sntp": {
+          "version": "1.0.9",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "hoek": "2.16.3"
+          }
+        },
+        "sshpk": {
+          "version": "1.13.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "asn1": "0.2.3",
+            "assert-plus": "1.0.0",
+            "bcrypt-pbkdf": "1.0.1",
+            "dashdash": "1.14.1",
+            "ecc-jsbn": "0.1.1",
+            "getpass": "0.1.7",
+            "jodid25519": "1.0.2",
+            "jsbn": "0.1.1",
+            "tweetnacl": "0.14.5"
+          },
+          "dependencies": {
+            "assert-plus": {
+              "version": "1.0.0",
+              "bundled": true,
+              "dev": true,
+              "optional": true
+            }
+          }
+        },
+        "string-width": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "code-point-at": "1.1.0",
+            "is-fullwidth-code-point": "1.0.0",
+            "strip-ansi": "3.0.1"
+          }
+        },
+        "string_decoder": {
+          "version": "1.0.1",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "safe-buffer": "5.0.1"
+          }
+        },
+        "stringstream": {
+          "version": "0.0.5",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "strip-ansi": {
+          "version": "3.0.1",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "ansi-regex": "2.1.1"
+          }
+        },
+        "strip-json-comments": {
+          "version": "2.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "tar": {
+          "version": "2.2.1",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "block-stream": "0.0.9",
+            "fstream": "1.0.11",
+            "inherits": "2.0.3"
+          }
+        },
+        "tar-pack": {
+          "version": "3.4.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "debug": "2.6.8",
+            "fstream": "1.0.11",
+            "fstream-ignore": "1.0.5",
+            "once": "1.4.0",
+            "readable-stream": "2.2.9",
+            "rimraf": "2.6.1",
+            "tar": "2.2.1",
+            "uid-number": "0.0.6"
+          }
+        },
+        "tough-cookie": {
+          "version": "2.3.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "punycode": "1.4.1"
+          }
+        },
+        "tunnel-agent": {
+          "version": "0.6.0",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "safe-buffer": "5.0.1"
+          }
+        },
+        "tweetnacl": {
+          "version": "0.14.5",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "uid-number": {
+          "version": "0.0.6",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "util-deprecate": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true
+        },
+        "uuid": {
+          "version": "3.0.1",
+          "bundled": true,
+          "dev": true,
+          "optional": true
+        },
+        "verror": {
+          "version": "1.3.6",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "extsprintf": "1.0.2"
+          }
+        },
+        "wide-align": {
+          "version": "1.1.2",
+          "bundled": true,
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "string-width": "1.0.2"
+          }
+        },
+        "wrappy": {
+          "version": "1.0.2",
+          "bundled": true,
+          "dev": true
+        }
+      }
+    },
+    "function-bind": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
+      "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A=="
+    },
+    "gaze": {
+      "version": "0.5.2",
+      "resolved": "https://registry.npmjs.org/gaze/-/gaze-0.5.2.tgz",
+      "integrity": "sha1-QLcJU30k0dRXZ9takIaJ3+aaxE8=",
+      "dev": true,
+      "requires": {
+        "globule": "0.1.0"
+      }
+    },
+    "get-stream": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz",
+      "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=",
+      "dev": true
+    },
+    "get-value": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz",
+      "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=",
+      "dev": true
+    },
+    "glob": {
+      "version": "4.5.3",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-4.5.3.tgz",
+      "integrity": "sha1-xstz0yJsHv7wTePFbQEvAzd+4V8=",
+      "dev": true,
+      "requires": {
+        "inflight": "1.0.6",
+        "inherits": "2.0.3",
+        "minimatch": "2.0.10",
+        "once": "1.4.0"
+      }
+    },
+    "glob-base": {
+      "version": "0.3.0",
+      "resolved": "https://registry.npmjs.org/glob-base/-/glob-base-0.3.0.tgz",
+      "integrity": "sha1-27Fk9iIbHAscz4Kuoyi0l98Oo8Q=",
+      "dev": true,
+      "requires": {
+        "glob-parent": "2.0.0",
+        "is-glob": "2.0.1"
+      },
+      "dependencies": {
+        "is-extglob": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz",
+          "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=",
+          "dev": true
+        },
+        "is-glob": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz",
+          "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=",
+          "dev": true,
+          "requires": {
+            "is-extglob": "1.0.0"
+          }
+        }
+      }
+    },
+    "glob-parent": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-2.0.0.tgz",
+      "integrity": "sha1-gTg9ctsFT8zPUzbaqQLxgvbtuyg=",
+      "dev": true,
+      "requires": {
+        "is-glob": "2.0.1"
+      },
+      "dependencies": {
+        "is-extglob": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz",
+          "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=",
+          "dev": true
+        },
+        "is-glob": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz",
+          "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=",
+          "dev": true,
+          "requires": {
+            "is-extglob": "1.0.0"
+          }
+        }
+      }
+    },
+    "glob-stream": {
+      "version": "3.1.18",
+      "resolved": "https://registry.npmjs.org/glob-stream/-/glob-stream-3.1.18.tgz",
+      "integrity": "sha1-kXCl8St5Awb9/lmPMT+PeVT9FDs=",
+      "dev": true,
+      "requires": {
+        "glob": "4.5.3",
+        "glob2base": "0.0.12",
+        "minimatch": "2.0.10",
+        "ordered-read-streams": "0.1.0",
+        "through2": "0.6.5",
+        "unique-stream": "1.0.0"
+      },
+      "dependencies": {
+        "isarray": {
+          "version": "0.0.1",
+          "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz",
+          "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=",
+          "dev": true
+        },
+        "readable-stream": {
+          "version": "1.0.34",
+          "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz",
+          "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=",
+          "dev": true,
+          "requires": {
+            "core-util-is": "1.0.2",
+            "inherits": "2.0.3",
+            "isarray": "0.0.1",
+            "string_decoder": "0.10.31"
+          }
+        },
+        "string_decoder": {
+          "version": "0.10.31",
+          "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz",
+          "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=",
+          "dev": true
+        },
+        "through2": {
+          "version": "0.6.5",
+          "resolved": "https://registry.npmjs.org/through2/-/through2-0.6.5.tgz",
+          "integrity": "sha1-QaucZ7KdVyCQcUEOHXp6lozTrUg=",
+          "dev": true,
+          "requires": {
+            "readable-stream": "1.0.34",
+            "xtend": "4.0.1"
+          }
+        }
+      }
+    },
+    "glob-watcher": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/glob-watcher/-/glob-watcher-0.0.6.tgz",
+      "integrity": "sha1-uVtKjfdLOcgymLDAXJeLTZo7cQs=",
+      "dev": true,
+      "requires": {
+        "gaze": "0.5.2"
+      }
+    },
+    "glob2base": {
+      "version": "0.0.12",
+      "resolved": "https://registry.npmjs.org/glob2base/-/glob2base-0.0.12.tgz",
+      "integrity": "sha1-nUGbPijxLoOjYhZKJ3BVkiycDVY=",
+      "dev": true,
+      "requires": {
+        "find-index": "0.1.1"
+      }
+    },
+    "global-dirs": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-0.1.1.tgz",
+      "integrity": "sha1-sxnA3UYH81PzvpzKTHL8FIxJ9EU=",
+      "dev": true,
+      "requires": {
+        "ini": "1.3.5"
+      }
+    },
+    "global-modules": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-1.0.0.tgz",
+      "integrity": "sha512-sKzpEkf11GpOFuw0Zzjzmt4B4UZwjOcG757PPvrfhxcLFbq0wpsgpOqxpxtxFiCG4DtG93M6XRVbF2oGdev7bg==",
+      "dev": true,
+      "requires": {
+        "global-prefix": "1.0.2",
+        "is-windows": "1.0.1",
+        "resolve-dir": "1.0.1"
+      }
+    },
+    "global-prefix": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-1.0.2.tgz",
+      "integrity": "sha1-2/dDxsFJklk8ZVVoy2btMsASLr4=",
+      "dev": true,
+      "requires": {
+        "expand-tilde": "2.0.2",
+        "homedir-polyfill": "1.0.1",
+        "ini": "1.3.5",
+        "is-windows": "1.0.1",
+        "which": "1.3.0"
+      }
+    },
+    "globule": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/globule/-/globule-0.1.0.tgz",
+      "integrity": "sha1-2cjt3h2nnRJaFRt5UzuXhnY0auU=",
+      "dev": true,
+      "requires": {
+        "glob": "3.1.21",
+        "lodash": "1.0.2",
+        "minimatch": "0.2.14"
+      },
+      "dependencies": {
+        "glob": {
+          "version": "3.1.21",
+          "resolved": "https://registry.npmjs.org/glob/-/glob-3.1.21.tgz",
+          "integrity": "sha1-0p4KBV3qUTj00H7UDomC6DwgZs0=",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "1.2.3",
+            "inherits": "1.0.2",
+            "minimatch": "0.2.14"
+          }
+        },
+        "graceful-fs": {
+          "version": "1.2.3",
+          "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-1.2.3.tgz",
+          "integrity": "sha1-FaSAaldUfLLS2/J/QuiajDRRs2Q=",
+          "dev": true
+        },
+        "inherits": {
+          "version": "1.0.2",
+          "resolved": "https://registry.npmjs.org/inherits/-/inherits-1.0.2.tgz",
+          "integrity": "sha1-ykMJ2t7mtUzAuNJH6NfHoJdb3Js=",
+          "dev": true
+        },
+        "minimatch": {
+          "version": "0.2.14",
+          "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-0.2.14.tgz",
+          "integrity": "sha1-x054BXT2PG+aCQ6Q775u9TpqdWo=",
+          "dev": true,
+          "requires": {
+            "lru-cache": "2.7.3",
+            "sigmund": "1.0.1"
+          }
+        }
+      }
+    },
+    "glogg": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/glogg/-/glogg-1.0.0.tgz",
+      "integrity": "sha1-f+DxmfV6yQbPUS/urY+Q7kooT8U=",
+      "dev": true,
+      "requires": {
+        "sparkles": "1.0.0"
+      }
+    },
+    "got": {
+      "version": "6.7.1",
+      "resolved": "https://registry.npmjs.org/got/-/got-6.7.1.tgz",
+      "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=",
+      "dev": true,
+      "requires": {
+        "create-error-class": "3.0.2",
+        "duplexer3": "0.1.4",
+        "get-stream": "3.0.0",
+        "is-redirect": "1.0.0",
+        "is-retry-allowed": "1.1.0",
+        "is-stream": "1.1.0",
+        "lowercase-keys": "1.0.0",
+        "safe-buffer": "5.1.1",
+        "timed-out": "4.0.1",
+        "unzip-response": "2.0.1",
+        "url-parse-lax": "1.0.0"
+      }
+    },
+    "graceful-fs": {
+      "version": "3.0.11",
+      "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-3.0.11.tgz",
+      "integrity": "sha1-dhPHeKGv6mLyXGMKCG1/Osu92Bg=",
+      "dev": true,
+      "requires": {
+        "natives": "1.1.1"
+      }
+    },
+    "graceful-readlink": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/graceful-readlink/-/graceful-readlink-1.0.1.tgz",
+      "integrity": "sha1-TK+tdrxi8C+gObL5Tpo906ORpyU="
+    },
+    "gulp": {
+      "version": "3.9.1",
+      "resolved": "https://registry.npmjs.org/gulp/-/gulp-3.9.1.tgz",
+      "integrity": "sha1-VxzkWSjdQK9lFPxAEYZgFsE4RbQ=",
+      "dev": true,
+      "requires": {
+        "archy": "1.0.0",
+        "chalk": "1.1.3",
+        "deprecated": "0.0.1",
+        "gulp-util": "3.0.8",
+        "interpret": "1.1.0",
+        "liftoff": "2.5.0",
+        "minimist": "1.2.0",
+        "orchestrator": "0.3.8",
+        "pretty-hrtime": "1.0.3",
+        "semver": "4.3.6",
+        "tildify": "1.2.0",
+        "v8flags": "2.1.1",
+        "vinyl-fs": "0.3.14"
+      },
+      "dependencies": {
+        "semver": {
+          "version": "4.3.6",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-4.3.6.tgz",
+          "integrity": "sha1-MAvG4OhjdPe6YQaLWx7NV/xlMto=",
+          "dev": true
+        }
+      }
+    },
+    "gulp-nodemon": {
+      "version": "2.2.1",
+      "resolved": "https://registry.npmjs.org/gulp-nodemon/-/gulp-nodemon-2.2.1.tgz",
+      "integrity": "sha1-2b8Zn1WFRYFZ09KZFT5gtGhotvQ=",
+      "dev": true,
+      "requires": {
+        "colors": "1.1.2",
+        "event-stream": "3.3.4",
+        "gulp": "3.9.1",
+        "nodemon": "1.14.7"
+      }
+    },
+    "gulp-util": {
+      "version": "3.0.8",
+      "resolved": "https://registry.npmjs.org/gulp-util/-/gulp-util-3.0.8.tgz",
+      "integrity": "sha1-AFTh50RQLifATBh8PsxQXdVLu08=",
+      "dev": true,
+      "requires": {
+        "array-differ": "1.0.0",
+        "array-uniq": "1.0.3",
+        "beeper": "1.1.1",
+        "chalk": "1.1.3",
+        "dateformat": "2.2.0",
+        "fancy-log": "1.3.2",
+        "gulplog": "1.0.0",
+        "has-gulplog": "0.1.0",
+        "lodash._reescape": "3.0.0",
+        "lodash._reevaluate": "3.0.0",
+        "lodash._reinterpolate": "3.0.0",
+        "lodash.template": "3.6.2",
+        "minimist": "1.2.0",
+        "multipipe": "0.1.2",
+        "object-assign": "3.0.0",
+        "replace-ext": "0.0.1",
+        "through2": "2.0.3",
+        "vinyl": "0.5.3"
+      },
+      "dependencies": {
+        "object-assign": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-3.0.0.tgz",
+          "integrity": "sha1-m+3VygiXlJvKR+f/QIBi1Un1h/I=",
+          "dev": true
+        }
+      }
+    },
+    "gulplog": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/gulplog/-/gulplog-1.0.0.tgz",
+      "integrity": "sha1-4oxNRdBey77YGDY86PnFkmIp/+U=",
+      "dev": true,
+      "requires": {
+        "glogg": "1.0.0"
+      }
+    },
+    "has": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/has/-/has-1.0.1.tgz",
+      "integrity": "sha1-hGFzP1OLCDfJNh45qauelwTcLyg=",
+      "requires": {
+        "function-bind": "1.1.1"
+      }
+    },
+    "has-ansi": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz",
+      "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=",
+      "dev": true,
+      "requires": {
+        "ansi-regex": "2.1.1"
+      }
+    },
+    "has-binary2": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/has-binary2/-/has-binary2-1.0.2.tgz",
+      "integrity": "sha1-6D26SfC5vk0CbSc2U1DZ8D9Uvpg=",
+      "requires": {
+        "isarray": "2.0.1"
+      },
+      "dependencies": {
+        "isarray": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz",
+          "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4="
+        }
+      }
+    },
+    "has-cors": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/has-cors/-/has-cors-1.1.0.tgz",
+      "integrity": "sha1-XkdHk/fqmEPRu5nCPu9J/xJv/zk="
+    },
+    "has-flag": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz",
+      "integrity": "sha1-6CB68cx7MNRGzHC3NLXovhj4jVE="
+    },
+    "has-gulplog": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/has-gulplog/-/has-gulplog-0.1.0.tgz",
+      "integrity": "sha1-ZBTIKRNpfaUVkDl9r7EvIpZ4Ec4=",
+      "dev": true,
+      "requires": {
+        "sparkles": "1.0.0"
+      }
+    },
+    "has-value": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz",
+      "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=",
+      "dev": true,
+      "requires": {
+        "get-value": "2.0.6",
+        "has-values": "1.0.0",
+        "isobject": "3.0.1"
+      }
+    },
+    "has-values": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz",
+      "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=",
+      "dev": true,
+      "requires": {
+        "is-number": "3.0.0",
+        "kind-of": "4.0.0"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz",
+          "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=",
+          "dev": true,
+          "requires": {
+            "is-buffer": "1.1.6"
+          }
+        }
+      }
+    },
+    "homedir-polyfill": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/homedir-polyfill/-/homedir-polyfill-1.0.1.tgz",
+      "integrity": "sha1-TCu8inWJmP7r9e1oWA921GdotLw=",
+      "dev": true,
+      "requires": {
+        "parse-passwd": "1.0.0"
+      }
+    },
+    "htmlparser2": {
+      "version": "3.9.2",
+      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.9.2.tgz",
+      "integrity": "sha1-G9+HrMoPP55T+k/M6w9LTLsAszg=",
+      "requires": {
+        "domelementtype": "1.3.0",
+        "domhandler": "2.4.1",
+        "domutils": "1.6.2",
+        "entities": "1.1.1",
+        "inherits": "2.0.3",
+        "readable-stream": "2.3.3"
+      }
+    },
+    "http-errors": {
+      "version": "1.6.2",
+      "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz",
+      "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=",
+      "requires": {
+        "depd": "1.1.1",
+        "inherits": "2.0.3",
+        "setprototypeof": "1.0.3",
+        "statuses": "1.3.1"
+      },
+      "dependencies": {
+        "setprototypeof": {
+          "version": "1.0.3",
+          "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz",
+          "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ="
+        }
+      }
+    },
+    "iconv-lite": {
+      "version": "0.4.19",
+      "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz",
+      "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ=="
+    },
+    "ignore-by-default": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
+      "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=",
+      "dev": true
+    },
+    "import-lazy": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz",
+      "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=",
+      "dev": true
+    },
+    "imurmurhash": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
+      "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
+      "dev": true
+    },
+    "indexof": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/indexof/-/indexof-0.0.1.tgz",
+      "integrity": "sha1-gtwzbSMrkGIXnQWrMpOmYFn9Q10="
+    },
+    "inflight": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+      "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
+      "dev": true,
+      "requires": {
+        "once": "1.4.0",
+        "wrappy": "1.0.2"
+      }
+    },
+    "inherits": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
+      "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
+    },
+    "ini": {
+      "version": "1.3.5",
+      "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz",
+      "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==",
+      "dev": true
+    },
+    "interpret": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.1.0.tgz",
+      "integrity": "sha1-ftGxQQxqDg94z5XTuEQMY/eLhhQ=",
+      "dev": true
+    },
+    "ipaddr.js": {
+      "version": "1.5.2",
+      "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.5.2.tgz",
+      "integrity": "sha1-1LUFvemUaYfM8PxY2QEP+WB+P6A="
+    },
+    "is-absolute": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-absolute/-/is-absolute-1.0.0.tgz",
+      "integrity": "sha512-dOWoqflvcydARa360Gvv18DZ/gRuHKi2NU/wU5X1ZFzdYfH29nkiNZsF3mp4OJ3H4yo9Mx8A/uAGNzpzPN3yBA==",
+      "dev": true,
+      "requires": {
+        "is-relative": "1.0.0",
+        "is-windows": "1.0.1"
+      }
+    },
+    "is-accessor-descriptor": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz",
+      "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==",
+      "dev": true,
+      "requires": {
+        "kind-of": "6.0.2"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "6.0.2",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz",
+          "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==",
+          "dev": true
+        }
+      }
+    },
+    "is-binary-path": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz",
+      "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=",
+      "dev": true,
+      "requires": {
+        "binary-extensions": "1.11.0"
+      }
+    },
+    "is-buffer": {
+      "version": "1.1.6",
+      "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz",
+      "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w=="
+    },
+    "is-data-descriptor": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz",
+      "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==",
+      "dev": true,
+      "requires": {
+        "kind-of": "6.0.2"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "6.0.2",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz",
+          "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==",
+          "dev": true
+        }
+      }
+    },
+    "is-descriptor": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz",
+      "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==",
+      "dev": true,
+      "requires": {
+        "is-accessor-descriptor": "1.0.0",
+        "is-data-descriptor": "1.0.0",
+        "kind-of": "6.0.2"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "6.0.2",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz",
+          "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==",
+          "dev": true
+        }
+      }
+    },
+    "is-dotfile": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/is-dotfile/-/is-dotfile-1.0.3.tgz",
+      "integrity": "sha1-pqLzL/0t+wT1yiXs0Pa4PPeYoeE=",
+      "dev": true
+    },
+    "is-equal-shallow": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/is-equal-shallow/-/is-equal-shallow-0.1.3.tgz",
+      "integrity": "sha1-IjgJj8Ih3gvPpdnqxMRdY4qhxTQ=",
+      "dev": true,
+      "requires": {
+        "is-primitive": "2.0.0"
+      }
+    },
+    "is-expression": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/is-expression/-/is-expression-2.1.0.tgz",
+      "integrity": "sha1-kb6dR968/vB3l36XIr5tz7RGXvA=",
+      "requires": {
+        "acorn": "3.3.0",
+        "object-assign": "4.1.1"
+      }
+    },
+    "is-extendable": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz",
+      "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=",
+      "dev": true
+    },
+    "is-extglob": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+      "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
+      "dev": true
+    },
+    "is-fullwidth-code-point": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
+      "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
+      "dev": true
+    },
+    "is-glob": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz",
+      "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=",
+      "dev": true,
+      "requires": {
+        "is-extglob": "2.1.1"
+      }
+    },
+    "is-installed-globally": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.1.0.tgz",
+      "integrity": "sha1-Df2Y9akRFxbdU13aZJL2e/PSWoA=",
+      "dev": true,
+      "requires": {
+        "global-dirs": "0.1.1",
+        "is-path-inside": "1.0.1"
+      }
+    },
+    "is-npm": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-1.0.0.tgz",
+      "integrity": "sha1-8vtjpl5JBbQGyGBydloaTceTufQ=",
+      "dev": true
+    },
+    "is-number": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz",
+      "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=",
+      "dev": true,
+      "requires": {
+        "kind-of": "3.2.2"
+      }
+    },
+    "is-obj": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz",
+      "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=",
+      "dev": true
+    },
+    "is-odd": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-odd/-/is-odd-1.0.0.tgz",
+      "integrity": "sha1-O4qTLrAos3dcObsJ6RdnrM22kIg=",
+      "dev": true,
+      "requires": {
+        "is-number": "3.0.0"
+      }
+    },
+    "is-path-inside": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz",
+      "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=",
+      "dev": true,
+      "requires": {
+        "path-is-inside": "1.0.2"
+      }
+    },
+    "is-plain-object": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz",
+      "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==",
+      "dev": true,
+      "requires": {
+        "isobject": "3.0.1"
+      }
+    },
+    "is-posix-bracket": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/is-posix-bracket/-/is-posix-bracket-0.1.1.tgz",
+      "integrity": "sha1-MzTceXdDaOkvAW5vvAqI9c1ua8Q=",
+      "dev": true
+    },
+    "is-primitive": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/is-primitive/-/is-primitive-2.0.0.tgz",
+      "integrity": "sha1-IHurkWOEmcB7Kt8kCkGochADRXU=",
+      "dev": true
+    },
+    "is-promise": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz",
+      "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o="
+    },
+    "is-redirect": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz",
+      "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=",
+      "dev": true
+    },
+    "is-regex": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.4.tgz",
+      "integrity": "sha1-VRdIm1RwkbCTDglWVM7SXul+lJE=",
+      "requires": {
+        "has": "1.0.1"
+      }
+    },
+    "is-relative": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-relative/-/is-relative-1.0.0.tgz",
+      "integrity": "sha512-Kw/ReK0iqwKeu0MITLFuj0jbPAmEiOsIwyIXvvbfa6QfmN9pkD1M+8pdk7Rl/dTKbH34/XBFMbgD4iMJhLQbGA==",
+      "dev": true,
+      "requires": {
+        "is-unc-path": "1.0.0"
+      }
+    },
+    "is-retry-allowed": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.1.0.tgz",
+      "integrity": "sha1-EaBgVotnM5REAz0BJaYaINVk+zQ=",
+      "dev": true
+    },
+    "is-stream": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz",
+      "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=",
+      "dev": true
+    },
+    "is-unc-path": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-unc-path/-/is-unc-path-1.0.0.tgz",
+      "integrity": "sha512-mrGpVd0fs7WWLfVsStvgF6iEJnbjDFZh9/emhRDcGWTduTfNHd9CHeUwH3gYIjdbwo4On6hunkztwOaAw0yllQ==",
+      "dev": true,
+      "requires": {
+        "unc-path-regex": "0.1.2"
+      }
+    },
+    "is-utf8": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz",
+      "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=",
+      "dev": true
+    },
+    "is-windows": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.1.tgz",
+      "integrity": "sha1-MQ23D3QtJZoWo2kgK1GvhCMzENk=",
+      "dev": true
+    },
+    "isarray": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
+      "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
+    },
+    "isexe": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+      "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=",
+      "dev": true
+    },
+    "isobject": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz",
+      "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=",
+      "dev": true
+    },
+    "js-stringify": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/js-stringify/-/js-stringify-1.0.2.tgz",
+      "integrity": "sha1-Fzb939lyTyijaCrcYjCufk6Weds="
+    },
+    "jstransformer": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/jstransformer/-/jstransformer-1.0.0.tgz",
+      "integrity": "sha1-7Yvwkh4vPx7U1cGkT2hwntJHIsM=",
+      "requires": {
+        "is-promise": "2.1.0",
+        "promise": "7.3.1"
+      }
+    },
+    "kind-of": {
+      "version": "3.2.2",
+      "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+      "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+      "requires": {
+        "is-buffer": "1.1.6"
+      }
+    },
+    "latest-version": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-3.1.0.tgz",
+      "integrity": "sha1-ogU4P+oyKzO1rjsYq+4NwvNW7hU=",
+      "dev": true,
+      "requires": {
+        "package-json": "4.0.1"
+      }
+    },
+    "lazy-cache": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-1.0.4.tgz",
+      "integrity": "sha1-odePw6UEdMuAhF07O24dpJpEbo4="
+    },
+    "liftoff": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/liftoff/-/liftoff-2.5.0.tgz",
+      "integrity": "sha1-IAkpG7Mc6oYbvxCnwVooyvdcMew=",
+      "dev": true,
+      "requires": {
+        "extend": "3.0.1",
+        "findup-sync": "2.0.0",
+        "fined": "1.1.0",
+        "flagged-respawn": "1.0.0",
+        "is-plain-object": "2.0.4",
+        "object.map": "1.0.1",
+        "rechoir": "0.6.2",
+        "resolve": "1.5.0"
+      }
+    },
+    "lodash": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/lodash/-/lodash-1.0.2.tgz",
+      "integrity": "sha1-j1dWDIO1n8JwvT1WG2kAQ0MOJVE=",
+      "dev": true
+    },
+    "lodash._basecopy": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/lodash._basecopy/-/lodash._basecopy-3.0.1.tgz",
+      "integrity": "sha1-jaDmqHbPNEwK2KVIghEd08XHyjY=",
+      "dev": true
+    },
+    "lodash._basetostring": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/lodash._basetostring/-/lodash._basetostring-3.0.1.tgz",
+      "integrity": "sha1-0YYdh3+CSlL2aYMtyvPuFVZqB9U=",
+      "dev": true
+    },
+    "lodash._basevalues": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/lodash._basevalues/-/lodash._basevalues-3.0.0.tgz",
+      "integrity": "sha1-W3dXYoAr3j0yl1A+JjAIIP32Ybc=",
+      "dev": true
+    },
+    "lodash._getnative": {
+      "version": "3.9.1",
+      "resolved": "https://registry.npmjs.org/lodash._getnative/-/lodash._getnative-3.9.1.tgz",
+      "integrity": "sha1-VwvH3t5G1hzc3mh9ZdPuy6o6r/U=",
+      "dev": true
+    },
+    "lodash._isiterateecall": {
+      "version": "3.0.9",
+      "resolved": "https://registry.npmjs.org/lodash._isiterateecall/-/lodash._isiterateecall-3.0.9.tgz",
+      "integrity": "sha1-UgOte6Ql+uhCRg5pbbnPPmqsBXw=",
+      "dev": true
+    },
+    "lodash._reescape": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/lodash._reescape/-/lodash._reescape-3.0.0.tgz",
+      "integrity": "sha1-Kx1vXf4HyKNVdT5fJ/rH8c3hYWo=",
+      "dev": true
+    },
+    "lodash._reevaluate": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/lodash._reevaluate/-/lodash._reevaluate-3.0.0.tgz",
+      "integrity": "sha1-WLx0xAZklTrgsSTYBpltrKQx4u0=",
+      "dev": true
+    },
+    "lodash._reinterpolate": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz",
+      "integrity": "sha1-DM8tiRZq8Ds2Y8eWU4t1rG4RTZ0=",
+      "dev": true
+    },
+    "lodash._root": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/lodash._root/-/lodash._root-3.0.1.tgz",
+      "integrity": "sha1-+6HEUkwZ7ppfgTa0YJ8BfPTe1pI=",
+      "dev": true
+    },
+    "lodash.clonedeep": {
+      "version": "4.5.0",
+      "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz",
+      "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8="
+    },
+    "lodash.escape": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/lodash.escape/-/lodash.escape-3.2.0.tgz",
+      "integrity": "sha1-mV7g3BjBtIzJLv+ucaEKq1tIdpg=",
+      "dev": true,
+      "requires": {
+        "lodash._root": "3.0.1"
+      }
+    },
+    "lodash.escaperegexp": {
+      "version": "4.1.2",
+      "resolved": "https://registry.npmjs.org/lodash.escaperegexp/-/lodash.escaperegexp-4.1.2.tgz",
+      "integrity": "sha1-ZHYsSGGAglGKw99Mz11YhtriA0c="
+    },
+    "lodash.isarguments": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz",
+      "integrity": "sha1-L1c9hcaiQon/AGY7SRwdM4/zRYo=",
+      "dev": true
+    },
+    "lodash.isarray": {
+      "version": "3.0.4",
+      "resolved": "https://registry.npmjs.org/lodash.isarray/-/lodash.isarray-3.0.4.tgz",
+      "integrity": "sha1-eeTriMNqgSKvhvhEqpvNhRtfu1U=",
+      "dev": true
+    },
+    "lodash.keys": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/lodash.keys/-/lodash.keys-3.1.2.tgz",
+      "integrity": "sha1-TbwEcrFWvlCgsoaFXRvQsMZWCYo=",
+      "dev": true,
+      "requires": {
+        "lodash._getnative": "3.9.1",
+        "lodash.isarguments": "3.1.0",
+        "lodash.isarray": "3.0.4"
+      }
+    },
+    "lodash.mergewith": {
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/lodash.mergewith/-/lodash.mergewith-4.6.0.tgz",
+      "integrity": "sha1-FQzwoWeR9ZA7iJHqsVRgknS96lU="
+    },
+    "lodash.restparam": {
+      "version": "3.6.1",
+      "resolved": "https://registry.npmjs.org/lodash.restparam/-/lodash.restparam-3.6.1.tgz",
+      "integrity": "sha1-k2pOMJ7zMKdkXtQUWYbIWuWyCAU=",
+      "dev": true
+    },
+    "lodash.template": {
+      "version": "3.6.2",
+      "resolved": "https://registry.npmjs.org/lodash.template/-/lodash.template-3.6.2.tgz",
+      "integrity": "sha1-+M3sxhaaJVvpCYrosMU9N4kx0U8=",
+      "dev": true,
+      "requires": {
+        "lodash._basecopy": "3.0.1",
+        "lodash._basetostring": "3.0.1",
+        "lodash._basevalues": "3.0.0",
+        "lodash._isiterateecall": "3.0.9",
+        "lodash._reinterpolate": "3.0.0",
+        "lodash.escape": "3.2.0",
+        "lodash.keys": "3.1.2",
+        "lodash.restparam": "3.6.1",
+        "lodash.templatesettings": "3.1.1"
+      }
+    },
+    "lodash.templatesettings": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/lodash.templatesettings/-/lodash.templatesettings-3.1.1.tgz",
+      "integrity": "sha1-+zB4RHU7Zrnxr6VOJix0UwfbqOU=",
+      "dev": true,
+      "requires": {
+        "lodash._reinterpolate": "3.0.0",
+        "lodash.escape": "3.2.0"
+      }
+    },
+    "longest": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/longest/-/longest-1.0.1.tgz",
+      "integrity": "sha1-MKCy2jj3N3DoKUoNIuZiXtd9AJc="
+    },
+    "lowercase-keys": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.0.tgz",
+      "integrity": "sha1-TjNms55/VFfjXxMkvfb4jQv8cwY=",
+      "dev": true
+    },
+    "lru-cache": {
+      "version": "2.7.3",
+      "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-2.7.3.tgz",
+      "integrity": "sha1-bUUk6LlV+V1PW1iFHOId1y+06VI=",
+      "dev": true
+    },
+    "make-dir": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-1.1.0.tgz",
+      "integrity": "sha512-0Pkui4wLJ7rxvmfUvs87skoEaxmu0hCUApF8nonzpl7q//FWp9zu8W61Scz4sd/kUiqDxvUhtoam2efDyiBzcA==",
+      "dev": true,
+      "requires": {
+        "pify": "3.0.0"
+      }
+    },
+    "make-iterator": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/make-iterator/-/make-iterator-1.0.0.tgz",
+      "integrity": "sha1-V7713IXSOSO6I3ZzJNjo+PPZaUs=",
+      "dev": true,
+      "requires": {
+        "kind-of": "3.2.2"
+      }
+    },
+    "map-cache": {
+      "version": "0.2.2",
+      "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz",
+      "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=",
+      "dev": true
+    },
+    "map-stream": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/map-stream/-/map-stream-0.1.0.tgz",
+      "integrity": "sha1-5WqpTEyAVaFkBKBnS3jyFffI4ZQ=",
+      "dev": true
+    },
+    "map-visit": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz",
+      "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=",
+      "dev": true,
+      "requires": {
+        "object-visit": "1.0.1"
+      }
+    },
+    "media-typer": {
+      "version": "0.3.0",
+      "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
+      "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
+    },
+    "merge-descriptors": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
+      "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
+    },
+    "methods": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
+      "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
+    },
+    "micromatch": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.4.tgz",
+      "integrity": "sha512-kFRtviKYoAJT+t7HggMl0tBFGNAKLw/S7N+CO9qfEQyisob1Oy4pao+geRbkyeEd+V9aOkvZ4mhuyPvI/q9Sfg==",
+      "dev": true,
+      "requires": {
+        "arr-diff": "4.0.0",
+        "array-unique": "0.3.2",
+        "braces": "2.3.0",
+        "define-property": "1.0.0",
+        "extend-shallow": "2.0.1",
+        "extglob": "2.0.3",
+        "fragment-cache": "0.2.1",
+        "kind-of": "6.0.2",
+        "nanomatch": "1.2.6",
+        "object.pick": "1.3.0",
+        "regex-not": "1.0.0",
+        "snapdragon": "0.8.1",
+        "to-regex": "3.0.1"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "6.0.2",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz",
+          "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==",
+          "dev": true
+        }
+      }
+    },
+    "mime": {
+      "version": "1.4.1",
+      "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz",
+      "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ=="
+    },
+    "mime-db": {
+      "version": "1.30.0",
+      "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.30.0.tgz",
+      "integrity": "sha1-dMZD2i3Z1qRTmZY0ZbJtXKfXHwE="
+    },
+    "mime-types": {
+      "version": "2.1.17",
+      "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.17.tgz",
+      "integrity": "sha1-Cdejk/A+mVp5+K+Fe3Cp4KsWVXo=",
+      "requires": {
+        "mime-db": "1.30.0"
+      }
+    },
+    "minimatch": {
+      "version": "2.0.10",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-2.0.10.tgz",
+      "integrity": "sha1-jQh8OcazjAAbl/ynzm0OHoCvusc=",
+      "dev": true,
+      "requires": {
+        "brace-expansion": "1.1.8"
+      }
+    },
+    "minimist": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz",
+      "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=",
+      "dev": true
+    },
+    "mixin-deep": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.0.tgz",
+      "integrity": "sha512-dgaCvoh6i1nosAUBKb0l0pfJ78K8+S9fluyIR2YvAeUD/QuMahnFnF3xYty5eYXMjhGSsB0DsW6A0uAZyetoAg==",
+      "dev": true,
+      "requires": {
+        "for-in": "1.0.2",
+        "is-extendable": "1.0.1"
+      },
+      "dependencies": {
+        "is-extendable": {
+          "version": "1.0.1",
+          "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz",
+          "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==",
+          "dev": true,
+          "requires": {
+            "is-plain-object": "2.0.4"
+          }
+        }
+      }
+    },
+    "mkdirp": {
+      "version": "0.5.1",
+      "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
+      "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=",
+      "dev": true,
+      "requires": {
+        "minimist": "0.0.8"
+      },
+      "dependencies": {
+        "minimist": {
+          "version": "0.0.8",
+          "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
+          "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=",
+          "dev": true
+        }
+      }
+    },
+    "mongodb": {
+      "version": "2.2.33",
+      "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-2.2.33.tgz",
+      "integrity": "sha1-tTfEcdNKZlG0jzb9vyl1A0Dgi1A=",
+      "requires": {
+        "es6-promise": "3.2.1",
+        "mongodb-core": "2.1.17",
+        "readable-stream": "2.2.7"
+      },
+      "dependencies": {
+        "readable-stream": {
+          "version": "2.2.7",
+          "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.2.7.tgz",
+          "integrity": "sha1-BwV6y+JGeyIELTb5jFrVBwVOlbE=",
+          "requires": {
+            "buffer-shims": "1.0.0",
+            "core-util-is": "1.0.2",
+            "inherits": "2.0.3",
+            "isarray": "1.0.0",
+            "process-nextick-args": "1.0.7",
+            "string_decoder": "1.0.3",
+            "util-deprecate": "1.0.2"
+          }
+        }
+      }
+    },
+    "mongodb-core": {
+      "version": "2.1.17",
+      "resolved": "https://registry.npmjs.org/mongodb-core/-/mongodb-core-2.1.17.tgz",
+      "integrity": "sha1-pBizN6FKFJkPtRC5I97mqBMXPfg=",
+      "requires": {
+        "bson": "1.0.4",
+        "require_optional": "1.0.1"
+      }
+    },
+    "mongojs": {
+      "version": "2.4.1",
+      "resolved": "https://registry.npmjs.org/mongojs/-/mongojs-2.4.1.tgz",
+      "integrity": "sha512-R344Q8ufjcqyFHO1CrxYboUBrEJwmsvMBtI8wsjCZq90mh/lzT0PBleAD6d1f8s07zeHSM2ebeu3OwMC4wxQlg==",
+      "requires": {
+        "each-series": "1.0.0",
+        "mongodb": "2.2.33",
+        "once": "1.4.0",
+        "parse-mongo-url": "1.1.1",
+        "readable-stream": "2.3.3",
+        "thunky": "1.0.2",
+        "to-mongodb-core": "2.0.0",
+        "xtend": "4.0.1"
+      }
+    },
+    "morgan": {
+      "version": "1.9.0",
+      "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.0.tgz",
+      "integrity": "sha1-0B+mxlhZt2/PMbPLU6OCGjEdgFE=",
+      "requires": {
+        "basic-auth": "2.0.0",
+        "debug": "2.6.9",
+        "depd": "1.1.1",
+        "on-finished": "2.3.0",
+        "on-headers": "1.0.1"
+      }
+    },
+    "ms": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+      "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
+    },
+    "multipipe": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/multipipe/-/multipipe-0.1.2.tgz",
+      "integrity": "sha1-Ko8t33Du1WTf8tV/HhoTfZ8FB4s=",
+      "dev": true,
+      "requires": {
+        "duplexer2": "0.0.2"
+      }
+    },
+    "nan": {
+      "version": "2.8.0",
+      "resolved": "https://registry.npmjs.org/nan/-/nan-2.8.0.tgz",
+      "integrity": "sha1-7XFfP+neArV6XmJS2QqWZ14fCFo=",
+      "dev": true,
+      "optional": true
+    },
+    "nanomatch": {
+      "version": "1.2.6",
+      "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.6.tgz",
+      "integrity": "sha512-WJ6XTCbvWXUFPbi/bDwKcYkCeOGUHzaJj72KbuPqGn78Ba/F5Vu26Zlo6SuMQbCIst1RGKL1zfWBCOGAlbRLAg==",
+      "dev": true,
+      "requires": {
+        "arr-diff": "4.0.0",
+        "array-unique": "0.3.2",
+        "define-property": "1.0.0",
+        "extend-shallow": "2.0.1",
+        "fragment-cache": "0.2.1",
+        "is-odd": "1.0.0",
+        "kind-of": "5.1.0",
+        "object.pick": "1.3.0",
+        "regex-not": "1.0.0",
+        "snapdragon": "0.8.1",
+        "to-regex": "3.0.1"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        }
+      }
+    },
+    "natives": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/natives/-/natives-1.1.1.tgz",
+      "integrity": "sha512-8eRaxn8u/4wN8tGkhlc2cgwwvOLMLUMUn4IYTexMgWd+LyUDfeXVkk2ygQR0hvIHbJQXgHujia3ieUUDwNGkEA==",
+      "dev": true
+    },
+    "negotiator": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz",
+      "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk="
+    },
+    "node-cron": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/node-cron/-/node-cron-1.2.1.tgz",
+      "integrity": "sha1-jJC8XccjpWKJsHhmVatKHEy2A2g="
+    },
+    "nodemon": {
+      "version": "1.14.7",
+      "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-1.14.7.tgz",
+      "integrity": "sha512-uEguLNr+QIk4TVd8swNvw7kHqOE/sjvNsIwhnc8CM7QdI+ezFvvkMRtCpCJ+DEVyIopLSTu2eayZ/ELKtswcbg==",
+      "dev": true,
+      "requires": {
+        "chokidar": "1.7.0",
+        "debug": "2.6.9",
+        "ignore-by-default": "1.0.1",
+        "minimatch": "3.0.4",
+        "pstree.remy": "1.1.0",
+        "touch": "3.1.0",
+        "undefsafe": "0.0.3",
+        "update-notifier": "2.3.0"
+      },
+      "dependencies": {
+        "minimatch": {
+          "version": "3.0.4",
+          "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
+          "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
+          "dev": true,
+          "requires": {
+            "brace-expansion": "1.1.8"
+          }
+        }
+      }
+    },
+    "nopt": {
+      "version": "1.0.10",
+      "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz",
+      "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=",
+      "dev": true,
+      "requires": {
+        "abbrev": "1.1.1"
+      }
+    },
+    "normalize-path": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz",
+      "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=",
+      "dev": true,
+      "requires": {
+        "remove-trailing-separator": "1.1.0"
+      }
+    },
+    "npm-run-path": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz",
+      "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=",
+      "dev": true,
+      "requires": {
+        "path-key": "2.0.1"
+      }
+    },
+    "number-is-nan": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
+      "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0="
+    },
+    "object-assign": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+      "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
+    },
+    "object-component": {
+      "version": "0.0.3",
+      "resolved": "https://registry.npmjs.org/object-component/-/object-component-0.0.3.tgz",
+      "integrity": "sha1-8MaapQ78lbhmwYb0AKM3acsvEpE="
+    },
+    "object-copy": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz",
+      "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=",
+      "dev": true,
+      "requires": {
+        "copy-descriptor": "0.1.1",
+        "define-property": "0.2.5",
+        "kind-of": "3.2.2"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "5.1.0",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+              "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+              "dev": true
+            }
+          }
+        }
+      }
+    },
+    "object-visit": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz",
+      "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=",
+      "dev": true,
+      "requires": {
+        "isobject": "3.0.1"
+      }
+    },
+    "object.defaults": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/object.defaults/-/object.defaults-1.1.0.tgz",
+      "integrity": "sha1-On+GgzS0B96gbaFtiNXNKeQ1/s8=",
+      "dev": true,
+      "requires": {
+        "array-each": "1.0.1",
+        "array-slice": "1.1.0",
+        "for-own": "1.0.0",
+        "isobject": "3.0.1"
+      }
+    },
+    "object.map": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/object.map/-/object.map-1.0.1.tgz",
+      "integrity": "sha1-z4Plncj8wK1fQlDh94s7gb2AHTc=",
+      "dev": true,
+      "requires": {
+        "for-own": "1.0.0",
+        "make-iterator": "1.0.0"
+      }
+    },
+    "object.omit": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/object.omit/-/object.omit-2.0.1.tgz",
+      "integrity": "sha1-Gpx0SCnznbuFjHbKNXmuKlTr0fo=",
+      "dev": true,
+      "requires": {
+        "for-own": "0.1.5",
+        "is-extendable": "0.1.1"
+      },
+      "dependencies": {
+        "for-own": {
+          "version": "0.1.5",
+          "resolved": "https://registry.npmjs.org/for-own/-/for-own-0.1.5.tgz",
+          "integrity": "sha1-UmXGgaTylNq78XyVCbZ2OqhFEM4=",
+          "dev": true,
+          "requires": {
+            "for-in": "1.0.2"
+          }
+        }
+      }
+    },
+    "object.pick": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz",
+      "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=",
+      "dev": true,
+      "requires": {
+        "isobject": "3.0.1"
+      }
+    },
+    "on-finished": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
+      "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
+      "requires": {
+        "ee-first": "1.1.1"
+      }
+    },
+    "on-headers": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.1.tgz",
+      "integrity": "sha1-ko9dD0cNSTQmUepnlLCFfBAGk/c="
+    },
+    "once": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+      "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
+      "requires": {
+        "wrappy": "1.0.2"
+      }
+    },
+    "orchestrator": {
+      "version": "0.3.8",
+      "resolved": "https://registry.npmjs.org/orchestrator/-/orchestrator-0.3.8.tgz",
+      "integrity": "sha1-FOfp4nZPcxX7rBhOUGx6pt+UrX4=",
+      "dev": true,
+      "requires": {
+        "end-of-stream": "0.1.5",
+        "sequencify": "0.0.7",
+        "stream-consume": "0.1.0"
+      }
+    },
+    "ordered-read-streams": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/ordered-read-streams/-/ordered-read-streams-0.1.0.tgz",
+      "integrity": "sha1-/VZamvjrRHO6abbtijQ1LLVS8SY=",
+      "dev": true
+    },
+    "os-homedir": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz",
+      "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=",
+      "dev": true
+    },
+    "p-finally": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz",
+      "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=",
+      "dev": true
+    },
+    "package-json": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/package-json/-/package-json-4.0.1.tgz",
+      "integrity": "sha1-iGmgQBJTZhxMTKPabCEh7VVfXu0=",
+      "dev": true,
+      "requires": {
+        "got": "6.7.1",
+        "registry-auth-token": "3.3.1",
+        "registry-url": "3.1.0",
+        "semver": "5.4.1"
+      }
+    },
+    "parse-filepath": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/parse-filepath/-/parse-filepath-1.0.2.tgz",
+      "integrity": "sha1-pjISf1Oq89FYdvWHLz/6x2PWyJE=",
+      "dev": true,
+      "requires": {
+        "is-absolute": "1.0.0",
+        "map-cache": "0.2.2",
+        "path-root": "0.1.1"
+      }
+    },
+    "parse-glob": {
+      "version": "3.0.4",
+      "resolved": "https://registry.npmjs.org/parse-glob/-/parse-glob-3.0.4.tgz",
+      "integrity": "sha1-ssN2z7EfNVE7rdFz7wu246OIORw=",
+      "dev": true,
+      "requires": {
+        "glob-base": "0.3.0",
+        "is-dotfile": "1.0.3",
+        "is-extglob": "1.0.0",
+        "is-glob": "2.0.1"
+      },
+      "dependencies": {
+        "is-extglob": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz",
+          "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=",
+          "dev": true
+        },
+        "is-glob": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz",
+          "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=",
+          "dev": true,
+          "requires": {
+            "is-extglob": "1.0.0"
+          }
+        }
+      }
+    },
+    "parse-mongo-url": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/parse-mongo-url/-/parse-mongo-url-1.1.1.tgz",
+      "integrity": "sha1-ZiON9fjnwMjKTNlw1KtqE3PrdbU="
+    },
+    "parse-passwd": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/parse-passwd/-/parse-passwd-1.0.0.tgz",
+      "integrity": "sha1-bVuTSkVpk7I9N/QKOC1vFmao5cY=",
+      "dev": true
+    },
+    "parseqs": {
+      "version": "0.0.5",
+      "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.5.tgz",
+      "integrity": "sha1-1SCKNzjkZ2bikbouoXNoSSGouJ0=",
+      "requires": {
+        "better-assert": "1.0.2"
+      }
+    },
+    "parseuri": {
+      "version": "0.0.5",
+      "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.5.tgz",
+      "integrity": "sha1-gCBKUNTbt3m/3G6+J3jZDkvOMgo=",
+      "requires": {
+        "better-assert": "1.0.2"
+      }
+    },
+    "parseurl": {
+      "version": "1.3.2",
+      "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz",
+      "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M="
+    },
+    "pascalcase": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz",
+      "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=",
+      "dev": true
+    },
+    "path-is-absolute": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+      "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
+      "dev": true
+    },
+    "path-is-inside": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz",
+      "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=",
+      "dev": true
+    },
+    "path-key": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz",
+      "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=",
+      "dev": true
+    },
+    "path-parse": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz",
+      "integrity": "sha1-PBrfhx6pzWyUMbbqK9dKD/BVxME="
+    },
+    "path-root": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/path-root/-/path-root-0.1.1.tgz",
+      "integrity": "sha1-mkpoFMrBwM1zNgqV8yCDyOpHRbc=",
+      "dev": true,
+      "requires": {
+        "path-root-regex": "0.1.2"
+      }
+    },
+    "path-root-regex": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/path-root-regex/-/path-root-regex-0.1.2.tgz",
+      "integrity": "sha1-v8zcjfWxLcUsi0PsONGNcsBLqW0=",
+      "dev": true
+    },
+    "path-to-regexp": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
+      "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
+    },
+    "pause-stream": {
+      "version": "0.0.11",
+      "resolved": "https://registry.npmjs.org/pause-stream/-/pause-stream-0.0.11.tgz",
+      "integrity": "sha1-/lo0sMvOErWqaitAPuLnO2AvFEU=",
+      "dev": true,
+      "requires": {
+        "through": "2.3.8"
+      }
+    },
+    "pify": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz",
+      "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=",
+      "dev": true
+    },
+    "posix-character-classes": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz",
+      "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=",
+      "dev": true
+    },
+    "postcss": {
+      "version": "6.0.15",
+      "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.15.tgz",
+      "integrity": "sha512-v/SpyMzLbtkmh45zUdaqLAaqXqzPdSrw8p4cQVO0/w6YiYfpj4k+Wkzhn68qk9br+H+0qfddhdPEVnbmBPfXVQ==",
+      "requires": {
+        "chalk": "2.3.0",
+        "source-map": "0.6.1",
+        "supports-color": "5.1.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.0",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.0.tgz",
+          "integrity": "sha512-NnSOmMEYtVR2JVMIGTzynRkkaxtiq1xnFBcdQD/DnNCYPoEPsVJhM98BDyaoNOQIi7p4okdi3E27eN7GQbsUug==",
+          "requires": {
+            "color-convert": "1.9.1"
+          }
+        },
+        "chalk": {
+          "version": "2.3.0",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.0.tgz",
+          "integrity": "sha512-Az5zJR2CBujap2rqXGaJKaPHyJ0IrUimvYNX+ncCy8PJP4ltOGTrHUIo097ZaL2zMeKYpiCdqDvS6zdrTFok3Q==",
+          "requires": {
+            "ansi-styles": "3.2.0",
+            "escape-string-regexp": "1.0.5",
+            "supports-color": "4.5.0"
+          },
+          "dependencies": {
+            "supports-color": {
+              "version": "4.5.0",
+              "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-4.5.0.tgz",
+              "integrity": "sha1-vnoN5ITexcXN34s9WRJQRJEvY1s=",
+              "requires": {
+                "has-flag": "2.0.0"
+              }
+            }
+          }
+        },
+        "source-map": {
+          "version": "0.6.1",
+          "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
+          "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g=="
+        },
+        "supports-color": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.1.0.tgz",
+          "integrity": "sha512-Ry0AwkoKjDpVKK4sV4h6o3UJmNRbjYm2uXhwfj3J56lMVdvnUNqzQVRztOOMGQ++w1K/TjNDFvpJk0F/LoeBCQ==",
+          "requires": {
+            "has-flag": "2.0.0"
+          }
+        }
+      }
+    },
+    "prepend-http": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz",
+      "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=",
+      "dev": true
+    },
+    "preserve": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/preserve/-/preserve-0.2.0.tgz",
+      "integrity": "sha1-gV7R9uvGWSb4ZbMQwHE7yzMVzks=",
+      "dev": true
+    },
+    "pretty-hrtime": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/pretty-hrtime/-/pretty-hrtime-1.0.3.tgz",
+      "integrity": "sha1-t+PqQkNaTJsnWdmeDyAesZWALuE=",
+      "dev": true
+    },
+    "process-nextick-args": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-1.0.7.tgz",
+      "integrity": "sha1-FQ4gt1ZZCtP5EJPyWk8q2L/zC6M="
+    },
+    "promise": {
+      "version": "7.3.1",
+      "resolved": "https://registry.npmjs.org/promise/-/promise-7.3.1.tgz",
+      "integrity": "sha512-nolQXZ/4L+bP/UGlkfaIujX9BKxGwmQ9OT4mOt5yvy8iK1h3wqTEJCijzGANTCCl9nWjY41juyAn2K3Q1hLLTg==",
+      "requires": {
+        "asap": "2.0.6"
+      }
+    },
+    "proxy-addr": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.2.tgz",
+      "integrity": "sha1-ZXFQT0e7mI7IGAJT+F3X4UlSvew=",
+      "requires": {
+        "forwarded": "0.1.2",
+        "ipaddr.js": "1.5.2"
+      }
+    },
+    "ps-tree": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/ps-tree/-/ps-tree-1.1.0.tgz",
+      "integrity": "sha1-tCGyQUDWID8e08dplrRCewjowBQ=",
+      "dev": true,
+      "requires": {
+        "event-stream": "3.3.4"
+      }
+    },
+    "pseudomap": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz",
+      "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=",
+      "dev": true
+    },
+    "pstree.remy": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.0.tgz",
+      "integrity": "sha512-q5I5vLRMVtdWa8n/3UEzZX7Lfghzrg9eG2IKk2ENLSofKRCXVqMvMUHxCKgXNaqH/8ebhBxrqftHWnyTFweJ5Q==",
+      "dev": true,
+      "requires": {
+        "ps-tree": "1.1.0"
+      }
+    },
+    "pug": {
+      "version": "2.0.0-rc.4",
+      "resolved": "https://registry.npmjs.org/pug/-/pug-2.0.0-rc.4.tgz",
+      "integrity": "sha512-SL7xovj6E2Loq9N0UgV6ynjMLW4urTFY/L/Fprhvz13Xc5vjzkjZjI1QHKq31200+6PSD8PyU6MqrtCTJj6/XA==",
+      "requires": {
+        "pug-code-gen": "2.0.0",
+        "pug-filters": "2.1.5",
+        "pug-lexer": "3.1.0",
+        "pug-linker": "3.0.3",
+        "pug-load": "2.0.9",
+        "pug-parser": "4.0.0",
+        "pug-runtime": "2.0.3",
+        "pug-strip-comments": "1.0.2"
+      }
+    },
+    "pug-attrs": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/pug-attrs/-/pug-attrs-2.0.2.tgz",
+      "integrity": "sha1-i+KyIlVo/6ddG4Zpgr/59BEa/8s=",
+      "requires": {
+        "constantinople": "3.1.0",
+        "js-stringify": "1.0.2",
+        "pug-runtime": "2.0.3"
+      }
+    },
+    "pug-code-gen": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/pug-code-gen/-/pug-code-gen-2.0.0.tgz",
+      "integrity": "sha512-E4oiJT+Jn5tyEIloj8dIJQognbiNNp0i0cAJmYtQTFS0soJ917nlIuFtqVss3IXMEyQKUew3F4gIkBpn18KbVg==",
+      "requires": {
+        "constantinople": "3.1.0",
+        "doctypes": "1.1.0",
+        "js-stringify": "1.0.2",
+        "pug-attrs": "2.0.2",
+        "pug-error": "1.3.2",
+        "pug-runtime": "2.0.3",
+        "void-elements": "2.0.1",
+        "with": "5.1.1"
+      }
+    },
+    "pug-error": {
+      "version": "1.3.2",
+      "resolved": "https://registry.npmjs.org/pug-error/-/pug-error-1.3.2.tgz",
+      "integrity": "sha1-U659nSm7A89WRJOgJhCfVMR/XyY="
+    },
+    "pug-filters": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/pug-filters/-/pug-filters-2.1.5.tgz",
+      "integrity": "sha512-xkw71KtrC4sxleKiq+cUlQzsiLn8pM5+vCgkChW2E6oNOzaqTSIBKIQ5cl4oheuDzvJYCTSYzRaVinMUrV4YLQ==",
+      "requires": {
+        "clean-css": "3.4.28",
+        "constantinople": "3.1.0",
+        "jstransformer": "1.0.0",
+        "pug-error": "1.3.2",
+        "pug-walk": "1.1.5",
+        "resolve": "1.5.0",
+        "uglify-js": "2.8.29"
+      }
+    },
+    "pug-lexer": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/pug-lexer/-/pug-lexer-3.1.0.tgz",
+      "integrity": "sha1-/QhzdtSmdbT1n4/vQiiDQ06VgaI=",
+      "requires": {
+        "character-parser": "2.2.0",
+        "is-expression": "3.0.0",
+        "pug-error": "1.3.2"
+      },
+      "dependencies": {
+        "acorn": {
+          "version": "4.0.13",
+          "resolved": "https://registry.npmjs.org/acorn/-/acorn-4.0.13.tgz",
+          "integrity": "sha1-EFSVrlNh1pe9GVyCUZLhrX8lN4c="
+        },
+        "is-expression": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/is-expression/-/is-expression-3.0.0.tgz",
+          "integrity": "sha1-Oayqa+f9HzRx3ELHQW5hwkMXrJ8=",
+          "requires": {
+            "acorn": "4.0.13",
+            "object-assign": "4.1.1"
+          }
+        }
+      }
+    },
+    "pug-linker": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/pug-linker/-/pug-linker-3.0.3.tgz",
+      "integrity": "sha512-DCKczglCXOzJ1lr4xUj/lVHYvS+lGmR2+KTCjZjtIpdwaN7lNOoX2SW6KFX5X4ElvW+6ThwB+acSUg08UJFN5A==",
+      "requires": {
+        "pug-error": "1.3.2",
+        "pug-walk": "1.1.5"
+      }
+    },
+    "pug-load": {
+      "version": "2.0.9",
+      "resolved": "https://registry.npmjs.org/pug-load/-/pug-load-2.0.9.tgz",
+      "integrity": "sha512-BDdZOCru4mg+1MiZwRQZh25+NTRo/R6/qArrdWIf308rHtWA5N9kpoUskRe4H6FslaQujC+DigH9LqlBA4gf6Q==",
+      "requires": {
+        "object-assign": "4.1.1",
+        "pug-walk": "1.1.5"
+      }
+    },
+    "pug-parser": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/pug-parser/-/pug-parser-4.0.0.tgz",
+      "integrity": "sha512-ocEUFPdLG9awwFj0sqi1uiZLNvfoodCMULZzkRqILryIWc/UUlDlxqrKhKjAIIGPX/1SNsvxy63+ayEGocGhQg==",
+      "requires": {
+        "pug-error": "1.3.2",
+        "token-stream": "0.0.1"
+      }
+    },
+    "pug-runtime": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/pug-runtime/-/pug-runtime-2.0.3.tgz",
+      "integrity": "sha1-mBYmB7D86eJU1CfzOYelrucWi9o="
+    },
+    "pug-strip-comments": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/pug-strip-comments/-/pug-strip-comments-1.0.2.tgz",
+      "integrity": "sha1-0xOvoBvMN0mA4TmeI+vy65vchRM=",
+      "requires": {
+        "pug-error": "1.3.2"
+      }
+    },
+    "pug-walk": {
+      "version": "1.1.5",
+      "resolved": "https://registry.npmjs.org/pug-walk/-/pug-walk-1.1.5.tgz",
+      "integrity": "sha512-rJlH1lXerCIAtImXBze3dtKq/ykZMA4rpO9FnPcIgsWcxZLOvd8zltaoeOVFyBSSqCkhhJWbEbTMga8UxWUUSA=="
+    },
+    "qs": {
+      "version": "6.5.1",
+      "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz",
+      "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A=="
+    },
+    "randomatic": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/randomatic/-/randomatic-1.1.7.tgz",
+      "integrity": "sha512-D5JUjPyJbaJDkuAazpVnSfVkLlpeO3wDlPROTMLGKG1zMFNFRgrciKo1ltz/AzNTkqE0HzDx655QOL51N06how==",
+      "dev": true,
+      "requires": {
+        "is-number": "3.0.0",
+        "kind-of": "4.0.0"
+      },
+      "dependencies": {
+        "kind-of": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz",
+          "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=",
+          "dev": true,
+          "requires": {
+            "is-buffer": "1.1.6"
+          }
+        }
+      }
+    },
+    "range-parser": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz",
+      "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4="
+    },
+    "raw-body": {
+      "version": "2.3.2",
+      "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.2.tgz",
+      "integrity": "sha1-vNYMd9Prk83gBQKVw/N5OJvIj4k=",
+      "requires": {
+        "bytes": "3.0.0",
+        "http-errors": "1.6.2",
+        "iconv-lite": "0.4.19",
+        "unpipe": "1.0.0"
+      }
+    },
+    "rc": {
+      "version": "1.2.2",
+      "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.2.tgz",
+      "integrity": "sha1-2M6ctX6NZNnHut2YdsfDTL48cHc=",
+      "dev": true,
+      "requires": {
+        "deep-extend": "0.4.2",
+        "ini": "1.3.5",
+        "minimist": "1.2.0",
+        "strip-json-comments": "2.0.1"
+      }
+    },
+    "readable-stream": {
+      "version": "2.3.3",
+      "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.3.tgz",
+      "integrity": "sha512-m+qzzcn7KUxEmd1gMbchF+Y2eIUbieUaxkWtptyHywrX0rE8QEYqPC07Vuy4Wm32/xE16NcdBctb8S0Xe/5IeQ==",
+      "requires": {
+        "core-util-is": "1.0.2",
+        "inherits": "2.0.3",
+        "isarray": "1.0.0",
+        "process-nextick-args": "1.0.7",
+        "safe-buffer": "5.1.1",
+        "string_decoder": "1.0.3",
+        "util-deprecate": "1.0.2"
+      }
+    },
+    "readdirp": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.1.0.tgz",
+      "integrity": "sha1-TtCtBg3zBzMAxIRANz9y0cxkLXg=",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "4.1.11",
+        "minimatch": "3.0.4",
+        "readable-stream": "2.3.3",
+        "set-immediate-shim": "1.0.1"
+      },
+      "dependencies": {
+        "graceful-fs": {
+          "version": "4.1.11",
+          "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz",
+          "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=",
+          "dev": true
+        },
+        "minimatch": {
+          "version": "3.0.4",
+          "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
+          "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
+          "dev": true,
+          "requires": {
+            "brace-expansion": "1.1.8"
+          }
+        }
+      }
+    },
+    "rechoir": {
+      "version": "0.6.2",
+      "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz",
+      "integrity": "sha1-hSBLVNuoLVdC4oyWdW70OvUOM4Q=",
+      "dev": true,
+      "requires": {
+        "resolve": "1.5.0"
+      }
+    },
+    "regex-cache": {
+      "version": "0.4.4",
+      "resolved": "https://registry.npmjs.org/regex-cache/-/regex-cache-0.4.4.tgz",
+      "integrity": "sha512-nVIZwtCjkC9YgvWkpM55B5rBhBYRZhAaJbgcFYXXsHnbZ9UZI9nnVWYZpBlCqv9ho2eZryPnWrZGsOdPwVWXWQ==",
+      "dev": true,
+      "requires": {
+        "is-equal-shallow": "0.1.3"
+      }
+    },
+    "regex-not": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.0.tgz",
+      "integrity": "sha1-Qvg+OXcWIt+CawKvF2Ul1qXxV/k=",
+      "dev": true,
+      "requires": {
+        "extend-shallow": "2.0.1"
+      }
+    },
+    "registry-auth-token": {
+      "version": "3.3.1",
+      "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-3.3.1.tgz",
+      "integrity": "sha1-+w0yie4Nmtosu1KvXf5mywcNMAY=",
+      "dev": true,
+      "requires": {
+        "rc": "1.2.2",
+        "safe-buffer": "5.1.1"
+      }
+    },
+    "registry-url": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-3.1.0.tgz",
+      "integrity": "sha1-PU74cPc93h138M+aOBQyRE4XSUI=",
+      "dev": true,
+      "requires": {
+        "rc": "1.2.2"
+      }
+    },
+    "remove-trailing-separator": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz",
+      "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=",
+      "dev": true
+    },
+    "repeat-element": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.2.tgz",
+      "integrity": "sha1-7wiaF40Ug7quTZPrmLT55OEdmQo=",
+      "dev": true
+    },
+    "repeat-string": {
+      "version": "1.6.1",
+      "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz",
+      "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc="
+    },
+    "replace-ext": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/replace-ext/-/replace-ext-0.0.1.tgz",
+      "integrity": "sha1-KbvZIHinOfC8zitO5B6DeVNSKSQ=",
+      "dev": true
+    },
+    "require_optional": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz",
+      "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==",
+      "requires": {
+        "resolve-from": "2.0.0",
+        "semver": "5.4.1"
+      }
+    },
+    "resolve": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.5.0.tgz",
+      "integrity": "sha512-hgoSGrc3pjzAPHNBg+KnFcK2HwlHTs/YrAGUr6qgTVUZmXv1UEXXl0bZNBKMA9fud6lRYFdPGz0xXxycPzmmiw==",
+      "requires": {
+        "path-parse": "1.0.5"
+      }
+    },
+    "resolve-dir": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/resolve-dir/-/resolve-dir-1.0.1.tgz",
+      "integrity": "sha1-eaQGRMNivoLybv/nOcm7U4IEb0M=",
+      "dev": true,
+      "requires": {
+        "expand-tilde": "2.0.2",
+        "global-modules": "1.0.0"
+      }
+    },
+    "resolve-from": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz",
+      "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c="
+    },
+    "resolve-url": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz",
+      "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=",
+      "dev": true
+    },
+    "right-align": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/right-align/-/right-align-0.1.3.tgz",
+      "integrity": "sha1-YTObci/mo1FWiSENJOFMlhSGE+8=",
+      "requires": {
+        "align-text": "0.1.4"
+      }
+    },
+    "safe-buffer": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz",
+      "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg=="
+    },
+    "sanitize-html": {
+      "version": "1.16.3",
+      "resolved": "https://registry.npmjs.org/sanitize-html/-/sanitize-html-1.16.3.tgz",
+      "integrity": "sha512-XpAJGnkMfNM7AzXLRw225blBB/pE4dM4jzRn98g4r88cfxwN6g+5IsRmCAh/gbhYGm6u6i97zsatMOM7Lr8wyw==",
+      "requires": {
+        "htmlparser2": "3.9.2",
+        "lodash.clonedeep": "4.5.0",
+        "lodash.escaperegexp": "4.1.2",
+        "lodash.mergewith": "4.6.0",
+        "postcss": "6.0.15",
+        "srcset": "1.0.0",
+        "xtend": "4.0.1"
+      }
+    },
+    "semver": {
+      "version": "5.4.1",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-5.4.1.tgz",
+      "integrity": "sha512-WfG/X9+oATh81XtllIo/I8gOiY9EXRdv1cQdyykeXK17YcUW3EXUAi2To4pcH6nZtJPr7ZOpM5OMyWJZm+8Rsg=="
+    },
+    "semver-diff": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-2.1.0.tgz",
+      "integrity": "sha1-S7uEN8jTfksM8aaP1ybsbWRdbTY=",
+      "dev": true,
+      "requires": {
+        "semver": "5.4.1"
+      }
+    },
+    "send": {
+      "version": "0.16.1",
+      "resolved": "https://registry.npmjs.org/send/-/send-0.16.1.tgz",
+      "integrity": "sha512-ElCLJdJIKPk6ux/Hocwhk7NFHpI3pVm/IZOYWqUmoxcgeyM+MpxHHKhb8QmlJDX1pU6WrgaHBkVNm73Sv7uc2A==",
+      "requires": {
+        "debug": "2.6.9",
+        "depd": "1.1.1",
+        "destroy": "1.0.4",
+        "encodeurl": "1.0.1",
+        "escape-html": "1.0.3",
+        "etag": "1.8.1",
+        "fresh": "0.5.2",
+        "http-errors": "1.6.2",
+        "mime": "1.4.1",
+        "ms": "2.0.0",
+        "on-finished": "2.3.0",
+        "range-parser": "1.2.0",
+        "statuses": "1.3.1"
+      }
+    },
+    "sequencify": {
+      "version": "0.0.7",
+      "resolved": "https://registry.npmjs.org/sequencify/-/sequencify-0.0.7.tgz",
+      "integrity": "sha1-kM/xnQLgcCf9dn9erT57ldHnOAw=",
+      "dev": true
+    },
+    "serve-favicon": {
+      "version": "2.4.5",
+      "resolved": "https://registry.npmjs.org/serve-favicon/-/serve-favicon-2.4.5.tgz",
+      "integrity": "sha512-s7F8h2NrslMkG50KxvlGdj+ApSwaLex0vexuJ9iFf3GLTIp1ph/l1qZvRe9T9TJEYZgmq72ZwJ2VYiAEtChknw==",
+      "requires": {
+        "etag": "1.8.1",
+        "fresh": "0.5.2",
+        "ms": "2.0.0",
+        "parseurl": "1.3.2",
+        "safe-buffer": "5.1.1"
+      }
+    },
+    "serve-static": {
+      "version": "1.13.1",
+      "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.1.tgz",
+      "integrity": "sha512-hSMUZrsPa/I09VYFJwa627JJkNs0NrfL1Uzuup+GqHfToR2KcsXFymXSV90hoyw3M+msjFuQly+YzIH/q0MGlQ==",
+      "requires": {
+        "encodeurl": "1.0.1",
+        "escape-html": "1.0.3",
+        "parseurl": "1.3.2",
+        "send": "0.16.1"
+      }
+    },
+    "set-getter": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/set-getter/-/set-getter-0.1.0.tgz",
+      "integrity": "sha1-12nBgsnVpR9AkUXy+6guXoboA3Y=",
+      "dev": true,
+      "requires": {
+        "to-object-path": "0.3.0"
+      }
+    },
+    "set-immediate-shim": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz",
+      "integrity": "sha1-SysbJ+uAip+NzEgaWOXlb1mfP2E=",
+      "dev": true
+    },
+    "set-value": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.0.tgz",
+      "integrity": "sha512-hw0yxk9GT/Hr5yJEYnHNKYXkIA8mVJgd9ditYZCe16ZczcaELYYcfvaXesNACk2O8O0nTiPQcQhGUQj8JLzeeg==",
+      "dev": true,
+      "requires": {
+        "extend-shallow": "2.0.1",
+        "is-extendable": "0.1.1",
+        "is-plain-object": "2.0.4",
+        "split-string": "3.1.0"
+      }
+    },
+    "setprototypeof": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz",
+      "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ=="
+    },
+    "shebang-command": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz",
+      "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=",
+      "dev": true,
+      "requires": {
+        "shebang-regex": "1.0.0"
+      }
+    },
+    "shebang-regex": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz",
+      "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=",
+      "dev": true
+    },
+    "sigmund": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/sigmund/-/sigmund-1.0.1.tgz",
+      "integrity": "sha1-P/IfGYytIXX587eBhT/ZTQ0ZtZA=",
+      "dev": true
+    },
+    "signal-exit": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz",
+      "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=",
+      "dev": true
+    },
+    "snapdragon": {
+      "version": "0.8.1",
+      "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.1.tgz",
+      "integrity": "sha1-4StUh/re0+PeoKyR6UAL91tAE3A=",
+      "dev": true,
+      "requires": {
+        "base": "0.11.2",
+        "debug": "2.6.9",
+        "define-property": "0.2.5",
+        "extend-shallow": "2.0.1",
+        "map-cache": "0.2.2",
+        "source-map": "0.5.7",
+        "source-map-resolve": "0.5.1",
+        "use": "2.0.2"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          }
+        },
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        },
+        "source-map": {
+          "version": "0.5.7",
+          "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz",
+          "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=",
+          "dev": true
+        }
+      }
+    },
+    "snapdragon-node": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz",
+      "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==",
+      "dev": true,
+      "requires": {
+        "define-property": "1.0.0",
+        "isobject": "3.0.1",
+        "snapdragon-util": "3.0.1"
+      }
+    },
+    "snapdragon-util": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz",
+      "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==",
+      "dev": true,
+      "requires": {
+        "kind-of": "3.2.2"
+      }
+    },
+    "socket.io": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-2.0.4.tgz",
+      "integrity": "sha1-waRZDO/4fs8TxyZS8Eb3FrKeYBQ=",
+      "requires": {
+        "debug": "2.6.9",
+        "engine.io": "3.1.4",
+        "socket.io-adapter": "1.1.1",
+        "socket.io-client": "2.0.4",
+        "socket.io-parser": "3.1.2"
+      }
+    },
+    "socket.io-adapter": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-1.1.1.tgz",
+      "integrity": "sha1-KoBeihTWNyEk3ZFZrUUC+MsH8Gs="
+    },
+    "socket.io-client": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-2.0.4.tgz",
+      "integrity": "sha1-CRilUkBtxeVAs4Dc2Xr8SmQzL44=",
+      "requires": {
+        "backo2": "1.0.2",
+        "base64-arraybuffer": "0.1.5",
+        "component-bind": "1.0.0",
+        "component-emitter": "1.2.1",
+        "debug": "2.6.9",
+        "engine.io-client": "3.1.4",
+        "has-cors": "1.1.0",
+        "indexof": "0.0.1",
+        "object-component": "0.0.3",
+        "parseqs": "0.0.5",
+        "parseuri": "0.0.5",
+        "socket.io-parser": "3.1.2",
+        "to-array": "0.1.4"
+      }
+    },
+    "socket.io-parser": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.1.2.tgz",
+      "integrity": "sha1-28IoIVH8T6675Aru3Ady66YZ9/I=",
+      "requires": {
+        "component-emitter": "1.2.1",
+        "debug": "2.6.9",
+        "has-binary2": "1.0.2",
+        "isarray": "2.0.1"
+      },
+      "dependencies": {
+        "isarray": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz",
+          "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4="
+        }
+      }
+    },
+    "source-map": {
+      "version": "0.4.4",
+      "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.4.4.tgz",
+      "integrity": "sha1-66T12pwNyZneaAMti092FzZSA2s=",
+      "requires": {
+        "amdefine": "1.0.1"
+      }
+    },
+    "source-map-resolve": {
+      "version": "0.5.1",
+      "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.1.tgz",
+      "integrity": "sha512-0KW2wvzfxm8NCTb30z0LMNyPqWCdDGE2viwzUaucqJdkTRXtZiSY3I+2A6nVAjmdOy0I4gU8DwnVVGsk9jvP2A==",
+      "dev": true,
+      "requires": {
+        "atob": "2.0.3",
+        "decode-uri-component": "0.2.0",
+        "resolve-url": "0.2.1",
+        "source-map-url": "0.4.0",
+        "urix": "0.1.0"
+      }
+    },
+    "source-map-url": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz",
+      "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=",
+      "dev": true
+    },
+    "sparkles": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/sparkles/-/sparkles-1.0.0.tgz",
+      "integrity": "sha1-Gsu/tZJDbRC76PeFt8xvgoFQEsM=",
+      "dev": true
+    },
+    "split": {
+      "version": "0.3.3",
+      "resolved": "https://registry.npmjs.org/split/-/split-0.3.3.tgz",
+      "integrity": "sha1-zQ7qXmOiEd//frDwkcQTPi0N0o8=",
+      "dev": true,
+      "requires": {
+        "through": "2.3.8"
+      }
+    },
+    "split-string": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz",
+      "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==",
+      "dev": true,
+      "requires": {
+        "extend-shallow": "3.0.2"
+      },
+      "dependencies": {
+        "extend-shallow": {
+          "version": "3.0.2",
+          "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz",
+          "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=",
+          "dev": true,
+          "requires": {
+            "assign-symbols": "1.0.0",
+            "is-extendable": "1.0.1"
+          }
+        },
+        "is-extendable": {
+          "version": "1.0.1",
+          "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz",
+          "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==",
+          "dev": true,
+          "requires": {
+            "is-plain-object": "2.0.4"
+          }
+        }
+      }
+    },
+    "srcset": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/srcset/-/srcset-1.0.0.tgz",
+      "integrity": "sha1-pWad4StC87HV6D7QPHEEb8SPQe8=",
+      "requires": {
+        "array-uniq": "1.0.3",
+        "number-is-nan": "1.0.1"
+      }
+    },
+    "static-extend": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz",
+      "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=",
+      "dev": true,
+      "requires": {
+        "define-property": "0.2.5",
+        "object-copy": "0.1.0"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          }
+        },
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        }
+      }
+    },
+    "statuses": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.3.1.tgz",
+      "integrity": "sha1-+vUbnrdKrvOzrPStX2Gr8ky3uT4="
+    },
+    "stream-combiner": {
+      "version": "0.0.4",
+      "resolved": "https://registry.npmjs.org/stream-combiner/-/stream-combiner-0.0.4.tgz",
+      "integrity": "sha1-TV5DPBhSYd3mI8o/RMWGvPXErRQ=",
+      "dev": true,
+      "requires": {
+        "duplexer": "0.1.1"
+      }
+    },
+    "stream-consume": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/stream-consume/-/stream-consume-0.1.0.tgz",
+      "integrity": "sha1-pB6tGm1ggc63n2WwYZAbbY89HQ8=",
+      "dev": true
+    },
+    "string-width": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
+      "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
+      "dev": true,
+      "requires": {
+        "is-fullwidth-code-point": "2.0.0",
+        "strip-ansi": "4.0.0"
+      },
+      "dependencies": {
+        "ansi-regex": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz",
+          "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=",
+          "dev": true
+        },
+        "strip-ansi": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
+          "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=",
+          "dev": true,
+          "requires": {
+            "ansi-regex": "3.0.0"
+          }
+        }
+      }
+    },
+    "string_decoder": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz",
+      "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==",
+      "requires": {
+        "safe-buffer": "5.1.1"
+      }
+    },
+    "strip-ansi": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+      "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
+      "dev": true,
+      "requires": {
+        "ansi-regex": "2.1.1"
+      }
+    },
+    "strip-bom": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-1.0.0.tgz",
+      "integrity": "sha1-hbiGLzhEtabV7IRnqTWYFzo295Q=",
+      "dev": true,
+      "requires": {
+        "first-chunk-stream": "1.0.0",
+        "is-utf8": "0.2.1"
+      }
+    },
+    "strip-eof": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz",
+      "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=",
+      "dev": true
+    },
+    "strip-json-comments": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
+      "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
+      "dev": true
+    },
+    "supports-color": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz",
+      "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=",
+      "dev": true
+    },
+    "term-size": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/term-size/-/term-size-1.2.0.tgz",
+      "integrity": "sha1-RYuDiH8oj8Vtb/+/rSYuJmOO+mk=",
+      "dev": true,
+      "requires": {
+        "execa": "0.7.0"
+      }
+    },
+    "through": {
+      "version": "2.3.8",
+      "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
+      "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
+      "dev": true
+    },
+    "through2": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.3.tgz",
+      "integrity": "sha1-AARWmzfHx0ujnEPzzteNGtlBQL4=",
+      "dev": true,
+      "requires": {
+        "readable-stream": "2.3.3",
+        "xtend": "4.0.1"
+      }
+    },
+    "thunky": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/thunky/-/thunky-1.0.2.tgz",
+      "integrity": "sha1-qGLgGOP7HqLsP85dVWBc9X8kc3E="
+    },
+    "tildify": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/tildify/-/tildify-1.2.0.tgz",
+      "integrity": "sha1-3OwD9V3Km3qj5bBPIYF+tW5jWIo=",
+      "dev": true,
+      "requires": {
+        "os-homedir": "1.0.2"
+      }
+    },
+    "time-stamp": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/time-stamp/-/time-stamp-1.1.0.tgz",
+      "integrity": "sha1-dkpaEa9QVhkhsTPztE5hhofg9cM=",
+      "dev": true
+    },
+    "timed-out": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz",
+      "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=",
+      "dev": true
+    },
+    "to-array": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/to-array/-/to-array-0.1.4.tgz",
+      "integrity": "sha1-F+bBH3PdTz10zaek/zI46a2b+JA="
+    },
+    "to-mongodb-core": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/to-mongodb-core/-/to-mongodb-core-2.0.0.tgz",
+      "integrity": "sha1-NZbsdhOsmtO5ioncua77pWnNJ+s="
+    },
+    "to-object-path": {
+      "version": "0.3.0",
+      "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz",
+      "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=",
+      "dev": true,
+      "requires": {
+        "kind-of": "3.2.2"
+      }
+    },
+    "to-regex": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.1.tgz",
+      "integrity": "sha1-FTWL7kosg712N3uh3ASdDxiDeq4=",
+      "dev": true,
+      "requires": {
+        "define-property": "0.2.5",
+        "extend-shallow": "2.0.1",
+        "regex-not": "1.0.0"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          }
+        },
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        }
+      }
+    },
+    "to-regex-range": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz",
+      "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=",
+      "dev": true,
+      "requires": {
+        "is-number": "3.0.0",
+        "repeat-string": "1.6.1"
+      }
+    },
+    "token-stream": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/token-stream/-/token-stream-0.0.1.tgz",
+      "integrity": "sha1-zu78cXp2xDFvEm0LnbqlXX598Bo="
+    },
+    "touch": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz",
+      "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==",
+      "dev": true,
+      "requires": {
+        "nopt": "1.0.10"
+      }
+    },
+    "type-is": {
+      "version": "1.6.15",
+      "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.15.tgz",
+      "integrity": "sha1-yrEPtJCeRByChC6v4a1kbIGARBA=",
+      "requires": {
+        "media-typer": "0.3.0",
+        "mime-types": "2.1.17"
+      }
+    },
+    "uglify-js": {
+      "version": "2.8.29",
+      "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.8.29.tgz",
+      "integrity": "sha1-KcVzMUgFe7Th913zW3qcty5qWd0=",
+      "requires": {
+        "source-map": "0.5.7",
+        "uglify-to-browserify": "1.0.2",
+        "yargs": "3.10.0"
+      },
+      "dependencies": {
+        "source-map": {
+          "version": "0.5.7",
+          "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz",
+          "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w="
+        }
+      }
+    },
+    "uglify-to-browserify": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz",
+      "integrity": "sha1-bgkk1r2mta/jSeOabWMoUKD4grc=",
+      "optional": true
+    },
+    "ultron": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.1.tgz",
+      "integrity": "sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og=="
+    },
+    "unc-path-regex": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/unc-path-regex/-/unc-path-regex-0.1.2.tgz",
+      "integrity": "sha1-5z3T17DXxe2G+6xrCufYxqadUPo=",
+      "dev": true
+    },
+    "undefsafe": {
+      "version": "0.0.3",
+      "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-0.0.3.tgz",
+      "integrity": "sha1-7Mo6A+VrmvFzhbqsgSrIO5lKli8=",
+      "dev": true
+    },
+    "underscore": {
+      "version": "1.8.3",
+      "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.8.3.tgz",
+      "integrity": "sha1-Tz+1OxBuYJf8+ctBCfKl6b36UCI="
+    },
+    "union-value": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.0.tgz",
+      "integrity": "sha1-XHHDTLW61dzr4+oM0IIHulqhrqQ=",
+      "dev": true,
+      "requires": {
+        "arr-union": "3.1.0",
+        "get-value": "2.0.6",
+        "is-extendable": "0.1.1",
+        "set-value": "0.4.3"
+      },
+      "dependencies": {
+        "set-value": {
+          "version": "0.4.3",
+          "resolved": "https://registry.npmjs.org/set-value/-/set-value-0.4.3.tgz",
+          "integrity": "sha1-fbCPnT0i3H945Trzw79GZuzfzPE=",
+          "dev": true,
+          "requires": {
+            "extend-shallow": "2.0.1",
+            "is-extendable": "0.1.1",
+            "is-plain-object": "2.0.4",
+            "to-object-path": "0.3.0"
+          }
+        }
+      }
+    },
+    "unique-stream": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/unique-stream/-/unique-stream-1.0.0.tgz",
+      "integrity": "sha1-1ZpKdUJ0R9mqbJHnAmP40mpLEEs=",
+      "dev": true
+    },
+    "unique-string": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-1.0.0.tgz",
+      "integrity": "sha1-nhBXzKhRq7kzmPizOuGHuZyuwRo=",
+      "dev": true,
+      "requires": {
+        "crypto-random-string": "1.0.0"
+      }
+    },
+    "unpipe": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
+      "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
+    },
+    "unset-value": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz",
+      "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=",
+      "dev": true,
+      "requires": {
+        "has-value": "0.3.1",
+        "isobject": "3.0.1"
+      },
+      "dependencies": {
+        "has-value": {
+          "version": "0.3.1",
+          "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz",
+          "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=",
+          "dev": true,
+          "requires": {
+            "get-value": "2.0.6",
+            "has-values": "0.1.4",
+            "isobject": "2.1.0"
+          },
+          "dependencies": {
+            "isobject": {
+              "version": "2.1.0",
+              "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz",
+              "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=",
+              "dev": true,
+              "requires": {
+                "isarray": "1.0.0"
+              }
+            }
+          }
+        },
+        "has-values": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz",
+          "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=",
+          "dev": true
+        }
+      }
+    },
+    "unzip-response": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz",
+      "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=",
+      "dev": true
+    },
+    "update-notifier": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-2.3.0.tgz",
+      "integrity": "sha1-TognpruRUUCrCTVZ1wFOPruDdFE=",
+      "dev": true,
+      "requires": {
+        "boxen": "1.3.0",
+        "chalk": "2.3.0",
+        "configstore": "3.1.1",
+        "import-lazy": "2.1.0",
+        "is-installed-globally": "0.1.0",
+        "is-npm": "1.0.0",
+        "latest-version": "3.1.0",
+        "semver-diff": "2.1.0",
+        "xdg-basedir": "3.0.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.0",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.0.tgz",
+          "integrity": "sha512-NnSOmMEYtVR2JVMIGTzynRkkaxtiq1xnFBcdQD/DnNCYPoEPsVJhM98BDyaoNOQIi7p4okdi3E27eN7GQbsUug==",
+          "dev": true,
+          "requires": {
+            "color-convert": "1.9.1"
+          }
+        },
+        "chalk": {
+          "version": "2.3.0",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.0.tgz",
+          "integrity": "sha512-Az5zJR2CBujap2rqXGaJKaPHyJ0IrUimvYNX+ncCy8PJP4ltOGTrHUIo097ZaL2zMeKYpiCdqDvS6zdrTFok3Q==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "3.2.0",
+            "escape-string-regexp": "1.0.5",
+            "supports-color": "4.5.0"
+          }
+        },
+        "supports-color": {
+          "version": "4.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-4.5.0.tgz",
+          "integrity": "sha1-vnoN5ITexcXN34s9WRJQRJEvY1s=",
+          "dev": true,
+          "requires": {
+            "has-flag": "2.0.0"
+          }
+        }
+      }
+    },
+    "urix": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz",
+      "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=",
+      "dev": true
+    },
+    "url-parse-lax": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz",
+      "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=",
+      "dev": true,
+      "requires": {
+        "prepend-http": "1.0.4"
+      }
+    },
+    "use": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/use/-/use-2.0.2.tgz",
+      "integrity": "sha1-riig1y+TvyJCKhii43mZMRLeyOg=",
+      "dev": true,
+      "requires": {
+        "define-property": "0.2.5",
+        "isobject": "3.0.1",
+        "lazy-cache": "2.0.2"
+      },
+      "dependencies": {
+        "define-property": {
+          "version": "0.2.5",
+          "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz",
+          "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=",
+          "dev": true,
+          "requires": {
+            "is-descriptor": "0.1.6"
+          }
+        },
+        "is-accessor-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz",
+          "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-data-descriptor": {
+          "version": "0.1.4",
+          "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz",
+          "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=",
+          "dev": true,
+          "requires": {
+            "kind-of": "3.2.2"
+          },
+          "dependencies": {
+            "kind-of": {
+              "version": "3.2.2",
+              "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz",
+              "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=",
+              "dev": true,
+              "requires": {
+                "is-buffer": "1.1.6"
+              }
+            }
+          }
+        },
+        "is-descriptor": {
+          "version": "0.1.6",
+          "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz",
+          "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==",
+          "dev": true,
+          "requires": {
+            "is-accessor-descriptor": "0.1.6",
+            "is-data-descriptor": "0.1.4",
+            "kind-of": "5.1.0"
+          }
+        },
+        "kind-of": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz",
+          "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==",
+          "dev": true
+        },
+        "lazy-cache": {
+          "version": "2.0.2",
+          "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-2.0.2.tgz",
+          "integrity": "sha1-uRkKT5EzVGlIQIWfio9whNiCImQ=",
+          "dev": true,
+          "requires": {
+            "set-getter": "0.1.0"
+          }
+        }
+      }
+    },
+    "user-home": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/user-home/-/user-home-1.1.1.tgz",
+      "integrity": "sha1-K1viOjK2Onyd640PKNSFcko98ZA=",
+      "dev": true
+    },
+    "util-deprecate": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+      "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
+    },
+    "utils-merge": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
+      "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
+    },
+    "uws": {
+      "version": "0.14.5",
+      "resolved": "https://registry.npmjs.org/uws/-/uws-0.14.5.tgz",
+      "integrity": "sha1-Z6rzPEaypYel9mZtAPdpEyjxSdw=",
+      "optional": true
+    },
+    "v8flags": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/v8flags/-/v8flags-2.1.1.tgz",
+      "integrity": "sha1-qrGh+jDUX4jdMhFIh1rALAtV5bQ=",
+      "dev": true,
+      "requires": {
+        "user-home": "1.1.1"
+      }
+    },
+    "vary": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
+      "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
+    },
+    "vinyl": {
+      "version": "0.5.3",
+      "resolved": "https://registry.npmjs.org/vinyl/-/vinyl-0.5.3.tgz",
+      "integrity": "sha1-sEVbOPxeDPMNQyUTLkYZcMIJHN4=",
+      "dev": true,
+      "requires": {
+        "clone": "1.0.3",
+        "clone-stats": "0.0.1",
+        "replace-ext": "0.0.1"
+      }
+    },
+    "vinyl-fs": {
+      "version": "0.3.14",
+      "resolved": "https://registry.npmjs.org/vinyl-fs/-/vinyl-fs-0.3.14.tgz",
+      "integrity": "sha1-mmhRzhysHBzqX+hsCTHWIMLPqeY=",
+      "dev": true,
+      "requires": {
+        "defaults": "1.0.3",
+        "glob-stream": "3.1.18",
+        "glob-watcher": "0.0.6",
+        "graceful-fs": "3.0.11",
+        "mkdirp": "0.5.1",
+        "strip-bom": "1.0.0",
+        "through2": "0.6.5",
+        "vinyl": "0.4.6"
+      },
+      "dependencies": {
+        "clone": {
+          "version": "0.2.0",
+          "resolved": "https://registry.npmjs.org/clone/-/clone-0.2.0.tgz",
+          "integrity": "sha1-xhJqkK1Pctv1rNskPMN3JP6T/B8=",
+          "dev": true
+        },
+        "isarray": {
+          "version": "0.0.1",
+          "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz",
+          "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=",
+          "dev": true
+        },
+        "readable-stream": {
+          "version": "1.0.34",
+          "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz",
+          "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=",
+          "dev": true,
+          "requires": {
+            "core-util-is": "1.0.2",
+            "inherits": "2.0.3",
+            "isarray": "0.0.1",
+            "string_decoder": "0.10.31"
+          }
+        },
+        "string_decoder": {
+          "version": "0.10.31",
+          "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz",
+          "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=",
+          "dev": true
+        },
+        "through2": {
+          "version": "0.6.5",
+          "resolved": "https://registry.npmjs.org/through2/-/through2-0.6.5.tgz",
+          "integrity": "sha1-QaucZ7KdVyCQcUEOHXp6lozTrUg=",
+          "dev": true,
+          "requires": {
+            "readable-stream": "1.0.34",
+            "xtend": "4.0.1"
+          }
+        },
+        "vinyl": {
+          "version": "0.4.6",
+          "resolved": "https://registry.npmjs.org/vinyl/-/vinyl-0.4.6.tgz",
+          "integrity": "sha1-LzVsh6VQolVGHza76ypbqL94SEc=",
+          "dev": true,
+          "requires": {
+            "clone": "0.2.0",
+            "clone-stats": "0.0.1"
+          }
+        }
+      }
+    },
+    "void-elements": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz",
+      "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w="
+    },
+    "which": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/which/-/which-1.3.0.tgz",
+      "integrity": "sha512-xcJpopdamTuY5duC/KnTTNBraPK54YwpenP4lzxU8H91GudWpFv38u0CKjclE1Wi2EH2EDz5LRcHcKbCIzqGyg==",
+      "dev": true,
+      "requires": {
+        "isexe": "2.0.0"
+      }
+    },
+    "widest-line": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-2.0.0.tgz",
+      "integrity": "sha1-AUKk6KJD+IgsAjOqDgKBqnYVInM=",
+      "dev": true,
+      "requires": {
+        "string-width": "2.1.1"
+      }
+    },
+    "window-size": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.1.0.tgz",
+      "integrity": "sha1-VDjNLqk7IC76Ohn+iIeu58lPnJ0="
+    },
+    "with": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/with/-/with-5.1.1.tgz",
+      "integrity": "sha1-+k2qktrzLE6pTtRTyB8EaGtXXf4=",
+      "requires": {
+        "acorn": "3.3.0",
+        "acorn-globals": "3.1.0"
+      }
+    },
+    "wordwrap": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.2.tgz",
+      "integrity": "sha1-t5Zpu0LstAn4PVg8rVLKF+qhZD8="
+    },
+    "wrappy": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+      "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8="
+    },
+    "write-file-atomic": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.3.0.tgz",
+      "integrity": "sha512-xuPeK4OdjWqtfi59ylvVL0Yn35SF3zgcAcv7rBPFHVaEapaDr4GdGgm3j7ckTwH9wHL7fGmgfAnb0+THrHb8tA==",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "4.1.11",
+        "imurmurhash": "0.1.4",
+        "signal-exit": "3.0.2"
+      },
+      "dependencies": {
+        "graceful-fs": {
+          "version": "4.1.11",
+          "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz",
+          "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=",
+          "dev": true
+        }
+      }
+    },
+    "ws": {
+      "version": "3.3.3",
+      "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz",
+      "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==",
+      "requires": {
+        "async-limiter": "1.0.0",
+        "safe-buffer": "5.1.1",
+        "ultron": "1.1.1"
+      }
+    },
+    "xdg-basedir": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-3.0.0.tgz",
+      "integrity": "sha1-SWsswQnsqNus/i3HK2A8F8WHCtQ=",
+      "dev": true
+    },
+    "xmlhttprequest-ssl": {
+      "version": "1.5.4",
+      "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.4.tgz",
+      "integrity": "sha1-BPVgkVcks4kIhxXMDteBPpZ3v1c="
+    },
+    "xtend": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz",
+      "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68="
+    },
+    "yallist": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz",
+      "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=",
+      "dev": true
+    },
+    "yargs": {
+      "version": "3.10.0",
+      "resolved": "https://registry.npmjs.org/yargs/-/yargs-3.10.0.tgz",
+      "integrity": "sha1-9+572FfdfB0tOMDnTvvWgdFDH9E=",
+      "requires": {
+        "camelcase": "1.2.1",
+        "cliui": "2.1.0",
+        "decamelize": "1.2.0",
+        "window-size": "0.1.0"
+      }
+    },
+    "yeast": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/yeast/-/yeast-0.1.2.tgz",
+      "integrity": "sha1-AI4G2AlDIMNy28L47XagymyKxBk="
+    }
+  }
+}
diff --git a/package.json b/package.json
new file mode 100644 (file)
index 0000000..01e7eb6
--- /dev/null
@@ -0,0 +1,34 @@
+{
+  "name": "qcm",
+  "version": "0.1.0",
+  "description": "QCM tool for exams",
+  "main": "bin/www",
+  "scripts": {
+    "start": "gulp server",
+    "test": "echo \"Error: no test specified\" && exit 1"
+  },
+  "repository": {
+    "type": "git",
+    "url": "git@auder.net:qcm"
+  },
+  "author": "Benjamin Auder",
+  "license": "ISC",
+  "dependencies": {
+    "body-parser": "^1.18.2",
+    "bson-objectid": "^1.2.2",
+    "cookie-parser": "^1.4.3",
+    "express": "^4.16.2",
+    "mongojs": "^2.4.1",
+    "morgan": "^1.9.0",
+    "node-cron": "^1.2.1",
+    "pug": "^2.0.0-rc.4",
+    "sanitize-html": "^1.16.3",
+    "serve-favicon": "^2.4.5",
+    "socket.io": "^2.0.4",
+    "underscore": "^1.8.3"
+  },
+  "devDependencies": {
+    "gulp": "^3.9.1",
+    "gulp-nodemon": "^2.2.1"
+  }
+}
diff --git a/public/favicon/LICENSE b/public/favicon/LICENSE
new file mode 100644 (file)
index 0000000..a76e70e
--- /dev/null
@@ -0,0 +1,3 @@
+This favicon is generated from an iconscout icon, thus falling under the
+iconscout regular license https://iconscout.com/legal#licenses
+You can use it freely on your website, but (of course) not sell it
diff --git a/public/favicon/android-chrome-192x192.png b/public/favicon/android-chrome-192x192.png
new file mode 100644 (file)
index 0000000..8ec0b88
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat bcd031181c55c67ab221bbfbd562f49a2c902137                 3725
diff --git a/public/favicon/android-chrome-512x512.png b/public/favicon/android-chrome-512x512.png
new file mode 100644 (file)
index 0000000..b14c7da
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat 6fddf2e88ddc418b4b1e6c436dbaaf780bf2497b                 8261
diff --git a/public/favicon/apple-touch-icon.png b/public/favicon/apple-touch-icon.png
new file mode 100644 (file)
index 0000000..3b5e1b4
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat 6c5687f8cea00ad423475093f0854bce975baf8c                 3434
diff --git a/public/favicon/browserconfig.xml b/public/favicon/browserconfig.xml
new file mode 100644 (file)
index 0000000..28c1987
--- /dev/null
@@ -0,0 +1,9 @@
+<?xml version="1.0" encoding="utf-8"?>
+<browserconfig>
+    <msapplication>
+        <tile>
+            <square150x150logo src="/favicon/mstile-150x150.png"/>
+            <TileColor>#00aba9</TileColor>
+        </tile>
+    </msapplication>
+</browserconfig>
diff --git a/public/favicon/favicon-16x16.png b/public/favicon/favicon-16x16.png
new file mode 100644 (file)
index 0000000..56089aa
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat b45cbbb34d4b06f9b079197b17f98196ddc5dc92                  638
diff --git a/public/favicon/favicon-32x32.png b/public/favicon/favicon-32x32.png
new file mode 100644 (file)
index 0000000..623c953
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat 5d804a72b035e65b9eec08b5dfb0a5dca5bf4edf                 1017
diff --git a/public/favicon/favicon.ico b/public/favicon/favicon.ico
new file mode 100644 (file)
index 0000000..ca71ccc
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat 40682600e89f78aca3ceabe47b3d057b4abd7396                15086
diff --git a/public/favicon/manifest.json b/public/favicon/manifest.json
new file mode 100644 (file)
index 0000000..9e4641d
--- /dev/null
@@ -0,0 +1,18 @@
+{
+    "name": "",
+    "icons": [
+        {
+            "src": "/favicon/android-chrome-192x192.png",
+            "sizes": "192x192",
+            "type": "image/png"
+        },
+        {
+            "src": "/favicon/android-chrome-512x512.png",
+            "sizes": "512x512",
+            "type": "image/png"
+        }
+    ],
+    "theme_color": "#ffffff",
+    "background_color": "#ffffff",
+    "display": "standalone"
+}
\ No newline at end of file
diff --git a/public/favicon/mstile-150x150.png b/public/favicon/mstile-150x150.png
new file mode 100644 (file)
index 0000000..fdccdda
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat 92f6a68050f68b6785ca67e80907474a5e1acaf2                 2609
diff --git a/public/favicon/safari-pinned-tab.svg b/public/favicon/safari-pinned-tab.svg
new file mode 100644 (file)
index 0000000..43b36f9
--- /dev/null
@@ -0,0 +1 @@
+<svg version="1" xmlns="http://www.w3.org/2000/svg" width="682.667" height="682.667" viewBox="0 0 512.000000 512.000000"><path d="M480.5 18.7c-88.8 51.9-164 93.9-204.5 114.1-30.7 15.4-37.5 18.1-41.6 16.7-4.9-1.8-6.9-5.7-7.2-14.9-.5-9.8 1.6-18.5 7.8-32.7 2.1-5.1 3.8-9.4 3.7-9.5-.6-.4-188.1 158.5-194.9 165.2-21.1 20.9-35.5 48.2-41.5 78.9-2.7 13.8-2.4 41.6.6 55.5 8.7 40.7 32.4 75.3 66.7 97.2 15 9.6 36.3 17.6 55.5 20.9 15.7 2.7 39.8 2.2 55.1-1 31.1-6.7 60.1-23.6 80.2-46.6 6.4-7.4 186.6-246.9 186.6-248 0-.3-5.1-.5-11.3-.5-14.3 0-20.7-1.7-26.1-7.1-3.7-3.7-3.8-4.2-3.3-8.7 1.9-16.3 35.7-80.7 89.4-170.5 8.5-14.2 15.3-26 15.1-26.2-.2-.2-13.8 7.5-30.3 17.2zm-30.8 44c-33.1 56.4-60.3 110.7-64.7 129.2-3.3 14.1 3.6 29.2 16.9 36.8 2.5 1.5 4.8 2.8 4.9 2.9.2.1-35.5 48.1-79.4 106.6-84.8 112.9-89.6 118.9-105.6 129.6-13.9 9.4-28.4 15.7-44.8 19.7-8.9 2.2-34.3 3.1-45.3 1.6-18-2.4-37.1-9.6-52.4-19.7-10.9-7.1-29-25.1-35.8-35.4-13.1-20.2-20.2-41.7-21.2-65-1.4-33.4 8.1-62.1 29.1-87.5 8.1-9.9 4.4-6.7 137.6-118.8l17.4-14.7 2.3 4.8c5.2 10.9 14.6 18 25.4 18.9 8.4.8 18.3-2.6 44.7-15.7 31.6-15.6 68.5-35.7 137.7-75.2C434.7 70.5 449.6 62 449.8 62c.2 0 .1.3-.1.7z"/><path d="M140 256.1c-49 4.1-88.5 41-96.5 90-6.6 40.9 11.6 82.6 46.1 105.5 30.3 20.1 66.8 23.8 100.8 10 19.3-7.8 39.9-25.4 50.3-43.1 25.7-43.4 19.3-96.3-15.8-131.4-22.5-22.6-52.9-33.7-84.9-31zm33.5 24.9c44.9 14.4 69.7 58.9 58 104-7.4 28.5-31.2 52.3-60.4 60.2-11.2 3.1-32.6 3.1-43.3.1-15.6-4.5-27.7-11.6-38.6-22.5-10.9-10.9-18-23-22.4-38.6-3-10.2-3.1-31.2-.4-41.7 8-30.9 31.9-54.8 62.4-62.5 11-2.8 34.5-2.3 44.7 1z"/></svg>
\ No newline at end of file
diff --git a/public/javascripts/assessment.js b/public/javascripts/assessment.js
new file mode 100644 (file)
index 0000000..1db80a0
--- /dev/null
@@ -0,0 +1,415 @@
+let socket = null; //monitor answers in real time
+
+function checkWindowSize()
+{
+       if (assessment.mode == "secure")
+       {
+               // NOTE: temporarily accept smartphone (security hole: pretend being a smartphone on desktop browser...)
+               if (navigator.userAgent.match(/(iPhone|iPod|iPad|Android|BlackBerry)/))
+                       return true;
+               let test = () => {
+                       return window.innerWidth < screen.width || window.innerHeight < screen.height;
+               };
+               const returnVal = test;
+               while (!test)
+                       alert("Please enter fullscreen mode (F11)");
+               return returnVal;
+       }
+       return true;
+};
+
+function libsRefresh()
+{
+       $("#statements").find("code[class^=language-]").each( (i,elem) => {
+               Prism.highlightElement(elem);
+       });
+       MathJax.Hub.Queue(["Typeset",MathJax.Hub,"statements"]);
+};
+
+// TODO: if display == "all", les envois devraient être non définitifs (possibilité de corriger)
+// Et, blur sur une (sous-)question devrait envoyer la version courante de la sous-question
+
+let V = new Vue({
+       el: "#assessment",
+       data: {
+               assessment: assessment,
+               inputs: [ ], //student's answers
+               student: { }, //filled later
+               // Stage 0: unauthenticated (number),
+               //       1: authenticated (got a name, unvalidated)
+               //       2: locked: password set, exam started
+               //       3: completed
+               //       4: show answers
+               stage: assessment.mode != "open" ? 0 : 1,
+               remainingTime: 0, //global, in seconds
+       },
+       components: {
+               "statements": {
+                       props: ['assessment','inputs','student','stage'],
+                       data: function() {
+                               return {
+                                       index: 0, //current question index in assessment.indices
+                               };
+                       },
+                       mounted: function() {
+                               if (assessment.mode != "secure")
+                                       return;
+                               $("#warning").modal({
+                                       complete: () => {
+                                               this.stage = 2;
+                                               this.resumeAssessment();
+                                       },
+                               });
+                               window.addEventListener("blur", () => {
+                                       if (this.stage == 2)
+                                               this.showWarning();
+                               }, false);
+                               window.addEventListener("resize", e => {
+                                       if (this.stage == 2 && !checkWindowSize())
+                                               this.showWarning();
+                               }, false);
+                               //socket.on("disconnect", () => { }); //TODO: notify monitor (highlight red)
+                       },
+                       updated: function() {
+                               libsRefresh();
+                       },
+                       // TODO: general render function for nested exercises
+                       // TODO: with answer if stage==4 : class "wrong" if ticked AND stage==4 AND received answers
+                       // class "right" if stage == 4 AND received answers (background-color: red / green)
+                       // There should be a questions navigator below, or next (visible if display=='all')
+                       // Full questions tree is rendered, but some parts hidden depending on display settings
+                       render(h) {
+                               let self = this;
+                               let questions = assessment.questions.map( (q,i) => {
+                                       let questionContent = [ ];
+                                       questionContent.push(
+                                               h(
+                                                       "div",
+                                                       {
+                                                               "class": {
+                                                                       "wording": true,
+                                                               },
+                                                               domProps: {
+                                                                       innerHTML: q.wording,
+                                                               },
+                                                       }
+                                               )
+                                       );
+                                       let optionsOrder = _.range(q.options.length);
+                                       if (!q.fixed)
+                                               optionsOrder = _.shuffle(optionsOrder);
+                                       let optionList = [ ];
+                                       optionsOrder.forEach( idx => {
+                                               let option = [ ];
+                                               option.push(
+                                                       h(
+                                                               "input",
+                                                               {
+                                                                       domProps: {
+                                                                               checked: this.inputs[i][idx],
+                                                                       },
+                                                                       attrs: {
+                                                                               id: this.inputId(i,idx),
+                                                                               type: "checkbox",
+                                                                       },
+                                                                       on: {
+                                                                               change: e => { this.inputs[i][idx] = e.target.checked; },
+                                                                       },
+                                                               },
+                                                       )
+                                               );
+                                               option.push(
+                                                       h(
+                                                               "label",
+                                                               {
+                                                                       domProps: {
+                                                                               innerHTML: q.options[idx],
+                                                                       },
+                                                                       attrs: {
+                                                                               "for": this.inputId(i,idx),
+                                                                       },
+                                                               }
+                                                       )
+                                               );
+                                               optionList.push(
+                                                       h(
+                                                               "div",
+                                                               {
+                                                                       "class": {
+                                                                               option: true,
+                                                                               choiceCorrect: this.stage == 4 && assessment.questions[i].answer.includes(idx),
+                                                                               choiceWrong: this.stage == 4 && this.inputs[i][idx] && !assessment.questions[i].answer.includes(idx),
+                                                                       },
+                                                               },
+                                                               option
+                                                       )
+                                               );
+                                       });
+                                       questionContent.push(
+                                               h(
+                                                       "div",
+                                                       {
+                                                               "class": {
+                                                                       optionList: true,
+                                                               },
+                                                       },
+                                                       optionList
+                                               )
+                                       );
+                                       return h(
+                                               "div",
+                                               {
+                                                       "class": {
+                                                               "question": true,
+                                                               "hide": this.stage == 2 && assessment.display == 'one' && assessment.indices[this.index] != i,
+                                                       },
+                                               },
+                                               questionContent
+                                       );
+                               });
+                               if (this.stage == 2)
+                               {
+                                       // TODO: one button per question
+                                       questions.unshift(
+                                               h(
+                                                       "button",
+                                                       {
+                                                               "class": {
+                                                                       "waves-effect": true,
+                                                                       "waves-light": true,
+                                                                       "btn": true,
+                                                               },
+                                                               on: {
+                                                                       click: () => this.sendAnswer(assessment.indices[this.index]),
+                                                               },
+                                                       },
+                                                       "Send"
+                                               )
+                                       );
+                               }
+                               return h(
+                                       "div",
+                                       {
+                                               attrs: {
+                                                       id: "statements",
+                                               },
+                                       },
+                                       questions
+                               );
+                       },
+                       methods: {
+                               // HELPERS:
+                               inputId: function(i,j) {
+                                       return "q" + i + "_" + "input" + j;
+                               },
+                               showWarning: function(action) {
+                                       this.sendAnswer(assessment.indices[this.index]);
+                                       this.stage = 32; //fictive stage to hide all elements
+                                       $("#warning").modal('open');
+                               },
+                               // stage 2
+                               sendAnswer: function(realIndex) {
+                                       if (this.index == assessment.questions.length - 1)
+                                               this.$emit("gameover");
+                                       else
+                                               this.index++;
+                                       if (assessment.mode == "open")
+                                               return; //only local
+                                       let answerData = {
+                                               aid: assessment._id,
+                                               answer: JSON.stringify({
+                                                       index:realIndex.toString(),
+                                                       input:this.inputs[realIndex]
+                                                               .map( (tf,i) => { return {val:tf,idx:i}; } )
+                                                               .filter( item => { return item.val; })
+                                                               .map( item => { return item.idx; })
+                                               }),
+                                               number: this.student.number,
+                                               password: this.student.password,
+                                       };
+                                       $.ajax("/send/answer", {
+                                               method: "GET",
+                                               data: answerData,
+                                               dataType: "json",
+                                               success: ret => {
+                                                       if (!!ret.errmsg)
+                                                               return alert(ret.errmsg);
+                                                       //socket.emit(message.newAnswer, answer);
+                                               },
+                                       });
+                               },
+                               // stage 2 after blur or resize
+                               resumeAssessment: function() {
+                                       checkWindowSize();
+                               },
+                       },
+               },
+       },
+       mounted: function() {
+               window.addEventListener("keydown", e => {
+                       // If F12 or ctrl+shift (ways to access devtools)
+                       if (e.keyCode == 123 || (e.ctrlKey && e.shiftKey))
+                               e.preventDefault();
+               }, false);
+               // Devtools detect based on https://jsfiddle.net/ebhjxfwv/4/
+               let div = document.createElement('div');
+               let devtoolsLoop = setInterval(
+                       () => {
+                               if (assessment.mode != "open")
+                               {
+                                       console.log(div);
+                                       console.clear();
+                               }
+                       },
+                       1000
+               );
+               Object.defineProperty(div, "id", {
+                       get: () => {
+                               clearInterval(devtoolsLoop);
+                               if (assessment.mode != "open")
+                               {
+                                       if (this.stage == 2)
+                                               this.endAssessment();
+                                       document.location.href = "/nodevtools";
+                               }
+                       }
+               });
+       },
+       computed: {
+               countdown: function() {
+                       let seconds = this.remainingTime % 60;
+                       let minutes = Math.floor(this.remainingTime / 60);
+                       return this.padWithZero(minutes) + ":" + this.padWithZero(seconds);
+               },
+       },
+       methods: {
+               // HELPERS:
+               padWithZero: function(x) {
+                       if (x < 10)
+                               return "0" + x;
+                       return x;
+               },
+               // stage 0 --> 1
+               getStudent: function(cb) {
+                       $.ajax("/get/student", {
+                               method: "GET",
+                               data: {
+                                       number: this.student.number,
+                                       cid: assessment.cid,
+                               },
+                               dataType: "json",
+                               success: s => {
+                                       if (!!s.errmsg)
+                                               return alert(s.errmsg);
+                                       this.stage = 1;
+                                       this.student = s.student;
+                                       Vue.nextTick( () => { Materialize.updateTextFields(); });
+                                       if (!!cb)
+                                               cb();
+                               },
+                       });
+               },
+               // stage 1 --> 0
+               cancelStudent: function() {
+                       this.stage = 0;
+               },
+               // stage 1 --> 2 (get all questions, set password)
+               startAssessment: function() {
+                       checkWindowSize();
+                       let initializeStage2 = questions => {
+                               $("#leftButton, #rightButton").hide();
+                               if (assessment.time > 0)
+                               {
+                                       this.remainingTime = assessment.time * 60;
+                                       this.runTimer();
+                               }
+                               // Initialize structured answer(s) based on questions type and nesting (TODO: more general)
+                               if (!!questions)
+                                       assessment.questions = questions;
+                               for (let q of assessment.questions)
+                                       this.inputs.push( _(q.options.length).times( _.constant(false) ) );
+                               assessment.indices = assessment.fixed
+                                       ? _.range(assessment.questions.length)
+                                       : _.shuffle( _.range(assessment.questions.length) );
+                               this.stage = 2;
+                               Vue.nextTick( () => { libsRefresh(); });
+                       };
+                       if (assessment.mode == "open")
+                               return initializeStage2();
+                       $.ajax("/start/assessment", {
+                               method: "GET",
+                               data: {
+                                       number: this.student.number,
+                                       aid: assessment._id
+                               },
+                               dataType: "json",
+                               success: s => {
+                                       if (!!s.errmsg)
+                                               return alert(s.errmsg);
+                                       this.student.password = s.password;
+                                       // Got password: students answers locked to this page until potential teacher
+                                       // action (power failure, computer down, ...)
+                                       // TODO: password also exchanged by sockets to check identity
+                                       //socket = io.connect("/" + assessment.name, {
+                                       //      query: "number=" + this.student.number + "&password=" + this.password
+                                       //});
+                                       //socket.on(message.allAnswers, this.setAnswers);
+                                       initializeStage2(s.questions);
+                               },
+                       });
+               },
+               // stage 2
+               runTimer: function() {
+                       if (assessment.time <= 0)
+                               return;
+                       let self = this;
+                       setInterval( function() {
+                               self.remainingTime--;
+                               if (self.remainingTime <= 0 || self.stage >= 4)
+                                       self.endAssessment();
+                                       clearInterval(this);
+                       }, 1000);
+               },
+               // stage 2 after disconnect (socket)
+               resumeAssessment: function() {
+                       // UNIMPLEMENTED
+                       // TODO: get stored answers (papers[number cookie]), inject (inputs), set index+indices
+               },
+               // stage 2 --> 3 (or 4)
+               // from a message by statements component
+               endAssessment: function() {
+                       // If time over or cheating: set endTime, destroy password
+                       $("#leftButton, #rightButton").show();
+                       //this.sendAnswer(...); //TODO: for each non-answered (and non-empty!) index (yet)
+                       if (assessment.mode != "open")
+                       {
+                               $.ajax("/end/assessment", {
+                                       method: "GET",
+                                       data: {
+                                               aid: assessment._id,
+                                               number: this.student.number,
+                                               password: this.student.password,
+                                       },
+                                       dataType: "json",
+                                       success: ret => {
+                                               if (!!ret.errmsg)
+                                                       return alert(ret.errmsg);
+                                               assessment.conclusion = ret.conclusion;
+                                               this.stage = 3;
+                                               delete this.student["password"]; //unable to send new answers now
+                                               //socket.disconnect();
+                                               //socket = null;
+                                       },
+                               });
+                       }
+                       else
+                               this.stage = 4;
+               },
+               // stage 3 --> 4 (on socket message "feedback")
+               setAnswers: function(answers) {
+                       for (let i=0; i<answers.length; i++)
+                               assessment.questions[i].answer = answers[i];
+                       this.stage = 4;
+               },
+       },
+});
diff --git a/public/javascripts/course.js b/public/javascripts/course.js
new file mode 100644 (file)
index 0000000..94e1b84
--- /dev/null
@@ -0,0 +1,360 @@
+// TODO: YAML format for questions, parsed from text (nested questions)
+// Then yaml parsed to json --> array of indexed questions
+// Use open mode for question banks: add setting "nbQuestions" to show nbQuestions
+// at random among active questions
+
+window.onload = function() {
+
+       V = new Vue({
+               el: '#course',
+               data: {
+                       display: "assessments", //or "students", or "grades" (admin mode)
+                       course: course,
+                       mode: "view", //or "edit" (some assessment)
+                       // assessment data:
+                       monitorPwd: "",
+                       newAssessment: { name: "" },
+                       assessmentArray: assessmentArray,
+                       assessmentIndex: 0, //current edited assessment index
+                       assessment: { }, //copy of assessment at editing index in array
+                       assessmentText: "", //questions in an assessment, in text format
+                       // grades data:
+                       settings: {
+                               totalPoints: 20,
+                               halfPoints: false,
+                               zeroSum: false,
+                       },
+                       group: 1, //for detailed grades tables
+                       grades: { }, //computed
+               },
+               mounted: function() {
+                       $('.modal').each( (i,elem) => {
+                               if (elem.id != "assessmentEdit")
+                                       $(elem).modal();
+                       });
+                       $('ul.tabs').tabs();
+                       $('#assessmentEdit').modal({
+                               complete: () => {
+                                       this.parseAssessment();
+                                       Vue.nextTick( () => {
+                                               $("#questionList").find("code[class^=language-]").each( (i,elem) => {
+                                                       Prism.highlightElement(elem);
+                                               });
+                                               MathJax.Hub.Queue(["Typeset",MathJax.Hub,"questionList"]);
+                                       });
+                               },
+                       });
+               },
+               methods: {
+                       // GENERAL:
+                       toggleDisplay: function(area) {
+                               if (this.display == area)
+                                       this.display = "";
+                               else
+                                       this.display = area;
+                       },
+                       studentList: function(group) {
+                               return this.course.students
+                                       .filter( s => { return group==0 || s.group == group; })
+                                       .map( s => { return Object.assign({}, s); }) //not altering initial array
+                                       .sort( (a,b) => {
+                                               let res = a.name.localeCompare(b.name);
+                                               if (res == 0)
+                                                       res += a.forename.localeCompare(b.forename);
+                                               return res;
+                                       });
+                       },
+                       // STUDENTS:
+                       uploadTrigger: function() {
+                               $("#upload").click();
+                       },
+                       upload: function(e) {
+                               let file = (e.target.files || e.dataTransfer.files)[0];
+                               Papa.parse(file, {
+                                       header: true,
+                                       skipEmptyLines: true,
+                                       complete: (results,file) => {
+                                               let students = [ ];
+                                               // Post-process: add group/number if missing
+                                               let number = 1;
+                                               results.data.forEach( d => {
+                                                       if (!d.group)
+                                                               d.group = 1;
+                                                       if (!d.number)
+                                                               d.number = number++;
+                                                       if (typeof d.number !== "string")
+                                                               d.number = d.number.toString();
+                                                       students.push(d);
+                                               });
+                                               $.ajax("/import/students", {
+                                                       method: "POST",
+                                                       data: {
+                                                               cid: this.course._id,
+                                                               students: JSON.stringify(students),
+                                                       },
+                                                       dataType: "json",
+                                                       success: res => {
+                                                               if (!res.errmsg)
+                                                                       this.course.students = students;
+                                                               else
+                                                                       alert(res.errmsg);
+                                                       },
+                                               });
+                                       },
+                               });
+                       },
+                       // ASSESSMENT:
+                       addAssessment: function() {
+                               if (!admin)
+                                       return;
+                               // modal, fill code and description
+                               let error = Validator.checkObject(this.newAssessment, "Assessment");
+                               if (!!error)
+                                       return alert(error);
+                               else
+                                       $('#newAssessment').modal('close');
+                               $.ajax("/add/assessment",
+                                       {
+                                               method: "GET",
+                                               data: {
+                                                       name: this.newAssessment.name,
+                                                       cid: course._id,
+                                               },
+                                               dataType: "json",
+                                               success: res => {
+                                                       if (!res.errmsg)
+                                                       {
+                                                               this.newAssessment["name"] = "";
+                                                               this.assessmentArray.push(res);
+                                                       }
+                                                       else
+                                                               alert(res.errmsg);
+                                               },
+                                       }
+                               );
+                       },
+                       materialOpenModal: function(id) {
+                               $("#" + id).modal("open");
+                               Materialize.updateTextFields(); //textareas, time field...
+                       },
+                       updateAssessment: function() {
+                               $.ajax("/update/assessment", {
+                                       method: "POST",
+                                       data: {assessment: JSON.stringify(this.assessment)},
+                                       dataType: "json",
+                                       success: res => {
+                                               if (!res.errmsg)
+                                               {
+                                                       this.assessmentArray[this.assessmentIndex] = this.assessment;
+                                                       this.mode = "view";
+                                               }
+                                               else
+                                                       alert(res.errmsg);
+                                       },
+                               });
+                       },
+                       deleteAssessment: function(assessment) {
+                               if (!admin)
+                                       return;
+                               if (confirm("Delete assessment '" + assessment.name + "' ?"))
+                               {
+                                       $.ajax("/remove/assessment",
+                                               {
+                                                       method: "GET",
+                                                       data: { qid: this.assessment._id },
+                                                       dataType: "json",
+                                                       success: res => {
+                                                               if (!res.errmsg)
+                                                                       this.assessmentArray.splice( this.assessmentArray.findIndex( item => {
+                                                                               return item._id == assessment._id;
+                                                                       }), 1 );
+                                                               else
+                                                                       alert(res.errmsg);
+                                                       },
+                                               }
+                                       );
+                               }
+                       },
+                       toggleState: function(questionIndex) {
+                               // add or remove from activeSet of current assessment
+                               let activeIndex = this.assessment.activeSet.findIndex( item => { return item == questionIndex; });
+                               if (activeIndex >= 0)
+                                       this.assessment.activeSet.splice(activeIndex, 1);
+                               else
+                                       this.assessment.activeSet.push(questionIndex);
+                       },
+                       setAssessmentText: function() {
+                               let txt = "";
+                               this.assessment.questions.forEach( q => {
+                                       txt += q.wording + "\n";
+                                       q.options.forEach( (o,i) => {
+                                               let symbol = q.answer.includes(i) ? "+" : "-";
+                                               txt += symbol + " " + o + "\n";
+                                       });
+                                       txt += "\n"; //separate questions by new line
+                               });
+                               this.assessmentText = txt;
+                       },
+                       parseAssessment: function() {
+                               let questions = [ ];
+                               let lines = this.assessmentText.split("\n").map( L => { return L.trim(); })
+                               lines.push(""); //easier parsing
+                               let emptyQuestion = () => {
+                                       return {
+                                               wording: "",
+                                               options: [ ],
+                                               answer: [ ],
+                                               active: true, //default
+                                       };
+                               };
+                               let q = emptyQuestion();
+                               lines.forEach( L => {
+                                       if (L.length > 0)
+                                       {
+                                               if (['+','-'].includes(L.charAt(0)))
+                                               {
+                                                       if (L.charAt(0) == '+')
+                                                               q.answer.push(q.options.length);
+                                                       q.options.push(L.slice(1).trim());
+                                               }
+                                               else if (L.charAt(0) == '*')
+                                               {
+                                                       // TODO: read current + next lines into q.answer (HTML, 1-elem array)
+                                               }
+                                               else
+                                                       q.wording += L + " "; //space required at line breaks, generally
+                                       }
+                                       else
+                                       {
+                                               // Flush current question (if any)
+                                               if (q.wording.length > 0)
+                                               {
+                                                       questions.push(q);
+                                                       q = emptyQuestion();
+                                               }
+                                       }
+                               });
+                               this.assessment.questions = questions;
+                       },
+                       actionAssessment: function(index) {
+                               if (admin)
+                               {
+                                       // Edit screen
+                                       this.assessmentIndex = index;
+                                       this.assessment = $.extend(true, {}, this.assessmentArray[index]);
+                                       this.setAssessmentText();
+                                       this.mode = "edit";
+                                       Vue.nextTick( () => {
+                                               $("#questionList").find("code[class^=language-]").each( (i,elem) => {
+                                                       Prism.highlightElement(elem);
+                                               });
+                                               MathJax.Hub.Queue(["Typeset",MathJax.Hub,"questionList"]);
+                                       });
+                               }
+                               else //external user: show assessment
+                                       this.redirect(this.assessmentArray[index].name);
+                       },
+                       redirect: function(assessmentName) {
+                               document.location.href = "/" + initials + "/" + course.code + "/" + assessmentName;
+                       },
+                       setPassword: function() {
+                               let hashPwd = Sha1.Compute(this.monitorPwd);
+                               let error = Validator.checkObject({password:hashPwd}, "Course");
+                               if (error.length > 0)
+                                       return alert(error);
+                               $.ajax("/set/password",
+                                       {
+                                               method: "GET",
+                                               data: {
+                                                       cid: this.course._id,
+                                                       pwd: hashPwd,
+                                               },
+                                               dataType: "json",
+                                               success: res => {
+                                                       if (!res.errmsg)
+                                                               alert("Password saved!");
+                                                       else
+                                                               alert(res.errmsg);
+                                               },
+                                       }
+                               );
+                       },
+                       // NOTE: artifact required for Vue v-model to behave well
+                       checkBoxFixedId: function(i) {
+                               return "questionFixed" + i;
+                       },
+                       checkBoxActiveId: function(i) {
+                               return "questionActive" + i;
+                       },
+                       // GRADES:
+                       gradeSettings: function() {
+                               $("#gradeSettings").modal("open");
+                               Materialize.updateTextFields(); //total points field in grade settings overlap
+                       },
+                       download: function() {
+                               // Download (all) grades as a CSV file
+                               let data = [ ];
+                               this.studentList(0).forEach( s => {
+                                       let finalGrade = 0.;
+                                       let gradesCount = 0;
+                                       if (!!this.grades[s.number])
+                                       {
+                                               Object.keys(this.grades[s.number]).forEach( assessmentName => {
+                                                       s[assessmentName] = this.grades[s.number][assessmentName];
+                                                       if (_.isNumeric(s[assessmentName]) && !isNaN(s[assessmentName]))
+                                                       {
+                                                               finalGrade += s[assessmentName];
+                                                               gradesCount++;
+                                                       }
+                                                       if (gradesCount >= 1)
+                                                               finalGrade /= gradesCount;
+                                                       s["final"] = finalGrade; //TODO: forbid "final" as assessment name
+                                               });
+                                       }
+                                       data.push(s); //number,forename,name,group,assessName1...assessNameN,final
+                               });
+                               let csv = Papa.unparse(data, {
+                                       quotes: true,
+                                       header: true,
+                               });
+                               let downloadAnchor = $("#download");
+                               downloadAnchor.attr("download", this.course.code + "_results.csv");
+                               downloadAnchor.attr("href", "data:text/plain;charset=utf-8," + encodeURIComponent(csv));
+                               this.$refs.download.click()
+                               //downloadAnchor.click(); //fails
+                       },
+                       showDetails: function(group) {
+                               this.group = group;
+                               $("#detailedGrades").modal("open");
+                       },
+                       groupList: function() {
+                               let maxGrp = 1;
+                               this.course.students.forEach( s => {
+                                       if (s.group > maxGrp)
+                                               maxGrp = s.group;
+                               });
+                               return _.range(1,maxGrp+1);
+                       },
+                       grade: function(assessmentIndex, studentNumber) {
+                               if (!this.grades[assessmentIndex] || !this.grades[assessmentIndex][studentNumber])
+                                       return ""; //no grade yet
+                               return this.grades[assessmentIndex][studentNumber];
+                       },
+                       groupId: function(group, hash) {
+                               return (!!hash?"#":"") + "group" + group;
+                       },
+                       togglePresence: function(number, index) {
+                               // UNIMPLEMENTED
+                               // TODO: if no grade (thus automatic 0), toggle "exempt" state on student for current exam
+                               // --> automatic update of grades view (just a few number to change)
+                       },
+                       computeGrades: function() {
+                               // UNIMPLEMENTED
+                               // TODO: compute all grades using settings (points, coefficients, bonus/malus...).
+                               // If some questions with free answers (open), display answers and ask teacher action.
+                               // TODO: need a setting for that too (by student, by exercice, by question)
+                       },
+               },
+       });
+
+};
diff --git a/public/javascripts/courseList.js b/public/javascripts/courseList.js
new file mode 100644 (file)
index 0000000..ae1f1d8
--- /dev/null
@@ -0,0 +1,69 @@
+window.onload = function() {
+
+       new Vue({
+               el: '#courseList',
+               data: {
+                       courseArray: courseArray,
+                       newCourse: {
+                               code: "",
+                               description: "",
+                       },
+               },
+               mounted: function() {
+                       $('.modal').modal();
+               },
+               methods: {
+                       redirect: function(code) {
+                               document.location.href = "/" + initials + "/" + code;
+                       },
+                       addCourse: function() {
+                               if (!admin)
+                                       return;
+                               // modal, fill code and description
+                               let error = Validator.checkCode(this.newCourse.code);
+                               if (!!error)
+                                       return alert(error);
+                               else
+                                       $('#newCourse').modal('close');
+                               $.ajax("/add/course",
+                                       {
+                                               method: "GET",
+                                               data: this.newCourse,
+                                               dataType: "json",
+                                               success: res => {
+                                                       if (!res.errmsg)
+                                                       {
+                                                               this.newCourse["code"] = "";
+                                                               this.newCourse["description"] = "";
+                                                               this.courseArray.push(res);
+                                                       }
+                                                       else
+                                                               alert(res.errmsg);
+                                               },
+                                       }
+                               );
+                       },
+                       deleteCourse: function(course) {
+                               if (!admin)
+                                       return;
+                               if (confirm("Delete course '" + course.code + "' ?"))
+                                       $.ajax("/remove/course",
+                                               {
+                                                       method: "GET",
+                                                       data: { cid: course._id },
+                                                       dataType: "json",
+                                                       success: res => {
+                                                               if (!res.errmsg)
+                                                                       this.courseArray.splice( this.courseArray.findIndex( item => {
+                                                                               return item._id == course._id;
+                                                                       }), 1 );
+                                                               else
+                                                                       alert(res.errmsg);
+                                                       },
+                                               }
+                                       );
+                               },
+                       }
+       });
+
+};
diff --git a/public/javascripts/login.js b/public/javascripts/login.js
new file mode 100644 (file)
index 0000000..1bb5b72
--- /dev/null
@@ -0,0 +1,89 @@
+window.onload = function() {
+
+       const messages = {
+               "login": "Go",
+               "register": "Send",
+       };
+
+       const ajaxUrl = {
+               "login": "/sendtoken",
+               "register": "/register",
+       };
+
+       const infos = {
+               "login": "Connection token sent. Check your emails!",
+               "register": "Registration complete! Please check your emails.",
+       };
+
+       const animationDuration = 300; //in milliseconds
+
+       // Basic anti-bot measure: force at least N seconds between arrival on page, and register form validation:
+       const enterTime = Date.now();
+
+       new Vue({
+               el: '#login',
+               data: {
+                       messages: messages,
+                       user: {
+                               forename: "",
+                               name: "",
+                               email: "",
+                       },
+                       stage: "login", //or "register"
+               },
+               mounted: function() {
+                       // https://laracasts.com/discuss/channels/vue/vuejs-set-focus-on-textfield
+                       this.$refs.userEmail.focus();
+               },
+               methods: {
+                       toggleStage: function(stage) {
+                               let $form = $("#form");
+                               $form.fadeOut(animationDuration);
+                               setTimeout( () => {
+                                       this.stage = stage;
+                                       $form.show(0);
+                               }, animationDuration);
+                       },
+                       submit: function() {
+                               if (this.stage=="register")
+                               {
+                                       if (Date.now() - enterTime < 5000)
+                                               return;
+                               }
+                               let error = Validator.checkObject({email: this.user.email}, "User");
+                               if (!error && this.stage == "register")
+                                       error = Validator.checkObject({forename: this.user.forename, name: this.user.name}, "User");
+                               let $dialog = $("#dialog");
+                               show($dialog);
+                               setTimeout(() => {hide($dialog);}, 3000);
+                               if (error.length > 0)
+                                       return showMsg($dialog, "error", error);
+                               showMsg($dialog, "process", "Processing... Please wait");
+                               $.ajax(ajaxUrl[this.stage],
+                                       {
+                                               method: "GET",
+                                               data:
+                                               {
+                                                       email: encodeURIComponent(this.user.email),
+                                                       forename: encodeURIComponent(this.user.forename), //may be unused
+                                                       name: encodeURIComponent(this.user.name), //may be unused
+                                               },
+                                               dataType: "json",
+                                               success: res => {
+                                                       if (!res.errmsg)
+                                                       {
+                                                               this.user["forename"] = "";
+                                                               this.user["name"] = "";
+                                                               this.user["email"] = "";
+                                                               showMsg($dialog, "info", infos[this.stage]);
+                                                       }
+                                                       else
+                                                               showMsg($dialog, "error", res.errmsg);
+                                               },
+                                       }
+                               );
+                       },
+               }
+       });
+
+};
diff --git a/public/javascripts/monitor.js b/public/javascripts/monitor.js
new file mode 100644 (file)
index 0000000..b58461b
--- /dev/null
@@ -0,0 +1,11 @@
+// UNIMPLEMENTED
+
+// TODO: onglets pour chaque groupe + section déroulante questionnaire (chargé avec réponses)
+//   NOM Prenom (par grp, puis alphabétique)
+//   réponse : vert si OK (+ choix), rouge si faux, gris si texte (clic pour voir)
+//   + temps total ?
+//   click sur en-tête de colonne : tri alphabétique, tri décroissant...
+// Affiché si (hash du) mdp du cours est correctement entré
+// Doit reprendre les données en base si refresh (sinon : sockets)
+
+// Also buttons "start exam", "end exam" for logged in teacher
diff --git a/public/javascripts/utils/dialog.js b/public/javascripts/utils/dialog.js
new file mode 100644 (file)
index 0000000..979e757
--- /dev/null
@@ -0,0 +1,30 @@
+function state2col(state)
+{
+       switch (state)
+       {
+               case "process":
+                       return "black";
+               case "error":
+                       return "red";
+               case "info":
+                       return "blue";
+               default: //idle
+                       return "white"; //irrelevant, dialog is hidden
+       }
+}
+
+function show($dialog)
+{
+       $dialog.removeClass("hide");
+}
+
+function hide($dialog)
+{
+       $dialog.addClass("hide");
+}
+
+function showMsg($dialog, state, msg)
+{
+       $dialog.html(msg);
+       $dialog.css("color", state2col(state));
+}
diff --git a/public/javascripts/utils/sha1.js b/public/javascripts/utils/sha1.js
new file mode 100644 (file)
index 0000000..d12006d
--- /dev/null
@@ -0,0 +1,124 @@
+var Sha1 = {};  // SHA-1 namespace
+
+// SHA-1 algorithm as described at http://en.wikipedia.org/wiki/SHA-1
+// The implementation follows http://fr.wikipedia.org/wiki/Sp%C3%A9cifications_SHA-1 (in french).
+// SHA-1 implementation of Chris Veness 2002-2010 [www.movable-type.co.uk] helped a lot for debugging,
+// and for hacks like toHexStr(). See his script at http://www.movable-type.co.uk/scripts/sha1.html
+Sha1.Compute = function(subject)
+{
+       var i, j, tmp, redIndex, a, b, c, d, e;
+
+       // 1) pretreatment
+
+       // note: no check on message length, since the 2^64 boundary is
+       // a lot longer than what would be allowed by HTML/PHP
+
+       // add trailing '1' bit (+ 0's padding) to string
+       subject += String.fromCharCode(0x80);
+
+       // add 8 for two last reserved words to store message length
+       // 8 = 2 x 4, one 32-bits word is 4 characters (bytes) length.
+       var L = subject.length + 8;
+
+       // initialize 512-bits blocks representing the message, each containing 16 32-bits words.
+       // NOTE: one char is 8 bits, so one block in the initial string is 64 chars.
+       var countBlocks = Math.ceil(L / 64);
+       var blocks = new Array(countBlocks);
+       for (i=0; i<countBlocks; i++)
+       {
+               var words = new Array(16);
+               for (j=0; j<16; j++)
+               {
+                       tmp = subject.substr(64 * i + 4 * j, 4);
+                       // note: running off the end of msg is ok because bitwise ops on NaN return 0
+                       words[j] = (1 << 24) * tmp.charCodeAt(0) | (1 << 16) * tmp.charCodeAt(1) | (1 << 8) * tmp.charCodeAt(2) | tmp.charCodeAt(3);
+               }
+               blocks[i] = words;
+       }
+
+       // note: 'subject' in our context will never be of length >= 2^32.
+       // therefore we don't need to fill before-last block.
+       blocks[countBlocks-1][15] = (subject.length-1) * 8;
+
+       // initialize parts of the final hash
+       var h0 = 0x67452301;
+       var h1 = 0xefcdab89;
+       var h2 = 0x98badcfe;
+       var h3 = 0x10325476;
+       var h4 = 0xc3d2e1f0;
+
+       // initialize constants array
+       var k = [0x5a827999,0x6ed9eba1,0x8f1bbcdc,0xca62c1d6];
+
+       // 2) computations
+
+       for (i=0; i<blocks.length; i++)
+       {
+               // initialize w array
+               var w = new Array(80);
+               for (j=0; j<16; j++) w[j] = blocks[i][j];
+               for (j=16; j<80; j++)
+                       w[j] = Sha1.LeftRotate(w[j-3] ^ w[j-8] ^ w[j-14] ^ w[j-16], 1);
+
+               // initialize a,b,c,d,e variables
+               a = h0;
+               b = h1;
+               c = h2;
+               d = h3;
+               e = h4;
+
+               // iterations over a,b,c,d,e
+               for (j=0; j<80; j++)
+               {
+                       // note: '& 0xffffffff' == 'modulo 2^32'.
+                       redIndex = Math.floor(j/20);
+                       tmp = (Sha1.LeftRotate(a, 5) + Sha1.BitOp(b, c, d, redIndex) + e + k[redIndex] + w[j]) & 0xffffffff;
+                       e = d;
+                       d = c;
+                       c = Sha1.LeftRotate(b, 30);
+                       b = a;
+                       a = tmp;
+               }
+
+               // update intermediate hash values
+               h0 = (h0+a) & 0xffffffff;
+               h1 = (h1+b) & 0xffffffff;
+               h2 = (h2+c) & 0xffffffff;
+               h3 = (h3+d) & 0xffffffff;
+               h4 = (h4+e) & 0xffffffff;
+       }
+
+       return Sha1.ToHexStr(h0)+Sha1.ToHexStr(h1)+Sha1.ToHexStr(h2)+Sha1.ToHexStr(h3)+Sha1.ToHexStr(h4);
+}
+
+// auxiliary functions.
+Sha1.BitOp = function(x, y, z, t)
+{
+       if (t == 0) return (x & y) ^ (~x & z);
+       if (t == 1) return x ^ y ^ z;
+       if (t == 2) return (x & y) ^ (x & z) ^ (y & z);
+       if (t == 3) return x ^ y ^ z;
+}
+
+// left rotation (within 32 bits).
+Sha1.LeftRotate = function(x, n)
+{
+       return (x << n) | (x >>> (32 - n));
+}
+
+// [copy-pasted from Chris Veness implementation]
+// Hexadecimal representation of a number
+// (note toString(16) is implementation-dependant, and
+// in IE returns signed numbers when used on full words)
+Sha1.ToHexStr = function(x)
+{
+       var s="";
+       for (var i=7; i>=0; i--)
+       {
+               var v = (x >>> (i*4)) & 0xf;
+               s += v.toString(16);
+       }
+       return s;
+}
+
+try { module.exports = Sha1; } catch (err) {}
diff --git a/public/javascripts/utils/socketMessages.js b/public/javascripts/utils/socketMessages.js
new file mode 100644 (file)
index 0000000..4ffd05b
--- /dev/null
@@ -0,0 +1,10 @@
+// Socket message list, easier life in case of a message renaming
+
+let message = {
+       // send answer (student --> server --> monitor)
+       newAnswer: "new answer",
+       // receive all answers to an exam (server --> student)
+       allAnswers: "all answers",
+};
+
+try { module.exports = message; } catch (err) {} //for server
diff --git a/public/javascripts/utils/validation.js b/public/javascripts/utils/validation.js
new file mode 100644 (file)
index 0000000..560bd5c
--- /dev/null
@@ -0,0 +1,241 @@
+try { var _ = require("underscore"); } catch (err) {} //for server
+
+let Validator = { };
+
+// Cell in assessment.questions array
+Validator.Question = {
+       "index": "section", //"2.2.1", "3.2", "1" ...etc
+       "wording": "string",
+       "options": "stringArray", //only for quiz
+       "fixed": "boolean",
+       "answer": "string", //both this and next are mutually exclusive
+       "choice": "integerArray",
+       "active": "boolean",
+       "points": "number",
+};
+
+Validator.Input = {
+       "index": "section",
+       "input": "stringOrIntegerArray",
+};
+
+// One student response to an exam
+Validator.Paper = {
+       "number": "code",
+       // (array of) strings for open questions, arrays of integers for quizzes:
+       "inputs": Validator.Input,
+       "startTime": "positiveInteger",
+       "endTime": "positiveInteger",
+       "password": "password",
+};
+
+Validator.Assessment = {
+       "_id": "bson",
+       "cid": "bson",
+       "name": "code",
+       "mode": "alphanumeric", //"open" or "exam", but alphanumeric is good enough
+       "active": "boolean",
+       "fixed": "boolean",
+       "display": "alphanumeric", //"one" or "all"
+       "time": "integer",
+       "introduction": "string",
+       "conclusion": "string",
+       "coefficient": "number",
+       "questions": Validator.Question,
+       "papers": Validator.Paper,
+};
+
+Validator.User = {
+       "_id": "bson",
+       "email": "email",
+       "forename": "name",
+       "name": "name",
+       "initials": "unchecked", //not a user input
+       "loginToken": "unchecked",
+       "sessionTokens": "unchecked",
+       "token": "alphanumeric", //exception: for the purpose of user registration
+};
+
+Validator.Student = {
+       "number": "code",
+       "forename": "name",
+       "name": "name",
+       "group": "positiveInteger",
+};
+
+Validator.Course = {
+       "_id": "bson",
+       "uid": "bson",
+       "code": "code",
+       "description": "string",
+       "password": "hash",
+       "students": Validator.Student,
+};
+
+Object.assign(Validator,
+{
+       // Recurse into sub-documents
+       checkObject_aux: function(obj, model)
+       {
+               for (let key of Object.keys(obj))
+               {
+                       if (!model[key])
+                               return "Unknown field";
+                       if (model[key] == "unchecked") //not a user input (ignored)
+                               continue;
+                       if (_.isObject(model[key]))
+                       {
+                               // TODO: next loop seems too heavy... (only a concern if big class import?)
+                               for (let item of obj[key])
+                               {
+                                       let error = Validator.checkObject_aux(item, model[key]);
+                                       if (error.length > 0)
+                                               return error;
+                               }
+                       }
+                       else
+                       {
+                               let error = Validator[ "check_" + model[key] ](obj[key]);
+                               if (error.length > 0)
+                                       return key + ": " + error;
+                       }
+               }
+               return "";
+       },
+
+       // Always check top-level object
+       checkObject: function(obj, name)
+       {
+               return Validator.checkObject_aux(obj, Validator[name]);
+       },
+
+       "check_string": function(arg)
+       {
+               return ""; //strings are unchecked, but sanitized
+       },
+
+       "check_section": function(arg)
+       {
+               if (!_.isString(arg))
+                       return "not a string";
+               if (!/^[0-9.]+$/.test(arg))
+                       return "digits and dot only";
+               return "";
+       },
+
+       "check_stringArray": function(arg)
+       {
+               return !_.isArray(arg) ? "not an array" : "";
+       },
+
+       "check_alphanumeric": function(arg)
+       {
+               return arg.match(/^[\w]{1,32}$/) === null ? "[1,32] alphanumerics" : "";
+       },
+
+       "check_bson": function(arg)
+       {
+               return arg.match(/^[a-z0-9]{24}$/) === null ? "not a BSON id" : "";
+       },
+
+       "check_name": function(arg)
+       {
+               if (!_.isString(arg))
+                       return "not a string";
+               if (!/^[a-zA-Z\u00C0-\u024F -]{1,32}$/.test(arg))
+                       return "[1,32] letters + hyphen/space";
+               return "";
+       },
+
+       "check_email": function(arg)
+       {
+               if (!_.isString(arg))
+                       return "not a string";
+               if (arg.length > 64)
+                       return "string too long: max. 64 characters";
+               // Regexp used in "type='email'" inputs ( http://emailregex.com/ )
+               if (!/^[a-zA-Z0-9.!#$%&’*+/=?^_`{|}~-]+@[a-zA-Z0-9-]+(?:\.[a-zA-Z0-9-]+)*$/.test(arg))
+                       return "X@Y, alphanumerics and . _ - +(X)";
+               return "";
+       },
+
+       "check_code": function(arg)
+       {
+               if (!_.isString(arg))
+                       return "not a string";
+               if (!/^[\w.-]{1,16}$/.test(arg))
+                       return "[1,16] alphanumerics and . _ -";
+               return "";
+       },
+
+       "check_number": function(arg)
+       {
+               if (!_.isNumber(arg))
+                       arg = parseFloat(arg);
+               if (isNaN(arg))
+                       return "not a number";
+               return "";
+       },
+
+       "check_integer": function(arg)
+       {
+               if (!_.isNumber(arg))
+                       arg = parseInt(arg);
+               if (isNaN(arg) || arg % 1 != 0)
+                       return "not an integer";
+               return "";
+       },
+
+       "check_positiveInteger": function(arg)
+       {
+               return Validator["check_integer"](arg) || (arg<0 ? "not positive" : "");
+       },
+
+       "check_boolean": function(arg)
+       {
+               if (!_.isBoolean(arg))
+                       return "not a boolean";
+               return "";
+       },
+
+       "check_password": function(arg)
+       {
+               if (!_.isString(arg))
+                       return "not a string";
+               if (!/^[\x21-\x7E]{1,16}$/.test(arg))
+                       return "[1,16] ASCII characters with code in [33,126]";
+               return "";
+       },
+
+       // Sha-1 hash: length 40, hexadecimal
+       "check_hash": function(arg)
+       {
+               if (!_.isString(arg))
+                       return "not a string";
+               if (!/^[a-f0-9]{40}$/.test(arg))
+                       return "not a sha-1 hash";
+               return "";
+       },
+
+       "check_integerArray": function(arg)
+       {
+               if (!_.isArray(arg))
+                       return "not an array";
+               for (let i=0; i<arg.length; i++)
+               {
+                       let error = Validator["check_integer"](arg[i]);
+                       if (error.length > 0)
+                               return error;
+               }
+               return "";
+       },
+
+       "check_stringOrIntegerArray": function(arg)
+       {
+               if (!_.isString(arg))
+                       return Validator["check_integerArray"](arg);
+               return "";
+       },
+});
+
+try { module.exports = Validator.checkObject; } catch (err) {} //for server
diff --git a/public/stylesheets/assessment.css b/public/stylesheets/assessment.css
new file mode 100644 (file)
index 0000000..3187907
--- /dev/null
@@ -0,0 +1,55 @@
+a#rightButton {
+       position: absolute;
+       top: 0;
+       right: 0;
+}
+
+.question {
+       margin: 20px 5px;
+       padding: 15px 0;
+}
+
+.question button {
+       display: block;
+       margin: 0 auto 15px auto;
+}
+
+.question label {
+       color: black;
+}
+
+.question .choiceCorrect {
+       background-color: lightgreen;
+}
+
+.question .choiceWrong {
+       background-color: peachpuff;
+}
+
+.question .wording {
+       margin-bottom: 10px;
+}
+
+.question .option {
+       margin-left: 15px;
+}
+
+.question p {
+       margin-top: 10px;
+}
+
+.questionInactive {
+       background-color: lightgrey;
+}
+
+.introduction {
+       padding: 20px 5px;
+}
+
+.conclusion {
+       padding: 20px 5px;
+}
+
+.timer {
+       font-size: 2rem;
+}
diff --git a/public/stylesheets/course.css b/public/stylesheets/course.css
new file mode 100644 (file)
index 0000000..2d61347
--- /dev/null
@@ -0,0 +1,56 @@
+h4.title {
+       cursor: pointer;
+       background-color: lightgrey;
+}
+
+tr.assessment {
+       cursor: pointer;
+}
+
+input#password {
+       width: auto;
+}
+
+table.result {
+       cursor: pointer;
+}
+
+tr.stats {
+       padding-top: 10px;
+}
+
+#questionList {
+       margin: 20px 5px;
+}
+
+.question {
+       margin: 20px 0;
+}
+
+.question .choiceCorrect {
+       background-color: lightgreen;
+}
+
+.question .wording {
+       margin-bottom: 10px;
+}
+
+.question .option {
+       margin-left: 15px;
+}
+
+.question p {
+       margin-top: 10px;
+}
+
+.questionInactive {
+       background-color: lightgrey;
+}
+
+.introduction {
+       margin-top: 20px;
+}
+
+.conclusion {
+       margin-bottom: 20px;
+}
diff --git a/public/stylesheets/courseList.css b/public/stylesheets/courseList.css
new file mode 100644 (file)
index 0000000..1846eb8
--- /dev/null
@@ -0,0 +1,3 @@
+tr.course {
+       cursor: pointer;
+}
diff --git a/public/stylesheets/index.css b/public/stylesheets/index.css
new file mode 100644 (file)
index 0000000..30c9107
--- /dev/null
@@ -0,0 +1,3 @@
+tr.teacher {
+       cursor: pointer;
+}
diff --git a/public/stylesheets/layout.css b/public/stylesheets/layout.css
new file mode 100644 (file)
index 0000000..d4dc11d
--- /dev/null
@@ -0,0 +1,3 @@
+.on-left {
+       margin-right: 20px;
+}
diff --git a/public/stylesheets/login.css b/public/stylesheets/login.css
new file mode 100644 (file)
index 0000000..f745f2e
--- /dev/null
@@ -0,0 +1,22 @@
+#form {
+       margin-top: 30px;
+       padding: 10px 20px 20px 20px;
+}
+
+#submit {
+       margin: 10px 0 15px 0;
+}
+
+#toggle {
+       color: darkblue;
+       cursor: pointer;
+}
+
+#toggle span:not(:last-child) {
+       margin-right: 30px;
+}
+
+#dialog {
+       margin-top: 20px;
+       padding: 20px;
+}
diff --git a/public/stylesheets/monitor.css b/public/stylesheets/monitor.css
new file mode 100644 (file)
index 0000000..8f414f5
--- /dev/null
@@ -0,0 +1 @@
+/* TODO */
diff --git a/public/vendor/prism/prism-components.zip b/public/vendor/prism/prism-components.zip
new file mode 100644 (file)
index 0000000..dbc818a
--- /dev/null
@@ -0,0 +1 @@
+#$# git-fat 0a0ef2276985c0f221752b8c330d9b04275cafc1               589999
diff --git a/public/vendor/prism/prism.css b/public/vendor/prism/prism.css
new file mode 100644 (file)
index 0000000..506acb3
--- /dev/null
@@ -0,0 +1,140 @@
+/* PrismJS 1.10.0
+http://prismjs.com/download.html?themes=prism&languages=clike+python+sql&plugins=autoloader */
+/**
+ * prism.js default theme for JavaScript, CSS and HTML
+ * Based on dabblet (http://dabblet.com)
+ * @author Lea Verou
+ */
+
+code[class*="language-"],
+pre[class*="language-"] {
+       color: black;
+       background: none;
+       text-shadow: 0 1px white;
+       font-family: Consolas, Monaco, 'Andale Mono', 'Ubuntu Mono', monospace;
+       text-align: left;
+       white-space: pre;
+       word-spacing: normal;
+       word-break: normal;
+       word-wrap: normal;
+       line-height: 1.5;
+
+       -moz-tab-size: 4;
+       -o-tab-size: 4;
+       tab-size: 4;
+
+       -webkit-hyphens: none;
+       -moz-hyphens: none;
+       -ms-hyphens: none;
+       hyphens: none;
+}
+
+pre[class*="language-"]::-moz-selection, pre[class*="language-"] ::-moz-selection,
+code[class*="language-"]::-moz-selection, code[class*="language-"] ::-moz-selection {
+       text-shadow: none;
+       background: #b3d4fc;
+}
+
+pre[class*="language-"]::selection, pre[class*="language-"] ::selection,
+code[class*="language-"]::selection, code[class*="language-"] ::selection {
+       text-shadow: none;
+       background: #b3d4fc;
+}
+
+@media print {
+       code[class*="language-"],
+       pre[class*="language-"] {
+               text-shadow: none;
+       }
+}
+
+/* Code blocks */
+pre[class*="language-"] {
+       padding: 1em;
+       margin: .5em 0;
+       overflow: auto;
+}
+
+:not(pre) > code[class*="language-"],
+pre[class*="language-"] {
+       background: #f5f2f0;
+}
+
+/* Inline code */
+:not(pre) > code[class*="language-"] {
+       padding: .1em;
+       border-radius: .3em;
+       white-space: normal;
+}
+
+.token.comment,
+.token.prolog,
+.token.doctype,
+.token.cdata {
+       color: slategray;
+}
+
+.token.punctuation {
+       color: #999;
+}
+
+.namespace {
+       opacity: .7;
+}
+
+.token.property,
+.token.tag,
+.token.boolean,
+.token.number,
+.token.constant,
+.token.symbol,
+.token.deleted {
+       color: #905;
+}
+
+.token.selector,
+.token.attr-name,
+.token.string,
+.token.char,
+.token.builtin,
+.token.inserted {
+       color: #690;
+}
+
+.token.operator,
+.token.entity,
+.token.url,
+.language-css .token.string,
+.style .token.string {
+       color: #a67f59;
+       background: hsla(0, 0%, 100%, .5);
+}
+
+.token.atrule,
+.token.attr-value,
+.token.keyword {
+       color: #07a;
+}
+
+.token.function {
+       color: #DD4A68;
+}
+
+.token.regex,
+.token.important,
+.token.variable {
+       color: #e90;
+}
+
+.token.important,
+.token.bold {
+       font-weight: bold;
+}
+.token.italic {
+       font-style: italic;
+}
+
+.token.entity {
+       cursor: help;
+}
+
diff --git a/public/vendor/prism/prism.js b/public/vendor/prism/prism.js
new file mode 100644 (file)
index 0000000..4b29e81
--- /dev/null
@@ -0,0 +1,7 @@
+/* PrismJS 1.10.0
+http://prismjs.com/download.html?themes=prism&languages=clike+python+sql&plugins=autoloader */
+var _self="undefined"!=typeof window?window:"undefined"!=typeof WorkerGlobalScope&&self instanceof WorkerGlobalScope?self:{},Prism=function(){var e=/\blang(?:uage)?-(\w+)\b/i,t=0,n=_self.Prism={manual:_self.Prism&&_self.Prism.manual,disableWorkerMessageHandler:_self.Prism&&_self.Prism.disableWorkerMessageHandler,util:{encode:function(e){return e instanceof r?new r(e.type,n.util.encode(e.content),e.alias):"Array"===n.util.type(e)?e.map(n.util.encode):e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/\u00a0/g," ")},type:function(e){return Object.prototype.toString.call(e).match(/\[object (\w+)\]/)[1]},objId:function(e){return e.__id||Object.defineProperty(e,"__id",{value:++t}),e.__id},clone:function(e){var t=n.util.type(e);switch(t){case"Object":var r={};for(var a in e)e.hasOwnProperty(a)&&(r[a]=n.util.clone(e[a]));return r;case"Array":return e.map(function(e){return n.util.clone(e)})}return e}},languages:{extend:function(e,t){var r=n.util.clone(n.languages[e]);for(var a in t)r[a]=t[a];return r},insertBefore:function(e,t,r,a){a=a||n.languages;var l=a[e];if(2==arguments.length){r=arguments[1];for(var i in r)r.hasOwnProperty(i)&&(l[i]=r[i]);return l}var o={};for(var s in l)if(l.hasOwnProperty(s)){if(s==t)for(var i in r)r.hasOwnProperty(i)&&(o[i]=r[i]);o[s]=l[s]}return n.languages.DFS(n.languages,function(t,n){n===a[e]&&t!=e&&(this[t]=o)}),a[e]=o},DFS:function(e,t,r,a){a=a||{};for(var l in e)e.hasOwnProperty(l)&&(t.call(e,l,e[l],r||l),"Object"!==n.util.type(e[l])||a[n.util.objId(e[l])]?"Array"!==n.util.type(e[l])||a[n.util.objId(e[l])]||(a[n.util.objId(e[l])]=!0,n.languages.DFS(e[l],t,l,a)):(a[n.util.objId(e[l])]=!0,n.languages.DFS(e[l],t,null,a)))}},plugins:{},highlightAll:function(e,t){n.highlightAllUnder(document,e,t)},highlightAllUnder:function(e,t,r){var a={callback:r,selector:'code[class*="language-"], [class*="language-"] code, code[class*="lang-"], [class*="lang-"] code'};n.hooks.run("before-highlightall",a);for(var l,i=a.elements||e.querySelectorAll(a.selector),o=0;l=i[o++];)n.highlightElement(l,t===!0,a.callback)},highlightElement:function(t,r,a){for(var l,i,o=t;o&&!e.test(o.className);)o=o.parentNode;o&&(l=(o.className.match(e)||[,""])[1].toLowerCase(),i=n.languages[l]),t.className=t.className.replace(e,"").replace(/\s+/g," ")+" language-"+l,t.parentNode&&(o=t.parentNode,/pre/i.test(o.nodeName)&&(o.className=o.className.replace(e,"").replace(/\s+/g," ")+" language-"+l));var s=t.textContent,g={element:t,language:l,grammar:i,code:s};if(n.hooks.run("before-sanity-check",g),!g.code||!g.grammar)return g.code&&(n.hooks.run("before-highlight",g),g.element.textContent=g.code,n.hooks.run("after-highlight",g)),n.hooks.run("complete",g),void 0;if(n.hooks.run("before-highlight",g),r&&_self.Worker){var u=new Worker(n.filename);u.onmessage=function(e){g.highlightedCode=e.data,n.hooks.run("before-insert",g),g.element.innerHTML=g.highlightedCode,a&&a.call(g.element),n.hooks.run("after-highlight",g),n.hooks.run("complete",g)},u.postMessage(JSON.stringify({language:g.language,code:g.code,immediateClose:!0}))}else g.highlightedCode=n.highlight(g.code,g.grammar,g.language),n.hooks.run("before-insert",g),g.element.innerHTML=g.highlightedCode,a&&a.call(t),n.hooks.run("after-highlight",g),n.hooks.run("complete",g)},highlight:function(e,t,a){var l=n.tokenize(e,t);return r.stringify(n.util.encode(l),a)},matchGrammar:function(e,t,r,a,l,i,o){var s=n.Token;for(var g in r)if(r.hasOwnProperty(g)&&r[g]){if(g==o)return;var u=r[g];u="Array"===n.util.type(u)?u:[u];for(var c=0;c<u.length;++c){var h=u[c],f=h.inside,d=!!h.lookbehind,m=!!h.greedy,p=0,y=h.alias;if(m&&!h.pattern.global){var v=h.pattern.toString().match(/[imuy]*$/)[0];h.pattern=RegExp(h.pattern.source,v+"g")}h=h.pattern||h;for(var b=a,k=l;b<t.length;k+=t[b].length,++b){var w=t[b];if(t.length>e.length)return;if(!(w instanceof s)){h.lastIndex=0;var _=h.exec(w),P=1;if(!_&&m&&b!=t.length-1){if(h.lastIndex=k,_=h.exec(e),!_)break;for(var A=_.index+(d?_[1].length:0),j=_.index+_[0].length,x=b,O=k,N=t.length;N>x&&(j>O||!t[x].type&&!t[x-1].greedy);++x)O+=t[x].length,A>=O&&(++b,k=O);if(t[b]instanceof s||t[x-1].greedy)continue;P=x-b,w=e.slice(k,O),_.index-=k}if(_){d&&(p=_[1].length);var A=_.index+p,_=_[0].slice(p),j=A+_.length,S=w.slice(0,A),C=w.slice(j),M=[b,P];S&&(++b,k+=S.length,M.push(S));var E=new s(g,f?n.tokenize(_,f):_,y,_,m);if(M.push(E),C&&M.push(C),Array.prototype.splice.apply(t,M),1!=P&&n.matchGrammar(e,t,r,b,k,!0,g),i)break}else if(i)break}}}}},tokenize:function(e,t){var r=[e],a=t.rest;if(a){for(var l in a)t[l]=a[l];delete t.rest}return n.matchGrammar(e,r,t,0,0,!1),r},hooks:{all:{},add:function(e,t){var r=n.hooks.all;r[e]=r[e]||[],r[e].push(t)},run:function(e,t){var r=n.hooks.all[e];if(r&&r.length)for(var a,l=0;a=r[l++];)a(t)}}},r=n.Token=function(e,t,n,r,a){this.type=e,this.content=t,this.alias=n,this.length=0|(r||"").length,this.greedy=!!a};if(r.stringify=function(e,t,a){if("string"==typeof e)return e;if("Array"===n.util.type(e))return e.map(function(n){return r.stringify(n,t,e)}).join("");var l={type:e.type,content:r.stringify(e.content,t,a),tag:"span",classes:["token",e.type],attributes:{},language:t,parent:a};if(e.alias){var i="Array"===n.util.type(e.alias)?e.alias:[e.alias];Array.prototype.push.apply(l.classes,i)}n.hooks.run("wrap",l);var o=Object.keys(l.attributes).map(function(e){return e+'="'+(l.attributes[e]||"").replace(/"/g,"&quot;")+'"'}).join(" ");return"<"+l.tag+' class="'+l.classes.join(" ")+'"'+(o?" "+o:"")+">"+l.content+"</"+l.tag+">"},!_self.document)return _self.addEventListener?(n.disableWorkerMessageHandler||_self.addEventListener("message",function(e){var t=JSON.parse(e.data),r=t.language,a=t.code,l=t.immediateClose;_self.postMessage(n.highlight(a,n.languages[r],r)),l&&_self.close()},!1),_self.Prism):_self.Prism;var a=document.currentScript||[].slice.call(document.getElementsByTagName("script")).pop();return a&&(n.filename=a.src,n.manual||a.hasAttribute("data-manual")||("loading"!==document.readyState?window.requestAnimationFrame?window.requestAnimationFrame(n.highlightAll):window.setTimeout(n.highlightAll,16):document.addEventListener("DOMContentLoaded",n.highlightAll))),_self.Prism}();"undefined"!=typeof module&&module.exports&&(module.exports=Prism),"undefined"!=typeof global&&(global.Prism=Prism);
+Prism.languages.clike={comment:[{pattern:/(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/,lookbehind:!0},{pattern:/(^|[^\\:])\/\/.*/,lookbehind:!0}],string:{pattern:/(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/,greedy:!0},"class-name":{pattern:/((?:\b(?:class|interface|extends|implements|trait|instanceof|new)\s+)|(?:catch\s+\())[\w.\\]+/i,lookbehind:!0,inside:{punctuation:/[.\\]/}},keyword:/\b(?:if|else|while|do|for|return|in|instanceof|function|new|try|throw|catch|finally|null|break|continue)\b/,"boolean":/\b(?:true|false)\b/,"function":/[a-z0-9_]+(?=\()/i,number:/\b-?(?:0x[\da-f]+|\d*\.?\d+(?:e[+-]?\d+)?)\b/i,operator:/--?|\+\+?|!=?=?|<=?|>=?|==?=?|&&?|\|\|?|\?|\*|\/|~|\^|%/,punctuation:/[{}[\];(),.:]/};
+Prism.languages.python={comment:{pattern:/(^|[^\\])#.*/,lookbehind:!0},"triple-quoted-string":{pattern:/("""|''')[\s\S]+?\1/,greedy:!0,alias:"string"},string:{pattern:/("|')(?:\\.|(?!\1)[^\\\r\n])*\1/,greedy:!0},"function":{pattern:/((?:^|\s)def[ \t]+)[a-zA-Z_]\w*(?=\s*\()/g,lookbehind:!0},"class-name":{pattern:/(\bclass\s+)\w+/i,lookbehind:!0},keyword:/\b(?:as|assert|async|await|break|class|continue|def|del|elif|else|except|exec|finally|for|from|global|if|import|in|is|lambda|nonlocal|pass|print|raise|return|try|while|with|yield)\b/,builtin:/\b(?:__import__|abs|all|any|apply|ascii|basestring|bin|bool|buffer|bytearray|bytes|callable|chr|classmethod|cmp|coerce|compile|complex|delattr|dict|dir|divmod|enumerate|eval|execfile|file|filter|float|format|frozenset|getattr|globals|hasattr|hash|help|hex|id|input|int|intern|isinstance|issubclass|iter|len|list|locals|long|map|max|memoryview|min|next|object|oct|open|ord|pow|property|range|raw_input|reduce|reload|repr|reversed|round|set|setattr|slice|sorted|staticmethod|str|sum|super|tuple|type|unichr|unicode|vars|xrange|zip)\b/,"boolean":/\b(?:True|False|None)\b/,number:/\b-?(?:0[bo])?(?:(?:\d|0x[\da-f])[\da-f]*\.?\d*|\.\d+)(?:e[+-]?\d+)?j?\b/i,operator:/[-+%=]=?|!=|\*\*?=?|\/\/?=?|<[<=>]?|>[=>]?|[&|^~]|\b(?:or|and|not)\b/,punctuation:/[{}[\];(),.:]/};
+Prism.languages.sql={comment:{pattern:/(^|[^\\])(?:\/\*[\s\S]*?\*\/|(?:--|\/\/|#).*)/,lookbehind:!0},string:{pattern:/(^|[^@\\])("|')(?:\\[\s\S]|(?!\2)[^\\])*\2/,greedy:!0,lookbehind:!0},variable:/@[\w.$]+|@(["'`])(?:\\[\s\S]|(?!\1)[^\\])+\1/,"function":/\b(?:COUNT|SUM|AVG|MIN|MAX|FIRST|LAST|UCASE|LCASE|MID|LEN|ROUND|NOW|FORMAT)(?=\s*\()/i,keyword:/\b(?:ACTION|ADD|AFTER|ALGORITHM|ALL|ALTER|ANALYZE|ANY|APPLY|AS|ASC|AUTHORIZATION|AUTO_INCREMENT|BACKUP|BDB|BEGIN|BERKELEYDB|BIGINT|BINARY|BIT|BLOB|BOOL|BOOLEAN|BREAK|BROWSE|BTREE|BULK|BY|CALL|CASCADED?|CASE|CHAIN|CHAR VARYING|CHARACTER (?:SET|VARYING)|CHARSET|CHECK|CHECKPOINT|CLOSE|CLUSTERED|COALESCE|COLLATE|COLUMN|COLUMNS|COMMENT|COMMIT|COMMITTED|COMPUTE|CONNECT|CONSISTENT|CONSTRAINT|CONTAINS|CONTAINSTABLE|CONTINUE|CONVERT|CREATE|CROSS|CURRENT(?:_DATE|_TIME|_TIMESTAMP|_USER)?|CURSOR|DATA(?:BASES?)?|DATE(?:TIME)?|DBCC|DEALLOCATE|DEC|DECIMAL|DECLARE|DEFAULT|DEFINER|DELAYED|DELETE|DELIMITER(?:S)?|DENY|DESC|DESCRIBE|DETERMINISTIC|DISABLE|DISCARD|DISK|DISTINCT|DISTINCTROW|DISTRIBUTED|DO|DOUBLE(?: PRECISION)?|DROP|DUMMY|DUMP(?:FILE)?|DUPLICATE KEY|ELSE|ENABLE|ENCLOSED BY|END|ENGINE|ENUM|ERRLVL|ERRORS|ESCAPE(?:D BY)?|EXCEPT|EXEC(?:UTE)?|EXISTS|EXIT|EXPLAIN|EXTENDED|FETCH|FIELDS|FILE|FILLFACTOR|FIRST|FIXED|FLOAT|FOLLOWING|FOR(?: EACH ROW)?|FORCE|FOREIGN|FREETEXT(?:TABLE)?|FROM|FULL|FUNCTION|GEOMETRY(?:COLLECTION)?|GLOBAL|GOTO|GRANT|GROUP|HANDLER|HASH|HAVING|HOLDLOCK|IDENTITY(?:_INSERT|COL)?|IF|IGNORE|IMPORT|INDEX|INFILE|INNER|INNODB|INOUT|INSERT|INT|INTEGER|INTERSECT|INTO|INVOKER|ISOLATION LEVEL|JOIN|KEYS?|KILL|LANGUAGE SQL|LAST|LEFT|LIMIT|LINENO|LINES|LINESTRING|LOAD|LOCAL|LOCK|LONG(?:BLOB|TEXT)|MATCH(?:ED)?|MEDIUM(?:BLOB|INT|TEXT)|MERGE|MIDDLEINT|MODIFIES SQL DATA|MODIFY|MULTI(?:LINESTRING|POINT|POLYGON)|NATIONAL(?: CHAR VARYING| CHARACTER(?: VARYING)?| VARCHAR)?|NATURAL|NCHAR(?: VARCHAR)?|NEXT|NO(?: SQL|CHECK|CYCLE)?|NONCLUSTERED|NULLIF|NUMERIC|OFF?|OFFSETS?|ON|OPEN(?:DATASOURCE|QUERY|ROWSET)?|OPTIMIZE|OPTION(?:ALLY)?|ORDER|OUT(?:ER|FILE)?|OVER|PARTIAL|PARTITION|PERCENT|PIVOT|PLAN|POINT|POLYGON|PRECEDING|PRECISION|PREV|PRIMARY|PRINT|PRIVILEGES|PROC(?:EDURE)?|PUBLIC|PURGE|QUICK|RAISERROR|READ(?:S SQL DATA|TEXT)?|REAL|RECONFIGURE|REFERENCES|RELEASE|RENAME|REPEATABLE|REPLICATION|REQUIRE|RESTORE|RESTRICT|RETURNS?|REVOKE|RIGHT|ROLLBACK|ROUTINE|ROW(?:COUNT|GUIDCOL|S)?|RTREE|RULE|SAVE(?:POINT)?|SCHEMA|SELECT|SERIAL(?:IZABLE)?|SESSION(?:_USER)?|SET(?:USER)?|SHARE MODE|SHOW|SHUTDOWN|SIMPLE|SMALLINT|SNAPSHOT|SOME|SONAME|START(?:ING BY)?|STATISTICS|STATUS|STRIPED|SYSTEM_USER|TABLES?|TABLESPACE|TEMP(?:ORARY|TABLE)?|TERMINATED BY|TEXT(?:SIZE)?|THEN|TIMESTAMP|TINY(?:BLOB|INT|TEXT)|TOP?|TRAN(?:SACTIONS?)?|TRIGGER|TRUNCATE|TSEQUAL|TYPES?|UNBOUNDED|UNCOMMITTED|UNDEFINED|UNION|UNIQUE|UNPIVOT|UPDATE(?:TEXT)?|USAGE|USE|USER|USING|VALUES?|VAR(?:BINARY|CHAR|CHARACTER|YING)|VIEW|WAITFOR|WARNINGS|WHEN|WHERE|WHILE|WITH(?: ROLLUP|IN)?|WORK|WRITE(?:TEXT)?)\b/i,"boolean":/\b(?:TRUE|FALSE|NULL)\b/i,number:/\b-?(?:0x)?\d*\.?[\da-f]+\b/,operator:/[-+*\/=%^~]|&&?|\|\|?|!=?|<(?:=>?|<|>)?|>[>=]?|\b(?:AND|BETWEEN|IN|LIKE|NOT|OR|IS|DIV|REGEXP|RLIKE|SOUNDS LIKE|XOR)\b/i,punctuation:/[;[\]()`,.]/};
+!function(){if("undefined"!=typeof self&&self.Prism&&self.document&&document.createElement){var e={javascript:"clike",actionscript:"javascript",arduino:"cpp",aspnet:"markup",bison:"c",c:"clike",csharp:"clike",cpp:"c",coffeescript:"javascript",crystal:"ruby","css-extras":"css",d:"clike",dart:"clike",django:"markup",fsharp:"clike",flow:"javascript",glsl:"clike",go:"clike",groovy:"clike",haml:"ruby",handlebars:"markup",haxe:"clike",java:"clike",jolie:"clike",kotlin:"clike",less:"css",markdown:"markup",n4js:"javascript",nginx:"clike",objectivec:"c",opencl:"cpp",parser:"markup",php:"clike","php-extras":"php",processing:"clike",protobuf:"clike",pug:"javascript",qore:"clike",jsx:["markup","javascript"],reason:"clike",ruby:"clike",sass:"css",scss:"css",scala:"java",smarty:"markup",swift:"clike",textile:"markup",twig:"markup",typescript:"javascript",vbnet:"basic",wiki:"markup",xeora:"markup"},a={},c="none",s=document.getElementsByTagName("script");s=s[s.length-1];var r="components/";if(s.hasAttribute("data-autoloader-path")){var t=s.getAttribute("data-autoloader-path").trim();t.length>0&&!/^[a-z]+:\/\//i.test(s.src)&&(r=t.replace(/\/?$/,"/"))}else/[\w-]+\.js$/.test(s.src)&&(r=s.src.replace(/[\w-]+\.js$/,"components/"));var n=Prism.plugins.autoloader={languages_path:r,use_minified:!0},s=function(e,a,c){var s=document.createElement("script");s.src=e,s.async=!0,s.onload=function(){document.body.removeChild(s),a&&a()},s.onerror=function(){document.body.removeChild(s),c&&c()},document.body.appendChild(s)},i=function(e){return n.languages_path+"prism-"+e+(n.use_minified?".min":"")+".js"},l=function(e,c){var s=a[e];s||(s=a[e]={});var r=c.getAttribute("data-dependencies");!r&&c.parentNode&&"pre"===c.parentNode.tagName.toLowerCase()&&(r=c.parentNode.getAttribute("data-dependencies")),r=r?r.split(/\s*,\s*/g):[],o(r,function(){u(e,function(){Prism.highlightElement(c)})})},o=function(e,a,c){"string"==typeof e&&(e=[e]);var s=0,r=e.length,t=function(){r>s?u(e[s],function(){s++,t()},function(){c&&c(e[s])}):s===r&&a&&a(e)};t()},u=function(c,r,t){var n=function(){var e=!1;c.indexOf("!")>=0&&(e=!0,c=c.replace("!",""));var n=a[c];if(n||(n=a[c]={}),r&&(n.success_callbacks||(n.success_callbacks=[]),n.success_callbacks.push(r)),t&&(n.error_callbacks||(n.error_callbacks=[]),n.error_callbacks.push(t)),!e&&Prism.languages[c])p(c);else if(!e&&n.error)k(c);else if(e||!n.loading){n.loading=!0;var l=i(c);s(l,function(){n.loading=!1,p(c)},function(){n.loading=!1,n.error=!0,k(c)})}},l=e[c];l&&l.length?o(l,n):n()},p=function(e){a[e]&&a[e].success_callbacks&&a[e].success_callbacks.length&&a[e].success_callbacks.forEach(function(a){a(e)})},k=function(e){a[e]&&a[e].error_callbacks&&a[e].error_callbacks.length&&a[e].error_callbacks.forEach(function(a){a(e)})};Prism.hooks.add("complete",function(e){e.element&&e.language&&!e.grammar&&e.language!==c&&l(e.language,e.element)})}}();
diff --git a/routes/all.js b/routes/all.js
new file mode 100644 (file)
index 0000000..1c0d052
--- /dev/null
@@ -0,0 +1,11 @@
+var router = require("express").Router();
+
+// AJAX requests:
+router.use("/", require("./users"));
+router.use("/", require("./courses"));
+router.use("/", require("./assessments"));
+
+// Pages:
+router.use("/", require("./pages"));
+
+module.exports = router;
diff --git a/routes/assessments.js b/routes/assessments.js
new file mode 100644 (file)
index 0000000..559f08f
--- /dev/null
@@ -0,0 +1,96 @@
+let router = require("express").Router();
+const access = require("../utils/access");
+const UserModel = require("../models/user");
+const AssessmentModel = require("../models/assessment");
+const AssessmentEntity = require("../entities/assessment");
+const CourseModel = require("../models/course");
+const params = require("../config/parameters");
+const validator = require("../public/javascripts/utils/validation");
+const ObjectId = require("bson-objectid");
+const sanitizeHtml = require('sanitize-html');
+
+router.get("/add/assessment", access.ajax, access.logged, (req,res) => {
+       const name = req.query["name"];
+       const cid = req.query["cid"];
+       let error = validator({cid:cid, name:name}, "Assessment");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       AssessmentModel.add(req.user._id, ObjectId(cid), name, (err,assessment) => {
+               access.checkRequest(res, err, assessment, "Assessment addition failed", () => {
+                       res.json(assessment);
+               });
+       });
+});
+
+router.post("/update/assessment", access.ajax, access.logged, (req,res) => {
+       const assessment = JSON.parse(req.body["assessment"]);
+       let error = validator(assessment, "Assessment");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       const sanitizeOpts = {allowedTags: sanitizeHtml.defaults.allowedTags.concat([ 'img' ]) };
+       assessment.introduction = sanitizeHtml(assessment.introduction, sanitizeOpts);
+       assessment.conclusion = sanitizeHtml(assessment.conclusion, sanitizeOpts);
+       assessment.questions.forEach( q => {
+               q.wording = sanitizeHtml(q.wording, sanitizeOpts);
+               //q.answer = sanitizeHtml(q.answer); //if text (TODO: it's an array in this case?!)
+               for (let i=0; i<q.options.length; i++) //if QCM
+                       q.options[i] = sanitizeHtml(q.options[i], sanitizeOpts);
+       });
+       AssessmentModel.update(req.user._id, assessment, (err,ret) => {
+               access.checkRequest(res, err, ret, "Assessment update failed", () => {
+                       res.json({});
+               });
+       });
+});
+
+// Generate and set student password, return it
+router.get("/start/assessment", access.ajax, (req,res) => {
+       let number = req.query["number"];
+       let aid = req.query["aid"];
+       let error = validator({ _id:aid, papers:[{number:number}] }, "Assessment");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       AssessmentModel.startSession(ObjectId(aid), number, (err,ret) => {
+               access.checkRequest(res,err,ret,"Failed session initialization", () => {
+                       // Set password
+                       res.cookie("password", ret.password, {
+                               httpOnly: true,
+                               maxAge: params.cookieExpire,
+                       });
+                       res.json(ret); //contains questions+password
+               });
+       });
+});
+
+router.get("/send/answer", access.ajax, (req,res) => {
+       let aid = req.query["aid"];
+       let number = req.query["number"];
+       let password = req.query["password"];
+       let input = JSON.parse(req.query["answer"]);
+       let error = validator({ _id:aid, papers:[{number:number,password:password,inputs:[input]}] }, "Assessment");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       AssessmentEntity.setInput(ObjectId(aid), number, password, input, (err,ret) => {
+               access.checkRequest(res,err,ret,"Cannot send answer", () => {
+                       res.json({});
+               });
+       });
+});
+
+router.get("/end/assessment", access.ajax, (req,res) => {
+       let aid = req.query["aid"];
+       let number = req.query["number"];
+       let password = req.query["password"];
+       let error = validator({ _id:aid, papers:[{number:number,password:password}] }, "Assessment");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       // Destroy pwd, set endTime, return conclusion
+       AssessmentModel.endSession(ObjectId(aid), number, password, (err,conclusion) => {
+               access.checkRequest(res,err,conclusion,"Cannot end assessment", () => {
+                       res.clearCookie('password');
+                       res.json(conclusion);
+               });
+       });
+});
+
+module.exports = router;
diff --git a/routes/courses.js b/routes/courses.js
new file mode 100644 (file)
index 0000000..d221858
--- /dev/null
@@ -0,0 +1,77 @@
+let router = require("express").Router();
+const access = require("../utils/access.js");
+const validator = require("../public/javascripts/utils/validation");
+const sanitizeHtml = require('sanitize-html');
+const ObjectId = require("bson-objectid");
+const CourseEntity = require("../entities/course");
+const CourseModel = require("../models/course");
+
+router.get('/add/course', access.ajax, access.logged, (req,res) => {
+       let code = req.query["code"];
+       let description = sanitizeHtml(req.query["description"]);
+       let error = validator({code:code}, "Course");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       CourseEntity.insert(req.user._id, code, description, (err,course) => {
+               access.checkRequest(res, err, course, "Course addition failed", () => {
+                       res.json(course);
+               });
+       });
+});
+
+router.get("/set/password", access.ajax, access.logged, (req,res) => {
+       let cid = req.query["cid"];
+       let pwd = req.query["pwd"];
+       let error = validator({password:pwd, _id:cid}, "Course");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       CourseModel.setPassword(req.user._id, ObjectId(cid), pwd, (err,ret) => {
+               access.checkRequest(res, err, ret, "password update failed", () => {
+                       res.json({});
+               });
+       });
+});
+
+router.post('/import/students', access.ajax, access.logged, (req,res) => {
+       let cid = req.body["cid"];
+       let students = JSON.parse(req.body["students"]);
+       let error = validator({_id:cid, students: students}, "Course");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       access.getUser(req, res, (err,user) => {
+               if (!!err)
+                       return res.json(err);
+               CourseModel.importStudents(req.user._id, ObjectId(cid), students, (err,ret) => {
+                       access.checkRequest(res, err, ret, "Students addition failed", () => {
+                               res.json({});
+                       });
+               });
+       });
+});
+
+router.get('/get/student', access.ajax, (req,res) => {
+       let number = req.query["number"];
+       let cid = req.query["cid"];
+       let error = validator({ _id: cid, students: [{number:number}] }, "Course");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       CourseEntity.getStudent(ObjectId(cid), number, (err,ret) => {
+               access.checkRequest(res, err, ret, "Failed retrieving student", () => {
+                       res.json({student: ret.students[0]});
+               });
+       });
+});
+
+router.get('/remove/course', access.ajax, access.logged, (req,res) => {
+       let cid = req.query["cid"];
+       let error = validator({_id:cid}, "Course");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       CourseModel.remove(req.user._id, ObjectId(cid), (err,ret) => {
+               access.checkRequest(res, err, ret, "Course removal failed", () => {
+                       res.json({});
+               });
+       });
+});
+
+module.exports = router;
diff --git a/routes/pages.js b/routes/pages.js
new file mode 100644 (file)
index 0000000..37b84cf
--- /dev/null
@@ -0,0 +1,136 @@
+let router = require("express").Router();
+const access = require("../utils/access");
+const UserEntity = require("../entities/user");
+const AssessmentEntity = require("../entities/assessment");
+const CourseModel = require("../models/course");
+const AssessmentModel = require("../models/assessment");
+
+// Actual pages (least specific last)
+
+// List initials and count assessments
+router.get("/", (req,res) => {
+       UserEntity.getAll( (err,userArray) => {
+               if (!!err)
+                       return res.json(err);
+               res.render("index", {
+                       title: "home",
+                       userArray: userArray,
+               });
+       });
+});
+
+// Login screen
+router.get("/login", access.unlogged, (req,res) => {
+       res.render("login", {
+               title: "login",
+       });
+});
+
+// Redirection screens when possible cheating attempt detected in exam
+router.get("/enablejs", (req,res) => {
+       res.render("enable-js", {
+               title: "JS disabled",
+       });
+});
+
+router.get("/nodevtools", (req,res) => {
+       res.render("no-devtools", {
+               title: "Devtools enabled",
+       });
+});
+
+// List courses of some user (should be [a-z]+[0-9]* but fails...)
+router.get("/:initials([a-z0-9]+)", (req,res) => {
+       let initials = req.params["initials"];
+       CourseModel.getByInitials(initials, (err,courseArray) => {
+               if (!!err)
+                       return res.json(err);
+               access.getUser(req, res, (err2,user) => {
+                       const isTeacher = !!user && user.initials == initials;
+                       // Strip students from courses if not course admin (TODO: not required in any case)
+                       if (!isTeacher)
+                       {
+                               courseArray.forEach( c => {
+                                       delete c["students"];
+                               });
+                       }
+                       res.render("course-list", {
+                               title: initials + " courses",
+                               courseArray: courseArray,
+                               teacher: isTeacher,
+                               initials: initials,
+                       });
+               });
+       });
+});
+
+// Detailed content of one course
+router.get("/:initials([a-z0-9]+)/:courseCode([a-z0-9._-]+)", (req,res) => {
+       let initials = req.params["initials"];
+       let code = req.params["courseCode"];
+       CourseModel.getByRefs(initials, code, (err,course) => {
+               access.checkRequest(res, err, course, "Course not found", () => {
+                       AssessmentEntity.getByCourse(course._id, (err2,assessmentArray) => {
+                               if (!!err)
+                                       return res.json(err);
+                               access.getUser(req, res, (err2,user) => {
+                                       const isTeacher = !!user && user.initials == initials;
+                                       // Strip students from course if not course admin
+                                       if (!isTeacher)
+                                               delete course["students"];
+                                       res.render("course", {
+                                               title: "course " + initials + "/" + code,
+                                               course: course,
+                                               assessmentArray: assessmentArray,
+                                               teacher: isTeacher,
+                                               initials: initials,
+                                       });
+                               });
+                       });
+               });
+       });
+});
+
+// Display assessment (exam or open status)
+router.get("/:initials([a-z0-9]+)/:courseCode([a-z0-9._-]+)/:assessmentName([a-z0-9._-]+)", (req,res) => {
+       let initials = req.params["initials"];
+       let code = req.params["courseCode"];
+       let name = req.params["assessmentName"];
+       AssessmentModel.getByRefs(initials, code, name, (err,assessment) => {
+               access.checkRequest(res, err, assessment, "Assessment not found", () => {
+                       if (!assessment.active)
+                               return res.json({errmsg: "Assessment is idle"});
+                       delete assessment["papers"]; //always remove recorded students answers
+                       if (assessment.mode == "exam")
+                       {
+                               if (!!req.headers['user-agent'].match(/(SpecialAgent|HeadlessChrome|PhantomJS)/))
+                               {
+                                       // Basic headless browser detection
+                                       return res.json({errmsg: "Headless browser detected"});
+                               }
+                               // Strip conclusion + questions if exam mode (stepwise process)
+                               delete assessment["conclusion"];
+                               delete assessment["questions"];
+                       }
+                       res.render("assessment", {
+                               title: "assessment " + initials + "/" + code + "/" + name,
+                               assessment: assessment,
+                       });
+               });
+       });
+});
+
+// Monitor: --> after identification (password), always send password hash with requests
+router.get("/:initials([a-z0-9]+)/:courseCode([a-z0-9._-]+)/:assessmentName([a-z0-9._-]+)/monitor", (req,res) => {
+       let initials = req.params["initials"];
+       let code = req.params["courseCode"];
+       let name = req.params["assessmentName"];
+       res.render("monitor", {
+               title: "monitor assessment " + code + "/" + name,
+               initials: initials,
+               code: code,
+               name: name,
+       });
+});
+
+module.exports = router;
diff --git a/routes/users.js b/routes/users.js
new file mode 100644 (file)
index 0000000..c42b447
--- /dev/null
@@ -0,0 +1,117 @@
+let router = require("express").Router();
+const validator = require('../public/javascripts/utils/validation');
+const UserModel = require('../models/user');
+const UserEntity = require('../entities/user');
+const maild = require('../utils/mailer');
+const TokenGen = require("../utils/tokenGenerator");
+const access = require("../utils/access");
+const params = require("../config/parameters");
+
+// to: object user
+function sendLoginToken(subject, to, res)
+{
+       // Set login token and send welcome(back) email with auth link
+       let token = TokenGen.generate(params.token.length);
+       UserEntity.setLoginToken(token, to._id, to.ip, (err,ret) => {
+               access.checkRequest(res, err, ret, "Cannot set login token", () => {
+                       maild.send({
+                               from: params.mail.from,
+                               to: to.email,
+                               subject: subject,
+                               body: "Hello " + to.initials + "!\n" +
+                                       "Access your account here: " +
+                                       params.siteURL + "/authenticate?token=" + token + "\\n" +
+                                       "Token will expire in " + params.token.expire/(1000*60) + " minutes."
+                       }, err => {
+                               res.json(err || {});
+                       });
+               });
+       });
+}
+
+router.get('/register', access.ajax, access.unlogged, (req,res) => {
+       let email = decodeURIComponent(req.query.email);
+       let forename = decodeURIComponent(req.query.forename);
+       let name = decodeURIComponent(req.query.name);
+       const newUser = {
+               email: email,
+               name: name,
+               forename: forename,
+       };
+       let error = validator(newUser, "User");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       if (!UserModel.whitelistCheck(newUser.email))
+               return res.json({errmsg: "Email not in whitelist"});
+       UserEntity.getByEmail(newUser.email, (err,user0) => {
+               access.checkRequest(res, err, !user0?["ok"]:{}, "An account exists with this email", () => {
+                       UserModel.create(newUser, (err,user) => {
+                               access.checkRequest(res, err, user, "Registration failed", () => {
+                                       user.ip = req.ip;
+                                       sendLoginToken("Welcome to " + params.siteURL, user, res);
+                               });
+                       });
+               });
+       });
+});
+
+// Login:
+router.get('/sendtoken', access.ajax, access.unlogged, (req,res) => {
+       let email = decodeURIComponent(req.query.email);
+       let error = validator({email:email}, "User");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       UserEntity.getByEmail(email, (err,user) => {
+               access.checkRequest(res, err, user, "Unknown user", () => {
+                       user.ip = req.ip;
+                       sendLoginToken("Token for " + params.siteURL, user, res);
+               });
+       });
+});
+
+// Authentication process, optionally with email changing:
+router.get('/authenticate', access.unlogged, (req,res) => {
+       let loginToken = req.query.token;
+       let error = validator({token:loginToken}, "User");
+       if (error.length > 0)
+               return res.json({errmsg:error});
+       UserEntity.getByLoginToken(loginToken, (err,user) => {
+               access.checkRequest(res, err, user, "Invalid token", () => {
+                       if (user.loginToken.ip != req.ip)
+                               return res.json({errmsg: "IP address mismatch"});
+                       let now = new Date();
+                       let tsNow = now.getTime();
+                       // If token older than params.tokenExpire, do nothing
+                       if (user.loginToken.timestamp + params.token.expire < tsNow)
+                               return res.json({errmsg: "Token expired"});
+                       // Generate and update session token + destroy login token
+                       let token = TokenGen.generate(params.token.length);
+                       UserEntity.setSessionToken(token, user._id, (err,ret) => {
+                               access.checkRequest(res, err, ret, "Authentication failed", () => {
+                                       // Set cookies and redirect to user main control panel
+                                       res.cookie("token", token, {
+                                               httpOnly: true,
+                                               maxAge: params.cookieExpire,
+                                       });
+                                       res.cookie("initials", user.initials, {
+                                               httpOnly: true,
+                                               maxAge: params.cookieExpire,
+                                       });
+                                       res.redirect("/" + user.initials);
+                               });
+                       });
+               });
+       });
+});
+
+router.get('/logout', access.logged, (req,res) => {
+       UserModel.logout(req.user._id, req.cookies.token, (err,ret) => {
+               access.checkRequest(res, err, ret, "Logout failed", () => {
+                       res.clearCookie("initials");
+                       res.clearCookie("token");
+                       res.redirect('/');
+               });
+       });
+});
+
+module.exports = router;
diff --git a/setup/README b/setup/README
new file mode 100644 (file)
index 0000000..473b00e
--- /dev/null
@@ -0,0 +1,24 @@
+## Prerequisites
+
+node (v.6.10+), npm, mongo (v3.4+)
+
+msmtp: for sending e-mails
+
+git-fat: for large binary objects
+
+## Local installation
+
+First of all:
+    npm i
+
+Copy the files config/\*.js.dist without .dist extension.
+Adjust their content to your liking (especially DB name and user & mail settings).
+
+Then start mongodb service, create a database and user corresponding to the parameters,
+and run the mongo script database.js in setup/ folder:
+    load("setup/database.js") [from mongo shell]
+
+All should be good now:
+    npm start
+
+Note the "student.csv.sample" file in this folder, to test features locally.
diff --git a/setup/database.js b/setup/database.js
new file mode 100644 (file)
index 0000000..b9a392d
--- /dev/null
@@ -0,0 +1,13 @@
+// TODO: createCollections users, courses, assessments
+// with:
+// users
+//   unique initials, email
+//   index initials, email
+// courses
+//   unique (code,uid)
+//   index (code,uid)
+// assessments
+//   unique (cid, name)
+//   index (cid, name)
+// db.assessments.createIndex( { cid: 1, name: 1 } );
+// https://docs.mongodb.com/manual/core/index-compound/
diff --git a/setup/students.sample.csv b/setup/students.sample.csv
new file mode 100644 (file)
index 0000000..91d1105
--- /dev/null
@@ -0,0 +1,28 @@
+forename,name
+William,Plaisance
+Rosamonde,Dupuy
+Fabienne,Aubin
+Grégoire,Léveillé
+Fantina,Quinn
+Amitee,Morneau
+William,Lespérance
+Dreux,Vadeboncoeur
+Brigliador,Cadieux
+Grosvenor,Provencher
+Landers,Devost
+Joy,Laprise
+Dielle,Séguin
+Sidney,DeGrasse
+Jules,Gaillou
+Archaimbau,Bizier
+Dexter,Aucoin
+Searlas,Rivière
+Germain,Charpie
+Honore,Charlesbois
+Georges,LaCaille
+Anne,Ruest
+Searlas,Rochefort
+Ferrau,Adler
+Aimée,Asselin
+Delit,Cyr
+Noël,Babin
diff --git a/sockets.js b/sockets.js
new file mode 100644 (file)
index 0000000..38a9929
--- /dev/null
@@ -0,0 +1,87 @@
+var message = require("./public/javascripts/utils/socketMessages.js");
+const params = require("./config/parameters");
+
+// TODO: when teacher connect on monitor, io.of("appropriate namespace").on(connect student) { ... }
+// --> 2 sockets on monitoring page: one with ns "/" et one dedicated to the exam, triggered after the first
+// --> The monitoring page should not be closed during exam (otherwise monitors won't receive any more data)
+
+function quizzRoom(socket) {
+       let students = { };
+
+       // Student or monitor stuff
+       const isTeacher = !!socket.handshake.query.secret && socket.handshake.query.secret == params.secret;
+
+       if (isTeacher)
+       {
+               // TODO: on student disconnect, too
+               socket.on(message.newAnswer, m => { //got answer from student
+                       socket.emit(message.newAnswer, m);
+               });
+               socket.on(message.socketFeedback, m => { //send feedback to student (answers)
+                       if (!!students[m.number])
+                               socket.broadcast.to(students[m.number]).emit(message.newFeedback, { feedback:m.feedback });
+               });
+               socket.on("disconnect", m => {
+                       // Reset student array if no more active teacher connections (TODO: condition)
+                       students = { };
+               });
+       }
+
+       else //student
+       {
+               const number = socket.handshake.query.number;
+               const password = socket.handshake.query.password;
+               // Prevent socket connection (just ignore) if student already connected
+               if (!!students[number] && students[number].password != password)
+                       return;
+               students[number] = {
+                       sid: socket.id,
+                       password: password,
+               };
+               socket.on(message.newFeedback, () => { //got feedback from teacher
+                       socket.emit(message.newFeedback, m);
+               });
+               // NOTE: nothing on disconnect --> teacher disconnect trigger students cleaning
+       }
+}
+
+module.exports = function(io) {
+
+       // NOTE: if prof connected with 2 tabs and close 1, quizz should not break, thus following counter
+       let namespaces = { };
+
+       io.of("/").on("connection", socketProf => {
+               function closeQuizz(fullPath) {
+                       namespaces[fullPath].counter--;
+                       if (namespaces[fullPath].counter == 0)
+                       {
+                               // https://stackoverflow.com/questions/26400595/socket-io-how-do-i-remove-a-namespace
+                               const connectedSockets = Object.keys(namespaces[fullPath].nsp.connected);
+                               connectedSockets.forEach( sid => {
+                                       namespaces[fullPath].nsp.connected[sid].disconnect();
+                               });
+                               namespaces[fullPath].nsp.removeAllListeners();
+                               delete io.nsps[fullPath];
+                       }
+               }
+               // Only prof account can connect default namespace
+               socketProf.on(message.startQuizz, m => {
+                       // m contient quizz ID + fullPath (initials+path+name)
+                       const quizzNamespace = io.of(m.fullPath);
+                       if (!namespaces[m.fullPath])
+                       {
+                               namespaces[m.fullPath] = { nsp:quizzNamespace, counter:1 };
+                               quizzNamespace.on("connection", quizzRoom); //après ça : prof can connect in quizz too
+                               socketProf.emit(message.quizzReady);
+                               socketProf.on(message.endQuizz, m2 => {
+                                       closeQuizz(m.fullPath);
+                               });
+                               socketProf.on("disconnect", m2 => {
+                                       closeQuizz(m.fullPath); //TODO: this should delete all students in array
+                               });
+                       }
+                       else
+                               namespaces[m.fullPath]++;
+               });
+       });
+}
diff --git a/utils/access.js b/utils/access.js
new file mode 100644 (file)
index 0000000..1f91724
--- /dev/null
@@ -0,0 +1,55 @@
+const _ = require("underscore");
+const UserEntity = require("../entities/user");
+
+let Access =
+{
+       getUser: function(req, res, callback)
+       {
+               if (!res.locals.loggedIn)
+                       return callback({errmsg: "Not logged in!"}, undefined);
+               UserEntity.getBySessionToken(req.cookies.token, function(err, user) {
+                       if (!user)
+                               return callback({errmsg: "Not logged in!"}, undefined);
+                       return callback(null, user);
+               });
+       },
+
+       // Before loading sensible content, check + save credentials
+       logged: function(req, res, next)
+       {
+               Access.getUser(req, res, (err,user) => {
+                       if (!!err)
+                               return res.json(err);
+                       req.user = user;
+                       next();
+               });
+       },
+
+       // Prevent access to "anonymous pages"
+       unlogged: function(req, res, next)
+       {
+               if (!!req.user)
+                       return res.json({errmsg: "Already logged in!"});
+               next();
+       },
+
+       // Prevent direct access to AJAX results
+       ajax: function(req, res, next)
+       {
+               if (!req.xhr)
+                       return res.json({errmsg: "Unauthorized access"});
+               next();
+       },
+
+       // Check for errors before callback (continue page loading). TODO: better name.
+       checkRequest: function(res, err, out, msg, cb)
+       {
+               if (!!err)
+                       return res.json(err);
+               if (!out || _.isEmpty(out))
+                       return res.json({errmsg: msg});
+               cb();
+       },
+};
+
+module.exports = Access;
diff --git a/utils/database.js b/utils/database.js
new file mode 100644 (file)
index 0000000..542588d
--- /dev/null
@@ -0,0 +1,9 @@
+const mongojs = require("mongojs");
+const params = require("../config/parameters");
+
+const connectionString =
+       params.db.user + ":" + params.db.password + "@" + params.db.host + ":" + params.db.port + "/" + params.db.name
+
+const db = mongojs(connectionString);
+
+module.exports = db;
diff --git a/utils/mailer.js b/utils/mailer.js
new file mode 100644 (file)
index 0000000..bcd3dc5
--- /dev/null
@@ -0,0 +1,42 @@
+const params = require("../config/parameters");
+const { exec } = require('child_process');
+
+let Mailer =
+{
+       // o: {from(*), to(*), subject, body} - (*): mandatory
+       send: function(o, callback)
+       {
+               let from = o.from;
+               let to = o.to;
+               let subject = !!o.subject ? o.subject : "[No subject]";
+               let body = !!o.body ? o.body : "";
+
+               // In development mode, just log message:
+               let env = process.env.NODE_ENV || 'development';
+               if ('development' === env)
+               {
+                       console.log("New mail: from " + from + " / to " + to);
+                       console.log("Subject: " + subject);
+                       let msgText = body.split('\\n');
+                       msgText.forEach(msg => { console.log(msg); });
+                       callback({});
+               }
+               else
+               {
+                       exec(
+                               "printf 'From: " + from + "\n" +
+                                       "To: " + to + "\n" +
+                                       "Subject: " + subject + "\n" +
+                                       body + "' | msmtp -a " + params.mail.account + " " + to,
+                               (err, stdout, stderr) => {
+                                       callback(err);
+                                       // the *entire* stdout and stderr (buffered)
+                                       //console.log("stdout: " + stdout);
+                                       //console.log("stderr: " + stderr);
+                               }
+                       );
+               }
+       }
+};
+
+module.exports = Mailer;
diff --git a/utils/tokenGenerator.js b/utils/tokenGenerator.js
new file mode 100644 (file)
index 0000000..054398e
--- /dev/null
@@ -0,0 +1,18 @@
+let TokenGen =
+{
+       rand: function()
+       {
+               return Math.random().toString(36).substr(2); // remove `0.`
+       },
+
+       generate: function(tlen)
+       {
+               var res = "";
+               var nbRands = Math.ceil(tlen/10); //10 = min length of a rand() string
+               for (var i = 0; i < nbRands; i++)
+                       res += TokenGen.rand();
+               return res.substr(0, tlen);
+       },
+};
+
+module.exports = TokenGen;
diff --git a/views/assessment.pug b/views/assessment.pug
new file mode 100644 (file)
index 0000000..9caa03d
--- /dev/null
@@ -0,0 +1,66 @@
+extends withQuestions
+
+block append stylesheets
+       link(rel="stylesheet" href="/stylesheets/assessment.css")
+       noscript
+               meta(http-equiv="Refresh" content="0; URL=/enablejs")
+
+block rightMenu
+       a#rightButton.btn-floating.btn-large.grey(href=assessment.name + "/monitor")
+               i.material-icons video_label
+
+block content
+       .container#assessment
+               .row
+                       #warning.modal
+                               .modal-content
+                                       p Your answer to the current question was sent to the server.
+                                       p To avoid future unpleasant surprises, please don't
+                                       ul
+                                               li resize the window, or
+                                               li lose window focus.
+                               .modal-footer
+                                       .center-align
+                                               a.modal-action.modal-close.waves-effect.waves-light.btn-flat(href="#!") Got it!
+               .row
+                       .col.s12.m10.offset-m1.l8.offset-l2.xl6.offset-xl3
+                               h4= assessment.name
+                               #stage0(v-if="stage==0")
+                                       .card
+                                               .input-field.inline.on-left
+                                                       label(for="number") Number
+                                                       input#number(type="text" v-model="student.number" @keyup.enter="getStudent()")
+                                               button.waves-effect.waves-light.btn(@click="getStudent()") Send
+                               #stage1(v-if="stage==1")
+                                       .card
+                                               if assessment.mode != "open"
+                                                       .input-field.inline.on-left
+                                                               label(for="forename") Forename
+                                                               input#forename(type="text" v-model="student.forename" disabled)
+                                                       .input-field.inline
+                                                               label(for="name") Name
+                                                               input#name(type="text" v-model="student.name" disabled)
+                                               p
+                                                       if assessment.mode != "open"
+                                                               button.waves-effect.waves-light.btn.on-left(@click="cancelStudent") Cancel
+                                                       button.waves-effect.waves-light.btn(@click="startAssessment") Start!
+                               #stage1_2_4(v-if="stage==1 || stage==2 || stage == 4")
+                                       .card
+                                               .introduction(v-html="assessment.introduction")
+                               #stage2_4(v-if="stage==2 || stage==4")
+                                       if assessment.time > 0
+                                               .card
+                                                       .timer.center(v-if="stage==2") {{ countdown }}
+                                       .card
+                                               statements(:assessment="assessment" :student="student" :stage="stage" :inputs="inputs" @gameover="endAssessment")
+                               #stage3(v-if="stage==3")
+                                       .card
+                                               .finish Exam completed &#9786; ...don't close the window!
+                               #stage3_4(v-if="stage==3 || stage==4")
+                                       .card
+                                               .conclusion(v-html="assessment.conclusion")
+
+block append javascripts
+       script.
+               let assessment = !{JSON.stringify(assessment)};
+       script(src="/javascripts/assessment.js")
diff --git a/views/course-list.pug b/views/course-list.pug
new file mode 100644 (file)
index 0000000..5ab2dcf
--- /dev/null
@@ -0,0 +1,48 @@
+extends layout
+
+block stylesheets
+       link(rel="stylesheet" href="/stylesheets/courseList.css")
+
+block content
+       .container#courseList
+               if teacher
+                       .row
+                               // Modal Structure
+                               #newCourse.modal
+                                       .modal-content
+                                               form(@submit.prevent="submit")
+                                                       .input-field
+                                                               input#code.validate(type="text" v-model="newCourse.code" autofocus required)
+                                                               label(for="code") Code
+                                                       .input-field
+                                                               input#description.validate(type="text" v-model="newCourse.description" required)
+                                                               label(for="description") Description
+                                       .modal-footer
+                                               #submit.center-align
+                                               a.waves-effect.waves-light.btn(href="#!" @click="addCourse()")
+                                                       span Submit
+                                                       i.material-icons.right send
+               .row
+                       .col.s12.m10.offset-m1.l8.offset-l2.xl6.offset-xl3
+                               .card
+                                       if teacher
+                                               // Modal Trigger
+                                               .center-align
+                                                       a.waves-effect.waves-light.btn.modal-trigger(href="#newCourse") New course
+                                       table
+                                               thead
+                                                       tr
+                                                               th Code
+                                                               th Description
+                                               tbody
+                                                       tr.course(v-for="course in courseArray" @click.left="redirect(course.code)" @contextmenu.prevent="deleteCourse(course)")
+                                                               td {{ course.code }}
+                                                               td(v-html="course.description")
+
+block javascripts
+       script(src="/javascripts/courseList.js")
+       script(src="/javascripts/utils/validation.js")
+       script.
+               let courseArray = !{JSON.stringify(courseArray)};
+               const initials = "#{initials}";
+               const admin = #{teacher};
diff --git a/views/course.pug b/views/course.pug
new file mode 100644 (file)
index 0000000..16552ce
--- /dev/null
@@ -0,0 +1,197 @@
+extends withQuestions
+
+block append stylesheets
+       link(rel="stylesheet" href="/stylesheets/course.css")
+
+block content
+       .container#course
+               if teacher
+                       #newAssessment.modal
+                               .modal-content
+                                       form(@submit.prevent="addAssessment")
+                                               .input-field
+                                                       input#assessmentName(type="text" v-model="newAssessment.name" required)
+                                                       label(for="assessmentName") Name
+                               .modal-footer
+                                       .center-align
+                                               a.waves-effect.waves-light.btn(href="#!" @click="addAssessment()")
+                                                       span Submit
+                                                       i.material-icons.right send
+                       #assessmentSettings.modal
+                               .modal-content
+                                       form
+                                               p
+                                                       input#active(type="checkbox" v-model="assessment.active")
+                                                       label(for="active") Active
+                                               p
+                                                       input#secure(name="status" type="radio" value="secure" v-model="assessment.mode")
+                                                       label(for="secure") Exam mode, secured (class only)
+                                               p
+                                                       input#exam(name="status" type="radio" value="exam" v-model="assessment.mode")
+                                                       label(for="exam") Exam mode, free (class only)
+                                               p
+                                                       input#open(name="status" type="radio" value="open" v-model="assessment.mode")
+                                                       label(for="open") Open to everyone
+                                               p
+                                                       input#fixed(type="checkbox" v-model="assessment.fixed")
+                                                       label(for="fixed") Fixed questions order
+                                               p
+                                                       input#displayOne(name="display" type="radio" value="one" v-model="assessment.display")
+                                                       label(for="displayOne") One question at a time
+                                               p
+                                                       input#displayAll(name="display" type="radio" value="all" v-model="assessment.display")
+                                                       label(for="displayAll") Display all questions
+                                               .input-field
+                                                       input#time(type="number" v-model.number="assessment.time")
+                                                       label(for="time") Time (minutes)
+                               .modal-footer
+                                       .center-align
+                                               a.modal-action.modal-close.waves-effect.waves-light.btn-flat(href="#!") Done
+                       #assessmentEdit.modal
+                               .modal-content
+                                       form
+                                               .input-field
+                                                       textarea#introduction.materialize-textarea(v-model="assessment.introduction")
+                                                       label(for="introduction") Introduction
+                                               .input-field
+                                                       textarea#assessmentEdition.materialize-textarea(v-model="assessmentText")
+                                                       label(for="assessmentEdition") Assessment in text format
+                                               .input-field
+                                                       textarea#conclusion.materialize-textarea(v-model="assessment.conclusion")
+                                                       label(for="conclusion") Conclusion
+                               .modal-footer
+                                       .center-align
+                                               a.modal-action.modal-close.waves-effect.waves-light.btn-flat(href="#!") Done
+                       #gradeSettings.modal
+                               .modal-content
+                                       form(@submit.prevent="computeGrades")
+                                               .input-field
+                                                       input#points(type="number" v-model.number="settings.totalPoints" required)
+                                                       label(for="points") Total points
+                                               p
+                                                       input#partial(type="checkbox" v-model="settings.halfPoint")
+                                                       label(for="partial") Half point for partial answers? (&ge; 50%)
+                                               p
+                                                       input#malus(type="checkbox" v-model="settings.zeroSum")
+                                                       label(for="malus") Lose points on wrong answers? ("Zero-sum" game)
+                               .modal-footer
+                                       .center-align
+                                               a.modal-action.modal-close.waves-effect.waves-light.btn(href="#!" @click="computeGrades()")
+                                                       span Compute
+                                                       i.material-icons.right send
+                       #detailedGrades.modal
+                               .modal-content
+                                       table
+                                               thead
+                                                       tr
+                                                               th Number
+                                                               th(v-for="assessment in assessmentArray") {{ assessment.name }}
+                                               tbody
+                                                       tr.grade(v-for="student in studentList(group)")
+                                                               td {{ student.number }}
+                                                               td(v-for="(assessment,i) in assessmentArray" @click="togglePresence(student.number,i)") {{ grade(i,student.number) }}
+                               .modal-footer
+                                       .center-align
+                                               a.modal-action.modal-close.waves-effect.waves-light.btn-flat(href="#!") Close
+               .row(v-show="mode=='view'")
+                       .col.s12.m10.offset-m1.l8.offset-l2.xl6.offset-xl3
+                               if teacher
+                                       h4.title(@click="toggleDisplay('students')") Students
+                                       .card(v-show="display=='students'")
+                                               .center-align
+                                                       input.hide#upload(type="file" @change="upload")
+                                                       button.on-left.waves-effect.waves-light.btn(@click="uploadTrigger()") Import
+                                               table
+                                                       thead
+                                                               tr
+                                                                       th Number
+                                                                       th Forename
+                                                                       th Name
+                                                                       th Group
+                                                       tbody
+                                                               tr.student(v-for="student in studentList(0)")
+                                                                       td {{ student.number }}
+                                                                       td {{ student.forename }}
+                                                                       td {{ student.name }}
+                                                                       td {{ student.group }}
+                               h4.title(@click="toggleDisplay('assessments')") Assessments
+                               .card(v-show="display=='assessments'")
+                                       if teacher
+                                               .center-align
+                                                       a.on-left.waves-effect.waves-light.btn.modal-trigger(href="#newAssessment") New assessment
+                                                       input#password(type="password" v-model="monitorPwd" @keyup.enter="setPassword" placeholder="Password" title="Monitoring password")
+                                       table
+                                               thead
+                                                       tr
+                                                               th Name
+                                                               th Coefficient
+                                                               th #Questions
+                                                               th Time
+                                               tbody
+                                                       tr.assessment(v-for="(assessment,i) in assessmentArray" @click.left="actionAssessment(i)" @contextmenu.prevent="deleteAssessment(assessment)")
+                                                               td {{ assessment.name }}
+                                                               td {{ assessment.coefficient }}
+                                                               td {{ assessment.questions.reduce( (a,b) => { return b.active ? a+1 : a; }, 0) }}
+                                                               td {{ assessment.time }}
+                               if teacher
+                                       h4.title(@click="toggleDisplay('grades')") Grades
+                                       .card(v-show="display=='grades'")
+                                               .center-align
+                                                       button.on-left.waves-effect.waves-light.btn(@click="gradeSettings()") Settings
+                                                       a#download.hide(href="#" ref="download")
+                                                       button.waves-effect.waves-light.btn(@click="download") Download
+                                               ul.tabs.tabs-fixed-width
+                                                       li.tab
+                                                               a(href="#group0") All
+                                                       li.tab(v-for="group in groupList()")
+                                                               a(:href="groupId(group,'hash')") G.{{ group }}
+                                               table.result(:id="groupId(group)" v-for="group in [0].concat(groupList())" @click="showDetails(group)")
+                                                       thead
+                                                               tr
+                                                                       th Number
+                                                                       th Forename
+                                                                       th Name
+                                                                       th Final
+                                                       tbody
+                                                               tr.grade(v-for="student in studentList(group)")
+                                                                       td {{ student.number }}
+                                                                       td {{ student.forename }}
+                                                                       td {{ student.name }}
+                                                                       td grade...
+                                                                       //td {{ grades[student.number].final }}
+                                                               tr.stats
+                                                                       td(colspan="4") Stats: range= stdev= mean=
+               if teacher
+                       .row(v-show="mode=='edit'")
+                               .col.s12.m10.offset-m1.l8.offset-l2.xl6.offset-xl3
+                                       h4 {{ assessment.name }}
+                                       .card
+                                               .center-align
+                                                       button.waves-effect.waves-light.btn.on-left(@click="materialOpenModal('assessmentSettings')") Settings
+                                                       button.waves-effect.waves-light.btn.on-left(@click="materialOpenModal('assessmentEdit')") Content
+                                                       button.waves-effect.waves-light.btn(@click="redirect(assessment.name)") View
+                                               #questionList
+                                                       .introduction(v-html="assessment.introduction")
+                                                       .question(v-for="(question,i) in assessment.questions" :class="{questionInactive:!question.active}")
+                                                               .wording(v-html="question.wording")
+                                                               .option(v-for="(option,j) in question.options" :class="{choiceCorrect:question.answer.includes(j)}" v-html="option")
+                                                               p
+                                                                       input(:id="checkBoxFixedId(i)" type="checkbox" v-model="question.fixed")
+                                                                       label.on-left(:for="checkBoxFixedId(i)") Fixed
+                                                                       input(:id="checkBoxActiveId(i)" type="checkbox" v-model="question.active")
+                                                                       label(:for="checkBoxActiveId(i)") Active
+                                                       .conclusion(v-html="assessment.conclusion")
+                                               .center-align
+                                                       button.waves-effect.waves-light.btn.on-left(@click="mode='view'") Cancel
+                                                       button.waves-effect.waves-light.btn(@click="updateAssessment") Send
+
+block append javascripts
+       script(src="//cdnjs.cloudflare.com/ajax/libs/PapaParse/4.3.6/papaparse.min.js")
+       script(src="/javascripts/utils/sha1.js")
+       script(src="/javascripts/utils/validation.js")
+       script.
+               let assessmentArray = !{JSON.stringify(assessmentArray)};
+               const course = !{JSON.stringify(course)};
+               const initials = "#{initials}";
+               const admin = #{teacher};
+       script(src="/javascripts/course.js")
diff --git a/views/enable-js.pug b/views/enable-js.pug
new file mode 100644 (file)
index 0000000..f05a445
--- /dev/null
@@ -0,0 +1,4 @@
+extends layout
+
+block content
+       p.warn Javascript must be enabled. Change settings, and reload exam page.
diff --git a/views/error.pug b/views/error.pug
new file mode 100644 (file)
index 0000000..2baa3e2
--- /dev/null
@@ -0,0 +1,11 @@
+doctype html
+html
+
+       head
+               meta(charset="utf-8")
+               title Error
+
+       body
+               h1= message
+               h2= error.status
+               pre #{error.stack}
diff --git a/views/index.pug b/views/index.pug
new file mode 100644 (file)
index 0000000..70ac099
--- /dev/null
@@ -0,0 +1,19 @@
+extends layout
+
+block stylesheets
+       link(rel="stylesheet", href="/stylesheets/index.css")
+
+block content
+       .container
+               .row
+                       .card.col.s12.m8.offset-m2.l6.offset-l3.xl4.offset-xl4
+                               table
+                                       thead
+                                               tr
+                                                       th Name
+                                                       th Forename
+                                       tbody
+                                               each user in userArray
+                                                       tr.teacher(onClick="document.location.href='/" + user.initials+ "'")
+                                                               td= user.name
+                                                               td= user.forename
diff --git a/views/layout.pug b/views/layout.pug
new file mode 100644 (file)
index 0000000..a547638
--- /dev/null
@@ -0,0 +1,60 @@
+doctype html
+html(lang="en")
+
+       head
+               meta(charset="UTF-8")
+               title qomet - #{title}
+               meta(name="viewport", content="width=device-width, initial-scale=1")
+               link(rel="stylesheet", href="//fonts.googleapis.com/icon?family=Material+Icons")
+               link(rel="stylesheet", href="//cdnjs.cloudflare.com/ajax/libs/materialize/0.100.2/css/materialize.min.css")
+               link(rel="stylesheet" href="/stylesheets/layout.css")
+               // favicon:
+               link(rel="apple-touch-icon", sizes="180x180", href="/favicon/apple-touch-icon.png")
+               link(rel="icon", type="image/png", sizes="32x32", href="/favicon/favicon-32x32.png")
+               link(rel="icon", type="image/png", sizes="16x16", href="/favicon/favicon-16x16.png")
+               link(rel="manifest", href="/favicon/manifest.json")
+               link(rel="mask-icon", href="/favicon/safari-pinned-tab.svg", color="#5bbad5")
+               link(rel="shortcut icon", href="/favicon/favicon.ico")
+               meta(name="msapplication-config", content="/favicon/browserconfig.xml")
+               meta(name="theme-color", content="#ffffff")
+               // -- end favicon
+               block stylesheets
+
+       body
+
+               header
+                       // Top-left menu
+                       button#leftButton.dropdown-button.btn.btn-floating.btn-large.waves-effect.waves-light.grey(data-activates="leftMenu" data-constrainwidth="false")
+                               i.material-icons menu
+                       ul#leftMenu.userMenu.dropdown-content
+                               li
+                                       a(href="/")
+                                               i.material-icons home
+                                               span Home
+                               if loggedIn
+                                       li
+                                               a(href="/" + myInitials)
+                                                       i.material-icons question_answer
+                                                       span Courses
+                               li.divider
+                               if loggedIn
+                                       li
+                                               a(href="/logout")
+                                                       i.material-icons power_settings_new
+                                                       span Logout
+                               else
+                                       li
+                                               a(href="/login")
+                                                       i.material-icons account_box
+                                                       span Login
+                       block rightMenu
+
+               main
+                       block content
+
+               script(src="//cdnjs.cloudflare.com/ajax/libs/underscore.js/1.8.3/underscore-min.js")
+               script(src="//code.jquery.com/jquery-3.2.1.min.js")
+               script(src="//cdnjs.cloudflare.com/ajax/libs/materialize/0.100.2/js/materialize.min.js")
+               script(src="//cdnjs.cloudflare.com/ajax/libs/vue/2.5.2/vue.js")
+               script(src="/javascripts/utils/socketMessages.js")
+               block javascripts
diff --git a/views/login.pug b/views/login.pug
new file mode 100644 (file)
index 0000000..215f83f
--- /dev/null
@@ -0,0 +1,33 @@
+extends layout
+
+block stylesheets
+       link(rel="stylesheet", href="/stylesheets/login.css")
+
+block content
+       .container#login
+               .row
+                       .col.s12.m8.offset-m2.l6.offset-l3.xl4.offset-xl4
+                               .card#form
+                                       form(@submit.prevent="submit")
+                                               .input-field
+                                                       input#email.validate(type="email", ref="userEmail", v-model="user.email", required)
+                                                       label(for="email") Email
+                                               .input-field(v-show="stage=='register'")
+                                                       input#forename.validate(type="text", v-model="user.forename", :required="stage=='register'")
+                                                       label(for="forename") Forename
+                                               .input-field(v-show="stage=='register'")
+                                                       input#name.validate(type="text", v-model="user.name", :required="stage=='register'")
+                                                       label(for="name") Name
+                                               #submit.center-align
+                                                       button#submit.waves-effect.waves-light.btn(@click.prevent="submit")
+                                                               span {{ messages[stage] }}
+                                                               i.material-icons.right send
+                                               #toggle.center-align
+                                                       span(v-show="stage!='login'", @click="toggleStage('login')") Login
+                                                       span(v-show="stage!='register'", @click="toggleStage('register')") Register
+                               .card#dialog.hide
+
+block javascripts
+       script(src="/javascripts/utils/dialog.js")
+       script(src="/javascripts/utils/validation.js")
+       script(src="/javascripts/login.js")
diff --git a/views/monitor.pug b/views/monitor.pug
new file mode 100644 (file)
index 0000000..e5dadfe
--- /dev/null
@@ -0,0 +1,12 @@
+extends withQuestions
+
+       //TODO: step 1: ask password (client side, store hash)
+       // step 2: when got hash, send request (with hash) to get monitoring page:
+       //   array with results + quiz details (displayed in another tab) + init socket (with hash too)
+       //   buttons "start quiz" and "stop quiz" for teacher only: trigger actions (impacting sockets)
+
+       body
+               p TODO
+
+block append javascripts
+       script. TODO
diff --git a/views/no-devtools.pug b/views/no-devtools.pug
new file mode 100644 (file)
index 0000000..bdc6021
--- /dev/null
@@ -0,0 +1,5 @@
+extends layout
+
+block content
+       p.warn Devtools is forbidden during an exam. Exit devtools and go back to exam page.
+       p.warn NOTE: if the exam was started already, then you cannot resume it :/
diff --git a/views/withQuestions.pug b/views/withQuestions.pug
new file mode 100644 (file)
index 0000000..e835911
--- /dev/null
@@ -0,0 +1,20 @@
+extends layout
+
+block stylesheets
+       link(href="/vendor/prism/prism.css" rel="stylesheet")
+
+block javascripts
+       script(src="/vendor/prism/prism.js")
+       script.
+               Prism.plugins.autoloader.languages_path = '/vendor/prism/components';
+       script(type="text/x-mathjax-config").
+               MathJax.Hub.Config({
+                       extensions: ["tex2jax.js"],
+                       jax: ["input/TeX", "output/SVG"],
+                       tex2jax: {
+                               inlineMath: [ ['$','$'] ],
+                               displayMath: [ ['$$','$$'] ],
+                               processEscapes: true
+                       },
+               });
+       script(src='//cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.2/MathJax.js')