├── .env.sample ├── .gitignore ├── Procfile ├── app.js ├── bin └── www ├── event_handler.js ├── models ├── connection.js ├── device.js ├── device_acl.js ├── message.js └── redis.js ├── monitor.js ├── package-lock.json ├── package.json ├── public └── stylesheets │ └── style.css ├── routes ├── devices.js ├── emqx_web_hook.js ├── messages.js ├── ota.js ├── tags.js └── tokens.js ├── samples ├── business_sim.js ├── perform_ota.js ├── ping.js ├── resp_to_data_request.js ├── rpc_ping.js └── update_shadow.js ├── services ├── emqx_service.js ├── influxdb_service.js ├── message_service.js ├── notify_service.js ├── ota_service.js └── utils_service.js └── views ├── error.jade └── layout.jade /.env.sample: -------------------------------------------------------------------------------- 1 | ###运行Iothub Server需要的配置项 2 | #API端口 3 | PORT=3000 4 | #Mongodb地址 5 | MONGODB_URL=mongodb://iot:iot@localhost:27017/iothub 6 | #需要和/etc/emqx_auth_jwt.conf中一致 7 | JWT_SECRET=emqxsecret 8 | 9 | #使用EMQX REST API的账号,可通过bin/emqx_ctl mgmt insert MaqueIotHub添加 10 | EMQX_APP_ID= 11 | EMQX_APP_SECRET= 12 | 13 | #EMQX REST API地址 14 | EMQX_API_URL=http://127.0.0.1:8080/api/v3/ 15 | 16 | #Redis 17 | REDIS_URL=redis://127.0.0.1:6379 18 | 19 | #RabbitMQ 20 | RABBITMQ_URL=amqp://localhost 21 | 22 | #InfluxDB 23 | INFLUXDB=localhost 24 | 25 | ###运行Sample Code需要的环境变量 26 | TARGET_DEVICE_NAME=#设备名DeviceName 27 | TARGET_PRODUCT_NAME=#设备的产品名ProductName 28 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (https://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # TypeScript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | 60 | # parcel-bundler cache (https://parceljs.org/) 61 | .cache 62 | 63 | # next.js build output 64 | .next 65 | 66 | # nuxt.js build output 67 | .nuxt 68 | 69 | # vuepress build output 70 | .vuepress/dist 71 | 72 | # Serverless directories 73 | .serverless 74 | .idea -------------------------------------------------------------------------------- /Procfile: -------------------------------------------------------------------------------- 1 | api: ./bin/www 2 | event_handler: node event_handler.js 3 | monitor: node monitor.js -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config() 2 | var mongoose = require('mongoose'); 3 | 4 | var createError = require('http-errors'); 5 | var express = require('express'); 6 | var path = require('path'); 7 | var cookieParser = require('cookie-parser'); 8 | var logger = require('morgan'); 9 | 10 | var deviceRouter = require('./routes/devices'); 11 | var tokensRouter = require('./routes/tokens') 12 | var webHookeRouter = require('./routes/emqx_web_hook') 13 | var messageRouter = require('./routes/messages') 14 | var tagsRouter = require('./routes/tags') 15 | var otaRouter = require('./routes/ota') 16 | 17 | 18 | var app = express(); 19 | mongoose.connect(process.env.MONGODB_URL, { useNewUrlParser: true }) 20 | 21 | 22 | // view engine setup 23 | app.set('views', path.join(__dirname, 'views')); 24 | app.set('view engine', 'jade'); 25 | 26 | app.use(logger('dev')); 27 | app.use(express.json()); 28 | app.use(express.urlencoded({ extended: false })); 29 | app.use(cookieParser()); 30 | app.use(express.static(path.join(__dirname, 'public'))); 31 | 32 | app.use('/devices', deviceRouter); 33 | app.use('/tokens', tokensRouter); 34 | app.use('/emqx_web_hook', webHookeRouter) 35 | app.use('/messages', messageRouter) 36 | app.use('/tags', tagsRouter) 37 | app.use('/ota', otaRouter) 38 | 39 | // catch 404 and forward to error handler 40 | app.use(function(req, res, next) { 41 | next(createError(404)); 42 | }); 43 | 44 | // error handler 45 | app.use(function(err, req, res, next) { 46 | // set locals, only providing error in development 47 | res.locals.message = err.message; 48 | res.locals.error = req.app.get('env') === 'development' ? err : {}; 49 | 50 | // render the error page 51 | res.status(err.status || 500); 52 | res.render('error'); 53 | }); 54 | 55 | module.exports = app; 56 | -------------------------------------------------------------------------------- /bin/www: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * Module dependencies. 5 | */ 6 | 7 | var app = require('../app'); 8 | var debug = require('debug')('iothub-server:server'); 9 | var http = require('http'); 10 | 11 | /** 12 | * Get port from environment and store in Express. 13 | */ 14 | 15 | var port = normalizePort(process.env.PORT || '3000'); 16 | app.set('port', port); 17 | 18 | /** 19 | * Create HTTP server. 20 | */ 21 | 22 | var server = http.createServer(app); 23 | 24 | /** 25 | * Listen on provided port, on all network interfaces. 26 | */ 27 | 28 | server.listen(port); 29 | server.on('error', onError); 30 | server.on('listening', onListening); 31 | 32 | /** 33 | * Normalize a port into a number, string, or false. 34 | */ 35 | 36 | function normalizePort(val) { 37 | var port = parseInt(val, 10); 38 | 39 | if (isNaN(port)) { 40 | // named pipe 41 | return val; 42 | } 43 | 44 | if (port >= 0) { 45 | // port number 46 | return port; 47 | } 48 | 49 | return false; 50 | } 51 | 52 | /** 53 | * Event listener for HTTP server "error" event. 54 | */ 55 | 56 | function onError(error) { 57 | if (error.syscall !== 'listen') { 58 | throw error; 59 | } 60 | 61 | var bind = typeof port === 'string' 62 | ? 'Pipe ' + port 63 | : 'Port ' + port; 64 | 65 | // handle specific listen errors with friendly messages 66 | switch (error.code) { 67 | case 'EACCES': 68 | console.error(bind + ' requires elevated privileges'); 69 | process.exit(1); 70 | break; 71 | case 'EADDRINUSE': 72 | console.error(bind + ' is already in use'); 73 | process.exit(1); 74 | break; 75 | default: 76 | throw error; 77 | } 78 | } 79 | 80 | /** 81 | * Event listener for HTTP server "listening" event. 82 | */ 83 | 84 | function onListening() { 85 | var addr = server.address(); 86 | var bind = typeof addr === 'string' 87 | ? 'pipe ' + addr 88 | : 'port ' + addr.port; 89 | debug('Listening on ' + bind); 90 | } 91 | -------------------------------------------------------------------------------- /event_handler.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config() 2 | const bson = require('bson') 3 | var mongoose = require('mongoose'); 4 | var amqp = require('amqplib/callback_api'); 5 | var messageService = require("./services/message_service") 6 | var Device = require("./models/device") 7 | mongoose.connect(process.env.MONGODB_URL, { useNewUrlParser: true }) 8 | 9 | var addHandler = function (channel, queue, event, handlerFunc) { 10 | var exchange = "mqtt.events" 11 | channel.assertQueue(queue, { 12 | durable: true 13 | }) 14 | channel.bindQueue(queue, exchange, event) 15 | channel.consume(queue, function (msg) { 16 | handlerFunc(bson.deserialize(msg.content)) 17 | channel.ack(msg) 18 | }) 19 | } 20 | amqp.connect(process.env.RABBITMQ_URL, function (error0, connection) { 21 | if (error0) { 22 | console.log(error0); 23 | } else { 24 | connection.createChannel(function (error1, channel) { 25 | if (error1) { 26 | console.log(error1) 27 | } else { 28 | addHandler(channel, "iothub_client_connected", "client.connected", function (event) { 29 | event.connected_at = Math.floor(event.connected_at / 1000) 30 | Device.addConnection(event) 31 | }) 32 | addHandler(channel, "iothub_client_disconnected", "client.disconnected", function (event) { 33 | Device.removeConnection(event) 34 | }) 35 | addHandler(channel, "iothub_message_publish", "message.publish", function (event) { 36 | messageService.dispatchMessage({ 37 | topic: event.topic, 38 | payload: event.payload.buffer, 39 | ts: Math.floor(event.published_at / 1000) 40 | }) 41 | }) 42 | } 43 | }); 44 | } 45 | }); -------------------------------------------------------------------------------- /models/connection.js: -------------------------------------------------------------------------------- 1 | var mongoose = require('mongoose'); 2 | var Schema = mongoose.Schema; 3 | 4 | const connectionSchema = new Schema({ 5 | connected: Boolean, 6 | client_id: String, 7 | keepalive: Number, 8 | ipaddress: String, 9 | proto_ver: Number, 10 | connected_at: Number, 11 | disconnect_at: Number, 12 | conn_ack: Number, 13 | device: {type: Schema.Types.ObjectId, ref: 'Device'} 14 | }) 15 | 16 | connectionSchema.methods.toJSONObject = function () { 17 | return { 18 | connected: this.connected, 19 | client_id: this.client_id, 20 | ipaddress: this.ipaddress, 21 | connected_at: this.connected_at, 22 | disconnect_at: this.disconnect_at 23 | } 24 | } 25 | 26 | const Connection = mongoose.model("Connection", connectionSchema); 27 | 28 | module.exports = Connection 29 | -------------------------------------------------------------------------------- /models/device.js: -------------------------------------------------------------------------------- 1 | var mongoose = require('mongoose'); 2 | var Schema = mongoose.Schema; 3 | var Connection = require('./connection') 4 | var DeviceACL = require("../models/device_acl") 5 | const emqxService = require("../services/emqx_service") 6 | const influxDBService = require("../services/influxdb_service") 7 | const ObjectId = require('bson').ObjectID; 8 | 9 | 10 | const deviceSchema = new Schema({ 11 | //ProductName 12 | product_name: { 13 | type: String, 14 | required: true 15 | }, 16 | //DeviceName 17 | device_name: { 18 | type: String, 19 | required: true, 20 | }, 21 | //接入EMQX时使用的username 22 | broker_username: { 23 | type: String, 24 | required: true 25 | }, 26 | //secret 27 | secret: { 28 | type: String, 29 | required: true, 30 | }, 31 | 32 | //可接入状态 33 | status: String, 34 | device_status: { 35 | type: String, 36 | default: "{}" 37 | }, 38 | last_status_update: Number, 39 | 40 | tags: { 41 | type: Array, 42 | default: [] 43 | }, 44 | 45 | tags_version: { 46 | type: Number, 47 | default: 1 48 | }, 49 | shadow: { 50 | type: String, 51 | default: JSON.stringify({ 52 | "state": {}, 53 | "metadata": {}, 54 | "version": 0 55 | }) 56 | } 57 | }) 58 | 59 | deviceSchema.methods.toJSONObject = function () { 60 | return { 61 | product_name: this.product_name, 62 | device_name: this.device_name, 63 | secret: this.secret, 64 | device_status: JSON.parse(this.device_status), 65 | tags: this.tags, 66 | shadow: JSON.parse(this.shadow), 67 | } 68 | } 69 | 70 | deviceSchema.statics.addConnection = function (event) { 71 | if(event.username != null) { 72 | var username_arr = event.username.split("/") 73 | let productName = username_arr[0]; 74 | let deviceName = username_arr[1]; 75 | this.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 76 | if (err == null && device != null) { 77 | Connection.findOneAndUpdate({ 78 | client_id: event.client_id, 79 | device: device._id 80 | }, { 81 | connected: true, 82 | client_id: event.client_id, 83 | keepalive: event.keepalive, 84 | ipaddress: event.ipaddress, 85 | proto_ver: event.proto_ver, 86 | connected_at: event.connected_at, 87 | conn_ack: event.conn_ack, 88 | device: device._id 89 | }, {upsert: true, useFindAndModify: false, new: true}).exec() 90 | influxDBService.writeConnectionData({ 91 | productName: productName, 92 | deviceName: deviceName, 93 | connected: true, 94 | ts: event.connected_at 95 | }) 96 | } 97 | }) 98 | } 99 | 100 | } 101 | 102 | deviceSchema.statics.removeConnection = function (event) { 103 | if(event.username != null) { 104 | var username_arr = event.username.split("/") 105 | let productName = username_arr[0]; 106 | let deviceName = username_arr[1]; 107 | this.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 108 | if (err == null && device != null) { 109 | Connection.findOneAndUpdate({client_id: event.client_id, device: device._id}, 110 | { 111 | connected: false, 112 | disconnect_at: Math.floor(Date.now() / 1000) 113 | }, {useFindAndModify: false}).exec() 114 | influxDBService.writeConnectionData({ 115 | productName: productName, 116 | deviceName: deviceName, 117 | connected: false 118 | }) 119 | } 120 | }) 121 | } 122 | } 123 | 124 | deviceSchema.methods.getACLRule = function () { 125 | const publish = [ 126 | `upload_data/${this.product_name}/${this.device_name}/+/+`, 127 | `update_status/${this.product_name}/${this.device_name}/+`, 128 | `cmd_resp/${this.product_name}/${this.device_name}/+/+/+`, 129 | `rpc_resp/${this.product_name}/${this.device_name}/+/+/+`, 130 | `get/${this.product_name}/${this.device_name}/+/+`, 131 | `m2m/${this.product_name}/+/${this.device_name}/+`, 132 | `update_ota_status/${this.product_name}/${this.device_name}/+`, 133 | ] 134 | const subscribe = [`tags/${this.product_name}/+/cmd/+/+/+/#`] 135 | const pubsub = [] 136 | return { 137 | publish: publish, 138 | subscribe: subscribe, 139 | pubsub: pubsub 140 | } 141 | } 142 | 143 | deviceSchema.methods.disconnect = function () { 144 | Connection.find({device: this._id}).exec(function (err, connections) { 145 | connections.forEach(function (conn) { 146 | emqxService.disconnectClient(conn.client_id) 147 | }) 148 | }) 149 | } 150 | 151 | 152 | deviceSchema.post("remove", function (device, next) { 153 | Connection.deleteMany({device: device._id}).exec() 154 | DeviceACL.deleteMany({broker_username: device.broker_username}).exec() 155 | next() 156 | }) 157 | 158 | deviceSchema.methods.sendCommand = function ({commandName, data, encoding = "plain", ttl = undefined, commandType = "cmd", qos = 1}) { 159 | return Device.sendCommand({ 160 | productName: this.product_name, 161 | deviceName: this.device_name, 162 | commandName: commandName, 163 | data: data, 164 | encoding: encoding, 165 | ttl: ttl, 166 | commandType: commandType, 167 | qos: qos 168 | }) 169 | } 170 | 171 | deviceSchema.statics.sendCommand = function ({productName, deviceName, commandName, data, encoding = "plain", ttl = undefined, commandType = "cmd", qos = 1}) { 172 | var requestId = new ObjectId().toHexString() 173 | var topic = `${commandType}/${productName}/${deviceName}/${commandName}/${encoding}/${requestId}` 174 | if (ttl != null) { 175 | topic = `${topic}/${Math.floor(Date.now() / 1000) + ttl}` 176 | } 177 | emqxService.publishTo({topic: topic, payload: data, qos: qos}) 178 | return requestId 179 | } 180 | 181 | deviceSchema.statics.sendCommandByTag = function ({productName, tag, commandName, data, encoding = "plain", ttl = undefined, qos = 1}) { 182 | var requestId = new ObjectId().toHexString() 183 | var topic = `tags/${productName}/${tag}/cmd/${commandName}/${encoding}/${requestId}` 184 | if (ttl != null) { 185 | topic = `${topic}/${Math.floor(Date.now() / 1000) + ttl}` 186 | } 187 | emqxService.publishTo({topic: topic, payload: data, qos: qos}) 188 | } 189 | 190 | deviceSchema.methods.sendTags = function () { 191 | this.sendCommand({ 192 | commandName: "$set_tags", 193 | data: JSON.stringify({tags: this.tags || [], tags_version: this.tags_version || 1}), 194 | qos: 0 195 | }) 196 | } 197 | 198 | deviceSchema.methods.updateShadowDesired = function (desired, version) { 199 | var ts = Math.floor(Date.now() / 1000) 200 | var shadow = JSON.parse(this.shadow) 201 | if (version > shadow.version) { 202 | shadow.state.desired = shadow.state.desired || {} 203 | shadow.metadata.desired = shadow.metadata.desired || {} 204 | for (var key in desired) { 205 | shadow.state.desired[key] = desired[key] 206 | shadow.metadata.desired[key] = {timestamp: ts} 207 | } 208 | shadow.version = version 209 | shadow.timestamp = ts 210 | this.shadow = JSON.stringify(shadow) 211 | this.save() 212 | this.sendUpdateShadow() 213 | return true 214 | } else { 215 | return false 216 | } 217 | } 218 | 219 | deviceSchema.methods.sendUpdateShadow = function () { 220 | this.sendCommand({ 221 | commandName: "$update_shadow", 222 | data: this.shadow, 223 | qos: 0 224 | }) 225 | } 226 | 227 | deviceSchema.methods.updateShadow = function (shadowUpdated) { 228 | var ts = Math.floor(Date.now() / 1000) 229 | var shadow = JSON.parse(this.shadow) 230 | if (shadow.version == shadowUpdated.version) { 231 | if (shadowUpdated.state.desired == null) { 232 | shadow.state.desired = shadow.state.desired || {} 233 | shadow.state.reported = shadow.state.reported || {} 234 | shadow.metadata.reported = shadow.metadata.reported || {} 235 | for (var key in shadow.state.desired) { 236 | if (shadow.state.desired[key] != null) { 237 | shadow.state.reported[key] = shadow.state.desired[key] 238 | shadow.metadata.reported[key] = {timestamp: ts} 239 | } else { 240 | delete(shadow.state.reported[key]) 241 | delete(shadow.metadata.reported[key]) 242 | } 243 | } 244 | shadow.timestamp = ts 245 | shadow.version = shadow.version + 1 246 | delete(shadow.state.desired) 247 | delete(shadow.metadata.desired) 248 | this.shadow = JSON.stringify(shadow) 249 | this.save() 250 | this.sendCommand({ 251 | commandName: "$shadow_reply", 252 | data: JSON.stringify({status: "success", timestamp: ts, version: shadow.version}), 253 | qos: 0 254 | }) 255 | } 256 | } else { 257 | this.sendUpdateShadow() 258 | } 259 | } 260 | 261 | deviceSchema.methods.reportShadow = function (shadowReported) { 262 | var ts = Math.floor(Date.now() / 1000) 263 | var shadow = JSON.parse(this.shadow) 264 | if (shadow.version == shadowReported.version) { 265 | shadow.state.reported = shadow.state.reported || {} 266 | shadow.metadata.reported = shadow.metadata.reported || {} 267 | for (var key in shadowReported.state.reported) { 268 | if (shadowReported.state.reported[key] != null) { 269 | shadow.state.reported[key] = shadowReported.state.reported[key] 270 | shadow.metadata.reported[key] = {timestamp: ts} 271 | } else { 272 | delete(shadow.state.reported[key]) 273 | delete(shadow.metadata.reported[key]) 274 | } 275 | } 276 | shadow.timestamp = ts 277 | shadow.version = shadow.version + 1 278 | this.shadow = JSON.stringify(shadow) 279 | this.save() 280 | this.sendCommand({ 281 | commandName: "$shadow_reply", 282 | data: JSON.stringify({status: "success", timestamp: ts, version: shadow.version}), 283 | qos: 0 284 | }) 285 | } else { 286 | this.sendUpdateShadow() 287 | } 288 | } 289 | 290 | const Device = mongoose.model("Device", deviceSchema); 291 | 292 | module.exports = Device; -------------------------------------------------------------------------------- /models/device_acl.js: -------------------------------------------------------------------------------- 1 | var mongoose = require('mongoose'); 2 | var Schema = mongoose.Schema; 3 | 4 | const deviceACLSchema = new Schema({ 5 | broker_username: String, 6 | publish: Array, 7 | subscribe: Array, 8 | pubsub: Array, 9 | }, { collection: 'device_acl' }) 10 | 11 | const DeviceACL = mongoose.model("DeviceACL", deviceACLSchema); 12 | 13 | module.exports = DeviceACL 14 | -------------------------------------------------------------------------------- /models/message.js: -------------------------------------------------------------------------------- 1 | var mongoose = require('mongoose'); 2 | var Schema = mongoose.Schema; 3 | 4 | const messageSchema = new Schema({ 5 | message_id: String, 6 | product_name: String, 7 | device_name: String, 8 | data_type: String, 9 | payload: Buffer, 10 | sent_at: Number 11 | }) 12 | 13 | messageSchema.methods.toJSONObject = function () { 14 | return { 15 | product_name: this.product_name, 16 | device_name: this.device_name, 17 | send_at: this.send_at, 18 | data_type: this.data_type, 19 | message_id: this.message_id, 20 | payload: this.payload.toString("base64") 21 | } 22 | } 23 | 24 | const Message = mongoose.model("Message", messageSchema); 25 | 26 | module.exports = Message 27 | -------------------------------------------------------------------------------- /models/redis.js: -------------------------------------------------------------------------------- 1 | const redis = require('redis'); 2 | const client = redis.createClient(process.env.REDIS_URL, {'return_buffers': true}); 3 | client.on("error", function (err) { 4 | console.log("Error " + err); 5 | }); 6 | module.exports = client; 7 | -------------------------------------------------------------------------------- /monitor.js: -------------------------------------------------------------------------------- 1 | var mqtt = require('mqtt') 2 | var InfluxDbService = require("./services/influxdb_service") 3 | const pathToRegexp = require('path-to-regexp') 4 | var client = mqtt.connect('mqtt://127.0.0.1:11883') 5 | client.on('connect', function () { 6 | client.subscribe("$queue/$SYS/brokers/+/stats/connections/count") 7 | }) 8 | 9 | client.on('message', function (topic, message) { 10 | var result 11 | var countRule = pathToRegexp("$SYS/brokers/:nodeName/stats/connections/count") 12 | if((result = countRule.exec(topic)) != null){ 13 | InfluxDbService.writeConnectionCount(result[1], parseInt(message.toString())) 14 | } 15 | }) -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "iothub-server", 3 | "version": "0.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "accepts": { 8 | "version": "1.3.7", 9 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 10 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 11 | "requires": { 12 | "mime-types": "~2.1.24", 13 | "negotiator": "0.6.2" 14 | } 15 | }, 16 | "acorn": { 17 | "version": "2.7.0", 18 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-2.7.0.tgz", 19 | "integrity": "sha1-q259nYhqrKiwhbwzEreaGYQz8Oc=" 20 | }, 21 | "acorn-globals": { 22 | "version": "1.0.9", 23 | "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-1.0.9.tgz", 24 | "integrity": "sha1-VbtemGkVB7dFedBRNBMhfDgMVM8=", 25 | "requires": { 26 | "acorn": "^2.1.0" 27 | } 28 | }, 29 | "ajv": { 30 | "version": "6.10.0", 31 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.0.tgz", 32 | "integrity": "sha512-nffhOpkymDECQyR0mnsUtoCE8RlX38G0rYP+wgLWFyZuUyuuojSSvi/+euOiQBIn63whYwYVIIH1TvE3tu4OEg==", 33 | "requires": { 34 | "fast-deep-equal": "^2.0.1", 35 | "fast-json-stable-stringify": "^2.0.0", 36 | "json-schema-traverse": "^0.4.1", 37 | "uri-js": "^4.2.2" 38 | } 39 | }, 40 | "align-text": { 41 | "version": "0.1.4", 42 | "resolved": "https://registry.npmjs.org/align-text/-/align-text-0.1.4.tgz", 43 | "integrity": "sha1-DNkKVhCT810KmSVsIrcGlDP60Rc=", 44 | "requires": { 45 | "kind-of": "^3.0.2", 46 | "longest": "^1.0.1", 47 | "repeat-string": "^1.5.2" 48 | } 49 | }, 50 | "amdefine": { 51 | "version": "1.0.1", 52 | "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", 53 | "integrity": "sha1-SlKCrBZHKek2Gbz9OtFR+BfOkfU=" 54 | }, 55 | "amqplib": { 56 | "version": "0.5.3", 57 | "resolved": "https://registry.npmjs.org/amqplib/-/amqplib-0.5.3.tgz", 58 | "integrity": "sha512-ZOdUhMxcF+u62rPI+hMtU1NBXSDFQ3eCJJrenamtdQ7YYwh7RZJHOIM1gonVbZ5PyVdYH4xqBPje9OYqk7fnqw==", 59 | "requires": { 60 | "bitsyntax": "~0.1.0", 61 | "bluebird": "^3.5.2", 62 | "buffer-more-ints": "~1.0.0", 63 | "readable-stream": "1.x >=1.1.9", 64 | "safe-buffer": "~5.1.2", 65 | "url-parse": "~1.4.3" 66 | }, 67 | "dependencies": { 68 | "bluebird": { 69 | "version": "3.5.5", 70 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.5.tgz", 71 | "integrity": "sha512-5am6HnnfN+urzt4yfg7IgTbotDjIT/u8AJpEt0sIU9FtXfVeezXAPKswrG+xKUCOYAINpSdgZVDU6QFh+cuH3w==" 72 | } 73 | } 74 | }, 75 | "array-flatten": { 76 | "version": "1.1.1", 77 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 78 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 79 | }, 80 | "asap": { 81 | "version": "1.0.0", 82 | "resolved": "https://registry.npmjs.org/asap/-/asap-1.0.0.tgz", 83 | "integrity": "sha1-sqRdpf36ILBJb8N2jMJ8EvqRan0=" 84 | }, 85 | "asn1": { 86 | "version": "0.2.4", 87 | "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", 88 | "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", 89 | "requires": { 90 | "safer-buffer": "~2.1.0" 91 | } 92 | }, 93 | "assert-plus": { 94 | "version": "1.0.0", 95 | "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", 96 | "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" 97 | }, 98 | "async": { 99 | "version": "2.6.2", 100 | "resolved": "https://registry.npmjs.org/async/-/async-2.6.2.tgz", 101 | "integrity": "sha512-H1qVYh1MYhEEFLsP97cVKqCGo7KfCyTt6uEWqsTBr9SO84oK9Uwbyd/yCW+6rKJLHksBNUVWZDAjfS+Ccx0Bbg==", 102 | "requires": { 103 | "lodash": "^4.17.11" 104 | } 105 | }, 106 | "async-limiter": { 107 | "version": "1.0.0", 108 | "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz", 109 | "integrity": "sha512-jp/uFnooOiO+L211eZOoSyzpOITMXx1rBITauYykG3BRYPu8h0UcxsPNB04RR5vo4Tyz3+ay17tR6JVf9qzYWg==" 110 | }, 111 | "asynckit": { 112 | "version": "0.4.0", 113 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 114 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 115 | }, 116 | "aws-sign2": { 117 | "version": "0.7.0", 118 | "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", 119 | "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" 120 | }, 121 | "aws4": { 122 | "version": "1.8.0", 123 | "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.8.0.tgz", 124 | "integrity": "sha512-ReZxvNHIOv88FlT7rxcXIIC0fPt4KZqZbOlivyWtXLt8ESx84zd3kMC6iK5jVeS2qt+g7ftS7ye4fi06X5rtRQ==" 125 | }, 126 | "balanced-match": { 127 | "version": "1.0.0", 128 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 129 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" 130 | }, 131 | "base64-js": { 132 | "version": "1.3.0", 133 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.0.tgz", 134 | "integrity": "sha512-ccav/yGvoa80BQDljCxsmmQ3Xvx60/UpBIij5QN21W3wBi/hhIC9OoO+KLpu9IJTS9j4DRVJ3aDDF9cMSoa2lw==" 135 | }, 136 | "basic-auth": { 137 | "version": "2.0.1", 138 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 139 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 140 | "requires": { 141 | "safe-buffer": "5.1.2" 142 | } 143 | }, 144 | "bcrypt-pbkdf": { 145 | "version": "1.0.2", 146 | "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", 147 | "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", 148 | "requires": { 149 | "tweetnacl": "^0.14.3" 150 | } 151 | }, 152 | "bitsyntax": { 153 | "version": "0.1.0", 154 | "resolved": "https://registry.npmjs.org/bitsyntax/-/bitsyntax-0.1.0.tgz", 155 | "integrity": "sha512-ikAdCnrloKmFOugAfxWws89/fPc+nw0OOG1IzIE72uSOg/A3cYptKCjSUhDTuj7fhsJtzkzlv7l3b8PzRHLN0Q==", 156 | "requires": { 157 | "buffer-more-ints": "~1.0.0", 158 | "debug": "~2.6.9", 159 | "safe-buffer": "~5.1.2" 160 | } 161 | }, 162 | "bl": { 163 | "version": "1.2.2", 164 | "resolved": "https://registry.npmjs.org/bl/-/bl-1.2.2.tgz", 165 | "integrity": "sha512-e8tQYnZodmebYDWGH7KMRvtzKXaJHx3BbilrgZCfvyLUYdKpK1t5PSPmpkny/SgiTSCnjfLW7v5rlONXVFkQEA==", 166 | "requires": { 167 | "readable-stream": "^2.3.5", 168 | "safe-buffer": "^5.1.1" 169 | }, 170 | "dependencies": { 171 | "isarray": { 172 | "version": "1.0.0", 173 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 174 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 175 | }, 176 | "readable-stream": { 177 | "version": "2.3.6", 178 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 179 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 180 | "requires": { 181 | "core-util-is": "~1.0.0", 182 | "inherits": "~2.0.3", 183 | "isarray": "~1.0.0", 184 | "process-nextick-args": "~2.0.0", 185 | "safe-buffer": "~5.1.1", 186 | "string_decoder": "~1.1.1", 187 | "util-deprecate": "~1.0.1" 188 | } 189 | }, 190 | "string_decoder": { 191 | "version": "1.1.1", 192 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 193 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 194 | "requires": { 195 | "safe-buffer": "~5.1.0" 196 | } 197 | } 198 | } 199 | }, 200 | "bluebird": { 201 | "version": "3.5.1", 202 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz", 203 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==" 204 | }, 205 | "body-parser": { 206 | "version": "1.18.3", 207 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.3.tgz", 208 | "integrity": "sha1-WykhmP/dVTs6DyDe0FkrlWlVyLQ=", 209 | "requires": { 210 | "bytes": "3.0.0", 211 | "content-type": "~1.0.4", 212 | "debug": "2.6.9", 213 | "depd": "~1.1.2", 214 | "http-errors": "~1.6.3", 215 | "iconv-lite": "0.4.23", 216 | "on-finished": "~2.3.0", 217 | "qs": "6.5.2", 218 | "raw-body": "2.3.3", 219 | "type-is": "~1.6.16" 220 | } 221 | }, 222 | "brace-expansion": { 223 | "version": "1.1.11", 224 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 225 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 226 | "requires": { 227 | "balanced-match": "^1.0.0", 228 | "concat-map": "0.0.1" 229 | } 230 | }, 231 | "bson": { 232 | "version": "4.0.2", 233 | "resolved": "https://registry.npmjs.org/bson/-/bson-4.0.2.tgz", 234 | "integrity": "sha512-rBdCxMBCg2aR420e1oKUejjcuPZLTibA7zEhWAlliFWEwzuBCC9Dkp5r7VFFIQB2t1WVsvTbohry575mc7Xw5A==", 235 | "requires": { 236 | "buffer": "^5.1.0", 237 | "long": "^4.0.0" 238 | } 239 | }, 240 | "buffer": { 241 | "version": "5.2.1", 242 | "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.2.1.tgz", 243 | "integrity": "sha512-c+Ko0loDaFfuPWiL02ls9Xd3GO3cPVmUobQ6t3rXNUk304u6hGq+8N/kFi+QEIKhzK3uwolVhLzszmfLmMLnqg==", 244 | "requires": { 245 | "base64-js": "^1.0.2", 246 | "ieee754": "^1.1.4" 247 | } 248 | }, 249 | "buffer-equal-constant-time": { 250 | "version": "1.0.1", 251 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 252 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 253 | }, 254 | "buffer-from": { 255 | "version": "1.1.1", 256 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 257 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" 258 | }, 259 | "buffer-more-ints": { 260 | "version": "1.0.0", 261 | "resolved": "https://registry.npmjs.org/buffer-more-ints/-/buffer-more-ints-1.0.0.tgz", 262 | "integrity": "sha512-EMetuGFz5SLsT0QTnXzINh4Ksr+oo4i+UGTXEshiGCQWnsgSs7ZhJ8fzlwQ+OzEMs0MpDAMr1hxnblp5a4vcHg==" 263 | }, 264 | "bytes": { 265 | "version": "3.0.0", 266 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", 267 | "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" 268 | }, 269 | "callback-stream": { 270 | "version": "1.1.0", 271 | "resolved": "https://registry.npmjs.org/callback-stream/-/callback-stream-1.1.0.tgz", 272 | "integrity": "sha1-RwGlEmbwbgbqpx/BcjOCLYdfSQg=", 273 | "requires": { 274 | "inherits": "^2.0.1", 275 | "readable-stream": "> 1.0.0 < 3.0.0" 276 | } 277 | }, 278 | "camelcase": { 279 | "version": "1.2.1", 280 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-1.2.1.tgz", 281 | "integrity": "sha1-m7UwTS4LVmmLLHWLCKPqqdqlijk=" 282 | }, 283 | "caseless": { 284 | "version": "0.12.0", 285 | "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", 286 | "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" 287 | }, 288 | "center-align": { 289 | "version": "0.1.3", 290 | "resolved": "https://registry.npmjs.org/center-align/-/center-align-0.1.3.tgz", 291 | "integrity": "sha1-qg0yYptu6XIgBBHL1EYckHvCt60=", 292 | "requires": { 293 | "align-text": "^0.1.3", 294 | "lazy-cache": "^1.0.3" 295 | } 296 | }, 297 | "character-parser": { 298 | "version": "1.2.1", 299 | "resolved": "https://registry.npmjs.org/character-parser/-/character-parser-1.2.1.tgz", 300 | "integrity": "sha1-wN3kqxgnE7kZuXCVmhI+zBow/NY=" 301 | }, 302 | "clean-css": { 303 | "version": "3.4.28", 304 | "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-3.4.28.tgz", 305 | "integrity": "sha1-vxlF6C/ICPVWlebd6uwBQA79A/8=", 306 | "requires": { 307 | "commander": "2.8.x", 308 | "source-map": "0.4.x" 309 | }, 310 | "dependencies": { 311 | "commander": { 312 | "version": "2.8.1", 313 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.8.1.tgz", 314 | "integrity": "sha1-Br42f+v9oMMwqh4qBy09yXYkJdQ=", 315 | "requires": { 316 | "graceful-readlink": ">= 1.0.0" 317 | } 318 | } 319 | } 320 | }, 321 | "cliui": { 322 | "version": "2.1.0", 323 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-2.1.0.tgz", 324 | "integrity": "sha1-S0dXYP+AJkx2LDoXGQMukcf+oNE=", 325 | "requires": { 326 | "center-align": "^0.1.1", 327 | "right-align": "^0.1.1", 328 | "wordwrap": "0.0.2" 329 | }, 330 | "dependencies": { 331 | "wordwrap": { 332 | "version": "0.0.2", 333 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.2.tgz", 334 | "integrity": "sha1-t5Zpu0LstAn4PVg8rVLKF+qhZD8=" 335 | } 336 | } 337 | }, 338 | "combined-stream": { 339 | "version": "1.0.8", 340 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 341 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 342 | "requires": { 343 | "delayed-stream": "~1.0.0" 344 | } 345 | }, 346 | "commander": { 347 | "version": "2.6.0", 348 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.6.0.tgz", 349 | "integrity": "sha1-nfflL7Kgyw+4kFjugMMQQiXzfh0=" 350 | }, 351 | "commist": { 352 | "version": "1.1.0", 353 | "resolved": "https://registry.npmjs.org/commist/-/commist-1.1.0.tgz", 354 | "integrity": "sha512-rraC8NXWOEjhADbZe9QBNzLAN5Q3fsTPQtBV+fEVj6xKIgDgNiEVE6ZNfHpZOqfQ21YUzfVNUXLOEZquYvQPPg==", 355 | "requires": { 356 | "leven": "^2.1.0", 357 | "minimist": "^1.1.0" 358 | }, 359 | "dependencies": { 360 | "minimist": { 361 | "version": "1.2.0", 362 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", 363 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=" 364 | } 365 | } 366 | }, 367 | "concat-map": { 368 | "version": "0.0.1", 369 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 370 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" 371 | }, 372 | "concat-stream": { 373 | "version": "1.6.2", 374 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", 375 | "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", 376 | "requires": { 377 | "buffer-from": "^1.0.0", 378 | "inherits": "^2.0.3", 379 | "readable-stream": "^2.2.2", 380 | "typedarray": "^0.0.6" 381 | }, 382 | "dependencies": { 383 | "isarray": { 384 | "version": "1.0.0", 385 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 386 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 387 | }, 388 | "readable-stream": { 389 | "version": "2.3.6", 390 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 391 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 392 | "requires": { 393 | "core-util-is": "~1.0.0", 394 | "inherits": "~2.0.3", 395 | "isarray": "~1.0.0", 396 | "process-nextick-args": "~2.0.0", 397 | "safe-buffer": "~5.1.1", 398 | "string_decoder": "~1.1.1", 399 | "util-deprecate": "~1.0.1" 400 | } 401 | }, 402 | "string_decoder": { 403 | "version": "1.1.1", 404 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 405 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 406 | "requires": { 407 | "safe-buffer": "~5.1.0" 408 | } 409 | } 410 | } 411 | }, 412 | "constantinople": { 413 | "version": "3.0.2", 414 | "resolved": "https://registry.npmjs.org/constantinople/-/constantinople-3.0.2.tgz", 415 | "integrity": "sha1-S5RdmTeQe82Y7ldRIsOBdRZUQUE=", 416 | "requires": { 417 | "acorn": "^2.1.0" 418 | } 419 | }, 420 | "content-disposition": { 421 | "version": "0.5.2", 422 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", 423 | "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" 424 | }, 425 | "content-type": { 426 | "version": "1.0.4", 427 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 428 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 429 | }, 430 | "cookie": { 431 | "version": "0.3.1", 432 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 433 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 434 | }, 435 | "cookie-parser": { 436 | "version": "1.4.4", 437 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.4.tgz", 438 | "integrity": "sha512-lo13tqF3JEtFO7FyA49CqbhaFkskRJ0u/UAiINgrIXeRCY41c88/zxtrECl8AKH3B0hj9q10+h3Kt8I7KlW4tw==", 439 | "requires": { 440 | "cookie": "0.3.1", 441 | "cookie-signature": "1.0.6" 442 | } 443 | }, 444 | "cookie-signature": { 445 | "version": "1.0.6", 446 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 447 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 448 | }, 449 | "core-util-is": { 450 | "version": "1.0.2", 451 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 452 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 453 | }, 454 | "css": { 455 | "version": "1.0.8", 456 | "resolved": "https://registry.npmjs.org/css/-/css-1.0.8.tgz", 457 | "integrity": "sha1-k4aBHKgrzMnuf7WnMrHioxfIo+c=", 458 | "requires": { 459 | "css-parse": "1.0.4", 460 | "css-stringify": "1.0.5" 461 | } 462 | }, 463 | "css-parse": { 464 | "version": "1.0.4", 465 | "resolved": "https://registry.npmjs.org/css-parse/-/css-parse-1.0.4.tgz", 466 | "integrity": "sha1-OLBQP7+dqfVOnB29pg4UXHcRe90=" 467 | }, 468 | "css-stringify": { 469 | "version": "1.0.5", 470 | "resolved": "https://registry.npmjs.org/css-stringify/-/css-stringify-1.0.5.tgz", 471 | "integrity": "sha1-sNBClG2ylTu50pKQCmy19tASIDE=" 472 | }, 473 | "d": { 474 | "version": "1.0.0", 475 | "resolved": "https://registry.npmjs.org/d/-/d-1.0.0.tgz", 476 | "integrity": "sha1-dUu1v+VUUdpppYuU1F9MWwRi1Y8=", 477 | "requires": { 478 | "es5-ext": "^0.10.9" 479 | } 480 | }, 481 | "dashdash": { 482 | "version": "1.14.1", 483 | "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", 484 | "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", 485 | "requires": { 486 | "assert-plus": "^1.0.0" 487 | } 488 | }, 489 | "debug": { 490 | "version": "2.6.9", 491 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 492 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 493 | "requires": { 494 | "ms": "2.0.0" 495 | } 496 | }, 497 | "decamelize": { 498 | "version": "1.2.0", 499 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 500 | "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" 501 | }, 502 | "delayed-stream": { 503 | "version": "1.0.0", 504 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 505 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 506 | }, 507 | "depd": { 508 | "version": "1.1.2", 509 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 510 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 511 | }, 512 | "destroy": { 513 | "version": "1.0.4", 514 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 515 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 516 | }, 517 | "dotenv": { 518 | "version": "8.0.0", 519 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.0.0.tgz", 520 | "integrity": "sha512-30xVGqjLjiUOArT4+M5q9sYdvuR4riM6yK9wMcas9Vbp6zZa+ocC9dp6QoftuhTPhFAiLK/0C5Ni2nou/Bk8lg==" 521 | }, 522 | "double-ended-queue": { 523 | "version": "2.1.0-0", 524 | "resolved": "https://registry.npmjs.org/double-ended-queue/-/double-ended-queue-2.1.0-0.tgz", 525 | "integrity": "sha1-ED01J/0xUo9AGIEwyEHv3XgmTlw=" 526 | }, 527 | "duplexify": { 528 | "version": "3.7.1", 529 | "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-3.7.1.tgz", 530 | "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", 531 | "requires": { 532 | "end-of-stream": "^1.0.0", 533 | "inherits": "^2.0.1", 534 | "readable-stream": "^2.0.0", 535 | "stream-shift": "^1.0.0" 536 | }, 537 | "dependencies": { 538 | "isarray": { 539 | "version": "1.0.0", 540 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 541 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 542 | }, 543 | "readable-stream": { 544 | "version": "2.3.6", 545 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 546 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 547 | "requires": { 548 | "core-util-is": "~1.0.0", 549 | "inherits": "~2.0.3", 550 | "isarray": "~1.0.0", 551 | "process-nextick-args": "~2.0.0", 552 | "safe-buffer": "~5.1.1", 553 | "string_decoder": "~1.1.1", 554 | "util-deprecate": "~1.0.1" 555 | } 556 | }, 557 | "string_decoder": { 558 | "version": "1.1.1", 559 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 560 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 561 | "requires": { 562 | "safe-buffer": "~5.1.0" 563 | } 564 | } 565 | } 566 | }, 567 | "ecc-jsbn": { 568 | "version": "0.1.2", 569 | "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", 570 | "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", 571 | "requires": { 572 | "jsbn": "~0.1.0", 573 | "safer-buffer": "^2.1.0" 574 | } 575 | }, 576 | "ecdsa-sig-formatter": { 577 | "version": "1.0.11", 578 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 579 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 580 | "requires": { 581 | "safe-buffer": "^5.0.1" 582 | } 583 | }, 584 | "ee-first": { 585 | "version": "1.1.1", 586 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 587 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 588 | }, 589 | "encodeurl": { 590 | "version": "1.0.2", 591 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 592 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 593 | }, 594 | "end-of-stream": { 595 | "version": "1.4.1", 596 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.1.tgz", 597 | "integrity": "sha512-1MkrZNvWTKCaigbn+W15elq2BB/L22nqrSY5DKlo3X6+vclJm8Bb5djXJBmEX6fS3+zCh/F4VBK5Z2KxJt4s2Q==", 598 | "requires": { 599 | "once": "^1.4.0" 600 | } 601 | }, 602 | "es5-ext": { 603 | "version": "0.10.50", 604 | "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.50.tgz", 605 | "integrity": "sha512-KMzZTPBkeQV/JcSQhI5/z6d9VWJ3EnQ194USTUwIYZ2ZbpN8+SGXQKt1h68EX44+qt+Fzr8DO17vnxrw7c3agw==", 606 | "requires": { 607 | "es6-iterator": "~2.0.3", 608 | "es6-symbol": "~3.1.1", 609 | "next-tick": "^1.0.0" 610 | } 611 | }, 612 | "es6-iterator": { 613 | "version": "2.0.3", 614 | "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz", 615 | "integrity": "sha1-p96IkUGgWpSwhUQDstCg+/qY87c=", 616 | "requires": { 617 | "d": "1", 618 | "es5-ext": "^0.10.35", 619 | "es6-symbol": "^3.1.1" 620 | } 621 | }, 622 | "es6-map": { 623 | "version": "0.1.5", 624 | "resolved": "https://registry.npmjs.org/es6-map/-/es6-map-0.1.5.tgz", 625 | "integrity": "sha1-kTbgUD3MBqMBaQ8LsU/042TpSfA=", 626 | "requires": { 627 | "d": "1", 628 | "es5-ext": "~0.10.14", 629 | "es6-iterator": "~2.0.1", 630 | "es6-set": "~0.1.5", 631 | "es6-symbol": "~3.1.1", 632 | "event-emitter": "~0.3.5" 633 | } 634 | }, 635 | "es6-set": { 636 | "version": "0.1.5", 637 | "resolved": "https://registry.npmjs.org/es6-set/-/es6-set-0.1.5.tgz", 638 | "integrity": "sha1-0rPsXU2ADO2BjbU40ol02wpzzLE=", 639 | "requires": { 640 | "d": "1", 641 | "es5-ext": "~0.10.14", 642 | "es6-iterator": "~2.0.1", 643 | "es6-symbol": "3.1.1", 644 | "event-emitter": "~0.3.5" 645 | } 646 | }, 647 | "es6-symbol": { 648 | "version": "3.1.1", 649 | "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.1.tgz", 650 | "integrity": "sha1-vwDvT9q2uhtG7Le2KbTH7VcVzHc=", 651 | "requires": { 652 | "d": "1", 653 | "es5-ext": "~0.10.14" 654 | } 655 | }, 656 | "escape-html": { 657 | "version": "1.0.3", 658 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 659 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 660 | }, 661 | "etag": { 662 | "version": "1.8.1", 663 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 664 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 665 | }, 666 | "event-emitter": { 667 | "version": "0.3.5", 668 | "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz", 669 | "integrity": "sha1-34xp7vFkeSPHFXuc6DhAYQsCzDk=", 670 | "requires": { 671 | "d": "1", 672 | "es5-ext": "~0.10.14" 673 | } 674 | }, 675 | "express": { 676 | "version": "4.16.4", 677 | "resolved": "https://registry.npmjs.org/express/-/express-4.16.4.tgz", 678 | "integrity": "sha512-j12Uuyb4FMrd/qQAm6uCHAkPtO8FDTRJZBDd5D2KOL2eLaz1yUNdUB/NOIyq0iU4q4cFarsUCrnFDPBcnksuOg==", 679 | "requires": { 680 | "accepts": "~1.3.5", 681 | "array-flatten": "1.1.1", 682 | "body-parser": "1.18.3", 683 | "content-disposition": "0.5.2", 684 | "content-type": "~1.0.4", 685 | "cookie": "0.3.1", 686 | "cookie-signature": "1.0.6", 687 | "debug": "2.6.9", 688 | "depd": "~1.1.2", 689 | "encodeurl": "~1.0.2", 690 | "escape-html": "~1.0.3", 691 | "etag": "~1.8.1", 692 | "finalhandler": "1.1.1", 693 | "fresh": "0.5.2", 694 | "merge-descriptors": "1.0.1", 695 | "methods": "~1.1.2", 696 | "on-finished": "~2.3.0", 697 | "parseurl": "~1.3.2", 698 | "path-to-regexp": "0.1.7", 699 | "proxy-addr": "~2.0.4", 700 | "qs": "6.5.2", 701 | "range-parser": "~1.2.0", 702 | "safe-buffer": "5.1.2", 703 | "send": "0.16.2", 704 | "serve-static": "1.13.2", 705 | "setprototypeof": "1.1.0", 706 | "statuses": "~1.4.0", 707 | "type-is": "~1.6.16", 708 | "utils-merge": "1.0.1", 709 | "vary": "~1.1.2" 710 | }, 711 | "dependencies": { 712 | "path-to-regexp": { 713 | "version": "0.1.7", 714 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 715 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 716 | } 717 | } 718 | }, 719 | "extend": { 720 | "version": "3.0.2", 721 | "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", 722 | "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" 723 | }, 724 | "extsprintf": { 725 | "version": "1.3.0", 726 | "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", 727 | "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" 728 | }, 729 | "fast-deep-equal": { 730 | "version": "2.0.1", 731 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", 732 | "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=" 733 | }, 734 | "fast-json-stable-stringify": { 735 | "version": "2.0.0", 736 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", 737 | "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=" 738 | }, 739 | "finalhandler": { 740 | "version": "1.1.1", 741 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", 742 | "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", 743 | "requires": { 744 | "debug": "2.6.9", 745 | "encodeurl": "~1.0.2", 746 | "escape-html": "~1.0.3", 747 | "on-finished": "~2.3.0", 748 | "parseurl": "~1.3.2", 749 | "statuses": "~1.4.0", 750 | "unpipe": "~1.0.0" 751 | } 752 | }, 753 | "forever-agent": { 754 | "version": "0.6.1", 755 | "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", 756 | "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" 757 | }, 758 | "form-data": { 759 | "version": "2.3.3", 760 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", 761 | "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", 762 | "requires": { 763 | "asynckit": "^0.4.0", 764 | "combined-stream": "^1.0.6", 765 | "mime-types": "^2.1.12" 766 | } 767 | }, 768 | "forwarded": { 769 | "version": "0.1.2", 770 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 771 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 772 | }, 773 | "fresh": { 774 | "version": "0.5.2", 775 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 776 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 777 | }, 778 | "fs.realpath": { 779 | "version": "1.0.0", 780 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 781 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" 782 | }, 783 | "getpass": { 784 | "version": "0.1.7", 785 | "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", 786 | "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", 787 | "requires": { 788 | "assert-plus": "^1.0.0" 789 | } 790 | }, 791 | "glob": { 792 | "version": "7.1.4", 793 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.4.tgz", 794 | "integrity": "sha512-hkLPepehmnKk41pUGm3sYxoFs/umurYfYJCerbXEyFIWcAzvpipAgVkBqqT9RBKMGjnq6kMuyYwha6csxbiM1A==", 795 | "requires": { 796 | "fs.realpath": "^1.0.0", 797 | "inflight": "^1.0.4", 798 | "inherits": "2", 799 | "minimatch": "^3.0.4", 800 | "once": "^1.3.0", 801 | "path-is-absolute": "^1.0.0" 802 | } 803 | }, 804 | "glob-parent": { 805 | "version": "3.1.0", 806 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", 807 | "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", 808 | "requires": { 809 | "is-glob": "^3.1.0", 810 | "path-dirname": "^1.0.0" 811 | } 812 | }, 813 | "glob-stream": { 814 | "version": "6.1.0", 815 | "resolved": "https://registry.npmjs.org/glob-stream/-/glob-stream-6.1.0.tgz", 816 | "integrity": "sha1-cEXJlBOz65SIjYOrRtC0BMx73eQ=", 817 | "requires": { 818 | "extend": "^3.0.0", 819 | "glob": "^7.1.1", 820 | "glob-parent": "^3.1.0", 821 | "is-negated-glob": "^1.0.0", 822 | "ordered-read-streams": "^1.0.0", 823 | "pumpify": "^1.3.5", 824 | "readable-stream": "^2.1.5", 825 | "remove-trailing-separator": "^1.0.1", 826 | "to-absolute-glob": "^2.0.0", 827 | "unique-stream": "^2.0.2" 828 | }, 829 | "dependencies": { 830 | "isarray": { 831 | "version": "1.0.0", 832 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 833 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 834 | }, 835 | "readable-stream": { 836 | "version": "2.3.6", 837 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 838 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 839 | "requires": { 840 | "core-util-is": "~1.0.0", 841 | "inherits": "~2.0.3", 842 | "isarray": "~1.0.0", 843 | "process-nextick-args": "~2.0.0", 844 | "safe-buffer": "~5.1.1", 845 | "string_decoder": "~1.1.1", 846 | "util-deprecate": "~1.0.1" 847 | } 848 | }, 849 | "string_decoder": { 850 | "version": "1.1.1", 851 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 852 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 853 | "requires": { 854 | "safe-buffer": "~5.1.0" 855 | } 856 | } 857 | } 858 | }, 859 | "graceful-readlink": { 860 | "version": "1.0.1", 861 | "resolved": "https://registry.npmjs.org/graceful-readlink/-/graceful-readlink-1.0.1.tgz", 862 | "integrity": "sha1-TK+tdrxi8C+gObL5Tpo906ORpyU=" 863 | }, 864 | "har-schema": { 865 | "version": "2.0.0", 866 | "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", 867 | "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=" 868 | }, 869 | "har-validator": { 870 | "version": "5.1.3", 871 | "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.3.tgz", 872 | "integrity": "sha512-sNvOCzEQNr/qrvJgc3UG/kD4QtlHycrzwS+6mfTrrSq97BvaYcPZZI1ZSqGSPR73Cxn4LKTD4PttRwfU7jWq5g==", 873 | "requires": { 874 | "ajv": "^6.5.5", 875 | "har-schema": "^2.0.0" 876 | } 877 | }, 878 | "help-me": { 879 | "version": "1.1.0", 880 | "resolved": "https://registry.npmjs.org/help-me/-/help-me-1.1.0.tgz", 881 | "integrity": "sha1-jy1QjQYAtKRW2i8IZVbn5cBWo8Y=", 882 | "requires": { 883 | "callback-stream": "^1.0.2", 884 | "glob-stream": "^6.1.0", 885 | "through2": "^2.0.1", 886 | "xtend": "^4.0.0" 887 | } 888 | }, 889 | "http-errors": { 890 | "version": "1.6.3", 891 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", 892 | "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", 893 | "requires": { 894 | "depd": "~1.1.2", 895 | "inherits": "2.0.3", 896 | "setprototypeof": "1.1.0", 897 | "statuses": ">= 1.4.0 < 2" 898 | } 899 | }, 900 | "http-signature": { 901 | "version": "1.2.0", 902 | "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", 903 | "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", 904 | "requires": { 905 | "assert-plus": "^1.0.0", 906 | "jsprim": "^1.2.2", 907 | "sshpk": "^1.7.0" 908 | } 909 | }, 910 | "iconv-lite": { 911 | "version": "0.4.23", 912 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", 913 | "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", 914 | "requires": { 915 | "safer-buffer": ">= 2.1.2 < 3" 916 | } 917 | }, 918 | "ieee754": { 919 | "version": "1.1.13", 920 | "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz", 921 | "integrity": "sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==" 922 | }, 923 | "inflight": { 924 | "version": "1.0.6", 925 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 926 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 927 | "requires": { 928 | "once": "^1.3.0", 929 | "wrappy": "1" 930 | } 931 | }, 932 | "influx": { 933 | "version": "5.0.7", 934 | "resolved": "https://registry.npmjs.org/influx/-/influx-5.0.7.tgz", 935 | "integrity": "sha1-NeZfa/E8uqF2MQi1WWqAanJ6Upo=" 936 | }, 937 | "inherits": { 938 | "version": "2.0.3", 939 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 940 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 941 | }, 942 | "ipaddr.js": { 943 | "version": "1.9.0", 944 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz", 945 | "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA==" 946 | }, 947 | "is-absolute": { 948 | "version": "1.0.0", 949 | "resolved": "https://registry.npmjs.org/is-absolute/-/is-absolute-1.0.0.tgz", 950 | "integrity": "sha512-dOWoqflvcydARa360Gvv18DZ/gRuHKi2NU/wU5X1ZFzdYfH29nkiNZsF3mp4OJ3H4yo9Mx8A/uAGNzpzPN3yBA==", 951 | "requires": { 952 | "is-relative": "^1.0.0", 953 | "is-windows": "^1.0.1" 954 | } 955 | }, 956 | "is-buffer": { 957 | "version": "1.1.6", 958 | "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", 959 | "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" 960 | }, 961 | "is-extglob": { 962 | "version": "2.1.1", 963 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 964 | "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=" 965 | }, 966 | "is-glob": { 967 | "version": "3.1.0", 968 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", 969 | "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", 970 | "requires": { 971 | "is-extglob": "^2.1.0" 972 | } 973 | }, 974 | "is-negated-glob": { 975 | "version": "1.0.0", 976 | "resolved": "https://registry.npmjs.org/is-negated-glob/-/is-negated-glob-1.0.0.tgz", 977 | "integrity": "sha1-aRC8pdqMleeEtXUbl2z1oQ/uNtI=" 978 | }, 979 | "is-promise": { 980 | "version": "2.1.0", 981 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", 982 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=" 983 | }, 984 | "is-relative": { 985 | "version": "1.0.0", 986 | "resolved": "https://registry.npmjs.org/is-relative/-/is-relative-1.0.0.tgz", 987 | "integrity": "sha512-Kw/ReK0iqwKeu0MITLFuj0jbPAmEiOsIwyIXvvbfa6QfmN9pkD1M+8pdk7Rl/dTKbH34/XBFMbgD4iMJhLQbGA==", 988 | "requires": { 989 | "is-unc-path": "^1.0.0" 990 | } 991 | }, 992 | "is-typedarray": { 993 | "version": "1.0.0", 994 | "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", 995 | "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" 996 | }, 997 | "is-unc-path": { 998 | "version": "1.0.0", 999 | "resolved": "https://registry.npmjs.org/is-unc-path/-/is-unc-path-1.0.0.tgz", 1000 | "integrity": "sha512-mrGpVd0fs7WWLfVsStvgF6iEJnbjDFZh9/emhRDcGWTduTfNHd9CHeUwH3gYIjdbwo4On6hunkztwOaAw0yllQ==", 1001 | "requires": { 1002 | "unc-path-regex": "^0.1.2" 1003 | } 1004 | }, 1005 | "is-windows": { 1006 | "version": "1.0.2", 1007 | "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", 1008 | "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==" 1009 | }, 1010 | "isarray": { 1011 | "version": "0.0.1", 1012 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 1013 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 1014 | }, 1015 | "isstream": { 1016 | "version": "0.1.2", 1017 | "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", 1018 | "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" 1019 | }, 1020 | "jade": { 1021 | "version": "1.11.0", 1022 | "resolved": "https://registry.npmjs.org/jade/-/jade-1.11.0.tgz", 1023 | "integrity": "sha1-nIDlOMEtP7lcjZu5VZ+gzAQEBf0=", 1024 | "requires": { 1025 | "character-parser": "1.2.1", 1026 | "clean-css": "^3.1.9", 1027 | "commander": "~2.6.0", 1028 | "constantinople": "~3.0.1", 1029 | "jstransformer": "0.0.2", 1030 | "mkdirp": "~0.5.0", 1031 | "transformers": "2.1.0", 1032 | "uglify-js": "^2.4.19", 1033 | "void-elements": "~2.0.1", 1034 | "with": "~4.0.0" 1035 | } 1036 | }, 1037 | "jsbn": { 1038 | "version": "0.1.1", 1039 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", 1040 | "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=" 1041 | }, 1042 | "json-schema": { 1043 | "version": "0.2.3", 1044 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", 1045 | "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" 1046 | }, 1047 | "json-schema-traverse": { 1048 | "version": "0.4.1", 1049 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 1050 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==" 1051 | }, 1052 | "json-stable-stringify-without-jsonify": { 1053 | "version": "1.0.1", 1054 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 1055 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=" 1056 | }, 1057 | "json-stringify-safe": { 1058 | "version": "5.0.1", 1059 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 1060 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" 1061 | }, 1062 | "jsonwebtoken": { 1063 | "version": "8.5.1", 1064 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", 1065 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", 1066 | "requires": { 1067 | "jws": "^3.2.2", 1068 | "lodash.includes": "^4.3.0", 1069 | "lodash.isboolean": "^3.0.3", 1070 | "lodash.isinteger": "^4.0.4", 1071 | "lodash.isnumber": "^3.0.3", 1072 | "lodash.isplainobject": "^4.0.6", 1073 | "lodash.isstring": "^4.0.1", 1074 | "lodash.once": "^4.0.0", 1075 | "ms": "^2.1.1", 1076 | "semver": "^5.6.0" 1077 | }, 1078 | "dependencies": { 1079 | "ms": { 1080 | "version": "2.1.1", 1081 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1082 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1083 | } 1084 | } 1085 | }, 1086 | "jsprim": { 1087 | "version": "1.4.1", 1088 | "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", 1089 | "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", 1090 | "requires": { 1091 | "assert-plus": "1.0.0", 1092 | "extsprintf": "1.3.0", 1093 | "json-schema": "0.2.3", 1094 | "verror": "1.10.0" 1095 | } 1096 | }, 1097 | "jstransformer": { 1098 | "version": "0.0.2", 1099 | "resolved": "https://registry.npmjs.org/jstransformer/-/jstransformer-0.0.2.tgz", 1100 | "integrity": "sha1-eq4pqQPRls+glz2IXT5HlH7Ndqs=", 1101 | "requires": { 1102 | "is-promise": "^2.0.0", 1103 | "promise": "^6.0.1" 1104 | } 1105 | }, 1106 | "jwa": { 1107 | "version": "1.4.1", 1108 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 1109 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 1110 | "requires": { 1111 | "buffer-equal-constant-time": "1.0.1", 1112 | "ecdsa-sig-formatter": "1.0.11", 1113 | "safe-buffer": "^5.0.1" 1114 | } 1115 | }, 1116 | "jws": { 1117 | "version": "3.2.2", 1118 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 1119 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 1120 | "requires": { 1121 | "jwa": "^1.4.1", 1122 | "safe-buffer": "^5.0.1" 1123 | } 1124 | }, 1125 | "kareem": { 1126 | "version": "2.3.0", 1127 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.0.tgz", 1128 | "integrity": "sha512-6hHxsp9e6zQU8nXsP+02HGWXwTkOEw6IROhF2ZA28cYbUk4eJ6QbtZvdqZOdD9YPKghG3apk5eOCvs+tLl3lRg==" 1129 | }, 1130 | "kind-of": { 1131 | "version": "3.2.2", 1132 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", 1133 | "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", 1134 | "requires": { 1135 | "is-buffer": "^1.1.5" 1136 | } 1137 | }, 1138 | "lazy-cache": { 1139 | "version": "1.0.4", 1140 | "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-1.0.4.tgz", 1141 | "integrity": "sha1-odePw6UEdMuAhF07O24dpJpEbo4=" 1142 | }, 1143 | "leven": { 1144 | "version": "2.1.0", 1145 | "resolved": "https://registry.npmjs.org/leven/-/leven-2.1.0.tgz", 1146 | "integrity": "sha1-wuep93IJTe6dNCAq6KzORoeHVYA=" 1147 | }, 1148 | "lodash": { 1149 | "version": "4.17.11", 1150 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", 1151 | "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==" 1152 | }, 1153 | "lodash.includes": { 1154 | "version": "4.3.0", 1155 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 1156 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" 1157 | }, 1158 | "lodash.isboolean": { 1159 | "version": "3.0.3", 1160 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 1161 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" 1162 | }, 1163 | "lodash.isinteger": { 1164 | "version": "4.0.4", 1165 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 1166 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" 1167 | }, 1168 | "lodash.isnumber": { 1169 | "version": "3.0.3", 1170 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 1171 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" 1172 | }, 1173 | "lodash.isplainobject": { 1174 | "version": "4.0.6", 1175 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 1176 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" 1177 | }, 1178 | "lodash.isstring": { 1179 | "version": "4.0.1", 1180 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 1181 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" 1182 | }, 1183 | "lodash.once": { 1184 | "version": "4.1.1", 1185 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 1186 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" 1187 | }, 1188 | "long": { 1189 | "version": "4.0.0", 1190 | "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz", 1191 | "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA==" 1192 | }, 1193 | "longest": { 1194 | "version": "1.0.1", 1195 | "resolved": "https://registry.npmjs.org/longest/-/longest-1.0.1.tgz", 1196 | "integrity": "sha1-MKCy2jj3N3DoKUoNIuZiXtd9AJc=" 1197 | }, 1198 | "media-typer": { 1199 | "version": "0.3.0", 1200 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1201 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 1202 | }, 1203 | "memory-pager": { 1204 | "version": "1.5.0", 1205 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1206 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 1207 | "optional": true 1208 | }, 1209 | "merge-descriptors": { 1210 | "version": "1.0.1", 1211 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1212 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 1213 | }, 1214 | "methods": { 1215 | "version": "1.1.2", 1216 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1217 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 1218 | }, 1219 | "mime": { 1220 | "version": "1.4.1", 1221 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz", 1222 | "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==" 1223 | }, 1224 | "mime-db": { 1225 | "version": "1.40.0", 1226 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.40.0.tgz", 1227 | "integrity": "sha512-jYdeOMPy9vnxEqFRRo6ZvTZ8d9oPb+k18PKoYNYUe2stVEBPPwsln/qWzdbmaIvnhZ9v2P+CuecK+fpUfsV2mA==" 1228 | }, 1229 | "mime-types": { 1230 | "version": "2.1.24", 1231 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.24.tgz", 1232 | "integrity": "sha512-WaFHS3MCl5fapm3oLxU4eYDw77IQM2ACcxQ9RIxfaC3ooc6PFuBMGZZsYpvoXS5D5QTWPieo1jjLdAm3TBP3cQ==", 1233 | "requires": { 1234 | "mime-db": "1.40.0" 1235 | } 1236 | }, 1237 | "minimatch": { 1238 | "version": "3.0.4", 1239 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 1240 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 1241 | "requires": { 1242 | "brace-expansion": "^1.1.7" 1243 | } 1244 | }, 1245 | "minimist": { 1246 | "version": "0.0.8", 1247 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 1248 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" 1249 | }, 1250 | "mkdirp": { 1251 | "version": "0.5.1", 1252 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 1253 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 1254 | "requires": { 1255 | "minimist": "0.0.8" 1256 | } 1257 | }, 1258 | "mongodb": { 1259 | "version": "3.2.4", 1260 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.2.4.tgz", 1261 | "integrity": "sha512-tjuwdRb89oyCamBG9lWkb7Sf4L/XJ0hsTC22NkbECl3j2X8n+rM+ZlK3dyIPdaJXdc4FOKIRd9dodVJgCtsYMA==", 1262 | "requires": { 1263 | "mongodb-core": "3.2.4", 1264 | "safe-buffer": "^5.1.2" 1265 | } 1266 | }, 1267 | "mongodb-core": { 1268 | "version": "3.2.4", 1269 | "resolved": "https://registry.npmjs.org/mongodb-core/-/mongodb-core-3.2.4.tgz", 1270 | "integrity": "sha512-Ea/fjVntuj0nhLPuDPj54Kce+w4Ee6b1oSM2EvNB4OdwXJ5WIEh79st9/FRZVKwGnc2oB19P1SMSC1noOBXUCQ==", 1271 | "requires": { 1272 | "bson": "^1.1.1", 1273 | "require_optional": "^1.0.1", 1274 | "safe-buffer": "^5.1.2", 1275 | "saslprep": "^1.0.0" 1276 | }, 1277 | "dependencies": { 1278 | "bson": { 1279 | "version": "1.1.1", 1280 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.1.tgz", 1281 | "integrity": "sha512-jCGVYLoYMHDkOsbwJZBCqwMHyH4c+wzgI9hG7Z6SZJRXWr+x58pdIbm2i9a/jFGCkRJqRUr8eoI7lDWa0hTkxg==" 1282 | } 1283 | } 1284 | }, 1285 | "mongoose": { 1286 | "version": "5.5.9", 1287 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.5.9.tgz", 1288 | "integrity": "sha512-0FQ9pJS9m0GpPVv6aUNfyhjW+VMKlX/VgnfMkk+sK50shjtO3uG3xweoZrdXuANBEJ9hGH63mivi4301DItjUg==", 1289 | "requires": { 1290 | "async": "2.6.2", 1291 | "bson": "~1.1.1", 1292 | "kareem": "2.3.0", 1293 | "mongodb": "3.2.4", 1294 | "mongodb-core": "3.2.4", 1295 | "mongoose-legacy-pluralize": "1.0.2", 1296 | "mpath": "0.6.0", 1297 | "mquery": "3.2.0", 1298 | "ms": "2.1.1", 1299 | "regexp-clone": "0.0.1", 1300 | "safe-buffer": "5.1.2", 1301 | "sift": "7.0.1", 1302 | "sliced": "1.0.1" 1303 | }, 1304 | "dependencies": { 1305 | "bson": { 1306 | "version": "1.1.1", 1307 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.1.tgz", 1308 | "integrity": "sha512-jCGVYLoYMHDkOsbwJZBCqwMHyH4c+wzgI9hG7Z6SZJRXWr+x58pdIbm2i9a/jFGCkRJqRUr8eoI7lDWa0hTkxg==" 1309 | }, 1310 | "ms": { 1311 | "version": "2.1.1", 1312 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 1313 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 1314 | } 1315 | } 1316 | }, 1317 | "mongoose-legacy-pluralize": { 1318 | "version": "1.0.2", 1319 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", 1320 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" 1321 | }, 1322 | "morgan": { 1323 | "version": "1.9.1", 1324 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.1.tgz", 1325 | "integrity": "sha512-HQStPIV4y3afTiCYVxirakhlCfGkI161c76kKFca7Fk1JusM//Qeo1ej2XaMniiNeaZklMVrh3vTtIzpzwbpmA==", 1326 | "requires": { 1327 | "basic-auth": "~2.0.0", 1328 | "debug": "2.6.9", 1329 | "depd": "~1.1.2", 1330 | "on-finished": "~2.3.0", 1331 | "on-headers": "~1.0.1" 1332 | } 1333 | }, 1334 | "mpath": { 1335 | "version": "0.6.0", 1336 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.6.0.tgz", 1337 | "integrity": "sha512-i75qh79MJ5Xo/sbhxrDrPSEG0H/mr1kcZXJ8dH6URU5jD/knFxCVqVC/gVSW7GIXL/9hHWlT9haLbCXWOll3qw==" 1338 | }, 1339 | "mqtt": { 1340 | "version": "3.0.0", 1341 | "resolved": "https://registry.npmjs.org/mqtt/-/mqtt-3.0.0.tgz", 1342 | "integrity": "sha512-0nKV6MAc1ibKZwaZQUTb3iIdT4NVpj541BsYrqrGBcQdQ7Jd0MnZD1/6/nj1UFdGTboK9ZEUXvkCu2nPCugHFA==", 1343 | "requires": { 1344 | "base64-js": "^1.3.0", 1345 | "commist": "^1.0.0", 1346 | "concat-stream": "^1.6.2", 1347 | "end-of-stream": "^1.4.1", 1348 | "es6-map": "^0.1.5", 1349 | "help-me": "^1.0.1", 1350 | "inherits": "^2.0.3", 1351 | "minimist": "^1.2.0", 1352 | "mqtt-packet": "^6.0.0", 1353 | "pump": "^3.0.0", 1354 | "readable-stream": "^2.3.6", 1355 | "reinterval": "^1.1.0", 1356 | "split2": "^3.1.0", 1357 | "websocket-stream": "^5.1.2", 1358 | "xtend": "^4.0.1" 1359 | }, 1360 | "dependencies": { 1361 | "isarray": { 1362 | "version": "1.0.0", 1363 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1364 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1365 | }, 1366 | "minimist": { 1367 | "version": "1.2.0", 1368 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", 1369 | "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=" 1370 | }, 1371 | "readable-stream": { 1372 | "version": "2.3.6", 1373 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 1374 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 1375 | "requires": { 1376 | "core-util-is": "~1.0.0", 1377 | "inherits": "~2.0.3", 1378 | "isarray": "~1.0.0", 1379 | "process-nextick-args": "~2.0.0", 1380 | "safe-buffer": "~5.1.1", 1381 | "string_decoder": "~1.1.1", 1382 | "util-deprecate": "~1.0.1" 1383 | } 1384 | }, 1385 | "string_decoder": { 1386 | "version": "1.1.1", 1387 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 1388 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 1389 | "requires": { 1390 | "safe-buffer": "~5.1.0" 1391 | } 1392 | } 1393 | } 1394 | }, 1395 | "mqtt-packet": { 1396 | "version": "6.1.2", 1397 | "resolved": "https://registry.npmjs.org/mqtt-packet/-/mqtt-packet-6.1.2.tgz", 1398 | "integrity": "sha512-yVG5PoS3wJ8TLzfS8pQMsDVLAf/EipnBAG5XQE9X/9L0EMxuduI9J2WnlRvJT497K1CUT4VJWjoP08+CKiKt1Q==", 1399 | "requires": { 1400 | "bl": "^1.2.2", 1401 | "inherits": "^2.0.3", 1402 | "process-nextick-args": "^2.0.0", 1403 | "safe-buffer": "^5.1.2" 1404 | } 1405 | }, 1406 | "mquery": { 1407 | "version": "3.2.0", 1408 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.0.tgz", 1409 | "integrity": "sha512-qPJcdK/yqcbQiKoemAt62Y0BAc0fTEKo1IThodBD+O5meQRJT/2HSe5QpBNwaa4CjskoGrYWsEyjkqgiE0qjhg==", 1410 | "requires": { 1411 | "bluebird": "3.5.1", 1412 | "debug": "3.1.0", 1413 | "regexp-clone": "0.0.1", 1414 | "safe-buffer": "5.1.2", 1415 | "sliced": "1.0.1" 1416 | }, 1417 | "dependencies": { 1418 | "debug": { 1419 | "version": "3.1.0", 1420 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 1421 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 1422 | "requires": { 1423 | "ms": "2.0.0" 1424 | } 1425 | } 1426 | } 1427 | }, 1428 | "ms": { 1429 | "version": "2.0.0", 1430 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1431 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 1432 | }, 1433 | "nanoid": { 1434 | "version": "2.0.2", 1435 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-2.0.2.tgz", 1436 | "integrity": "sha512-X4yQ8VHoFvHcykGunT2Jxrsm1c4vH5UKtau7LLJYXO1istCRE3jD8JxDyGCzN+h7dpWBCvWaSYgloRuphKRqUQ==" 1437 | }, 1438 | "negotiator": { 1439 | "version": "0.6.2", 1440 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 1441 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 1442 | }, 1443 | "next-tick": { 1444 | "version": "1.0.0", 1445 | "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.0.0.tgz", 1446 | "integrity": "sha1-yobR/ogoFpsBICCOPchCS524NCw=" 1447 | }, 1448 | "oauth-sign": { 1449 | "version": "0.9.0", 1450 | "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", 1451 | "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==" 1452 | }, 1453 | "on-finished": { 1454 | "version": "2.3.0", 1455 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1456 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 1457 | "requires": { 1458 | "ee-first": "1.1.1" 1459 | } 1460 | }, 1461 | "on-headers": { 1462 | "version": "1.0.2", 1463 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 1464 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 1465 | }, 1466 | "once": { 1467 | "version": "1.4.0", 1468 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1469 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 1470 | "requires": { 1471 | "wrappy": "1" 1472 | } 1473 | }, 1474 | "optimist": { 1475 | "version": "0.3.7", 1476 | "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.3.7.tgz", 1477 | "integrity": "sha1-yQlBrVnkJzMokjB00s8ufLxuwNk=", 1478 | "requires": { 1479 | "wordwrap": "~0.0.2" 1480 | } 1481 | }, 1482 | "ordered-read-streams": { 1483 | "version": "1.0.1", 1484 | "resolved": "https://registry.npmjs.org/ordered-read-streams/-/ordered-read-streams-1.0.1.tgz", 1485 | "integrity": "sha1-d8DLN8QVJdZBZtmQ/61+xqDhNj4=", 1486 | "requires": { 1487 | "readable-stream": "^2.0.1" 1488 | }, 1489 | "dependencies": { 1490 | "isarray": { 1491 | "version": "1.0.0", 1492 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1493 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1494 | }, 1495 | "readable-stream": { 1496 | "version": "2.3.6", 1497 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 1498 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 1499 | "requires": { 1500 | "core-util-is": "~1.0.0", 1501 | "inherits": "~2.0.3", 1502 | "isarray": "~1.0.0", 1503 | "process-nextick-args": "~2.0.0", 1504 | "safe-buffer": "~5.1.1", 1505 | "string_decoder": "~1.1.1", 1506 | "util-deprecate": "~1.0.1" 1507 | } 1508 | }, 1509 | "string_decoder": { 1510 | "version": "1.1.1", 1511 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 1512 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 1513 | "requires": { 1514 | "safe-buffer": "~5.1.0" 1515 | } 1516 | } 1517 | } 1518 | }, 1519 | "parseurl": { 1520 | "version": "1.3.3", 1521 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1522 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 1523 | }, 1524 | "path-dirname": { 1525 | "version": "1.0.2", 1526 | "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", 1527 | "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=" 1528 | }, 1529 | "path-is-absolute": { 1530 | "version": "1.0.1", 1531 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1532 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" 1533 | }, 1534 | "path-to-regexp": { 1535 | "version": "3.0.0", 1536 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-3.0.0.tgz", 1537 | "integrity": "sha512-ZOtfhPttCrqp2M1PBBH4X13XlvnfhIwD7yCLx+GoGoXRPQyxGOTdQMpIzPSPKXAJT/JQrdfFrgdJOyAzvgpQ9A==" 1538 | }, 1539 | "performance-now": { 1540 | "version": "2.1.0", 1541 | "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", 1542 | "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" 1543 | }, 1544 | "process-nextick-args": { 1545 | "version": "2.0.0", 1546 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz", 1547 | "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==" 1548 | }, 1549 | "promise": { 1550 | "version": "6.1.0", 1551 | "resolved": "https://registry.npmjs.org/promise/-/promise-6.1.0.tgz", 1552 | "integrity": "sha1-LOcp9rlLRcJoka0GAsXJDgTG7vY=", 1553 | "requires": { 1554 | "asap": "~1.0.0" 1555 | } 1556 | }, 1557 | "proxy-addr": { 1558 | "version": "2.0.5", 1559 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz", 1560 | "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==", 1561 | "requires": { 1562 | "forwarded": "~0.1.2", 1563 | "ipaddr.js": "1.9.0" 1564 | } 1565 | }, 1566 | "psl": { 1567 | "version": "1.1.31", 1568 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.1.31.tgz", 1569 | "integrity": "sha512-/6pt4+C+T+wZUieKR620OpzN/LlnNKuWjy1iFLQ/UG35JqHlR/89MP1d96dUfkf6Dne3TuLQzOYEYshJ+Hx8mw==" 1570 | }, 1571 | "pump": { 1572 | "version": "3.0.0", 1573 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 1574 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 1575 | "requires": { 1576 | "end-of-stream": "^1.1.0", 1577 | "once": "^1.3.1" 1578 | } 1579 | }, 1580 | "pumpify": { 1581 | "version": "1.5.1", 1582 | "resolved": "https://registry.npmjs.org/pumpify/-/pumpify-1.5.1.tgz", 1583 | "integrity": "sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==", 1584 | "requires": { 1585 | "duplexify": "^3.6.0", 1586 | "inherits": "^2.0.3", 1587 | "pump": "^2.0.0" 1588 | }, 1589 | "dependencies": { 1590 | "pump": { 1591 | "version": "2.0.1", 1592 | "resolved": "https://registry.npmjs.org/pump/-/pump-2.0.1.tgz", 1593 | "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", 1594 | "requires": { 1595 | "end-of-stream": "^1.1.0", 1596 | "once": "^1.3.1" 1597 | } 1598 | } 1599 | } 1600 | }, 1601 | "punycode": { 1602 | "version": "2.1.1", 1603 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 1604 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 1605 | }, 1606 | "qs": { 1607 | "version": "6.5.2", 1608 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", 1609 | "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" 1610 | }, 1611 | "querystringify": { 1612 | "version": "2.1.1", 1613 | "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.1.1.tgz", 1614 | "integrity": "sha512-w7fLxIRCRT7U8Qu53jQnJyPkYZIaR4n5151KMfcJlO/A9397Wxb1amJvROTK6TOnp7PfoAmg/qXiNHI+08jRfA==" 1615 | }, 1616 | "range-parser": { 1617 | "version": "1.2.1", 1618 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 1619 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 1620 | }, 1621 | "raw-body": { 1622 | "version": "2.3.3", 1623 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.3.tgz", 1624 | "integrity": "sha512-9esiElv1BrZoI3rCDuOuKCBRbuApGGaDPQfjSflGxdy4oyzqghxu6klEkkVIvBje+FF0BX9coEv8KqW6X/7njw==", 1625 | "requires": { 1626 | "bytes": "3.0.0", 1627 | "http-errors": "1.6.3", 1628 | "iconv-lite": "0.4.23", 1629 | "unpipe": "1.0.0" 1630 | } 1631 | }, 1632 | "readable-stream": { 1633 | "version": "1.1.14", 1634 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 1635 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 1636 | "requires": { 1637 | "core-util-is": "~1.0.0", 1638 | "inherits": "~2.0.1", 1639 | "isarray": "0.0.1", 1640 | "string_decoder": "~0.10.x" 1641 | } 1642 | }, 1643 | "redis": { 1644 | "version": "2.8.0", 1645 | "resolved": "https://registry.npmjs.org/redis/-/redis-2.8.0.tgz", 1646 | "integrity": "sha512-M1OkonEQwtRmZv4tEWF2VgpG0JWJ8Fv1PhlgT5+B+uNq2cA3Rt1Yt/ryoR+vQNOQcIEgdCdfH0jr3bDpihAw1A==", 1647 | "requires": { 1648 | "double-ended-queue": "^2.1.0-0", 1649 | "redis-commands": "^1.2.0", 1650 | "redis-parser": "^2.6.0" 1651 | } 1652 | }, 1653 | "redis-commands": { 1654 | "version": "1.5.0", 1655 | "resolved": "https://registry.npmjs.org/redis-commands/-/redis-commands-1.5.0.tgz", 1656 | "integrity": "sha512-6KxamqpZ468MeQC3bkWmCB1fp56XL64D4Kf0zJSwDZbVLLm7KFkoIcHrgRvQ+sk8dnhySs7+yBg94yIkAK7aJg==" 1657 | }, 1658 | "redis-parser": { 1659 | "version": "2.6.0", 1660 | "resolved": "https://registry.npmjs.org/redis-parser/-/redis-parser-2.6.0.tgz", 1661 | "integrity": "sha1-Uu0J2srBCPGmMcB+m2mUHnoZUEs=" 1662 | }, 1663 | "regexp-clone": { 1664 | "version": "0.0.1", 1665 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-0.0.1.tgz", 1666 | "integrity": "sha1-p8LgmJH9vzj7sQ03b7cwA+aKxYk=" 1667 | }, 1668 | "reinterval": { 1669 | "version": "1.1.0", 1670 | "resolved": "https://registry.npmjs.org/reinterval/-/reinterval-1.1.0.tgz", 1671 | "integrity": "sha1-M2Hs+jymwYKDOA3Qu5VG85D17Oc=" 1672 | }, 1673 | "remove-trailing-separator": { 1674 | "version": "1.1.0", 1675 | "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", 1676 | "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=" 1677 | }, 1678 | "repeat-string": { 1679 | "version": "1.6.1", 1680 | "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", 1681 | "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=" 1682 | }, 1683 | "request": { 1684 | "version": "2.88.0", 1685 | "resolved": "https://registry.npmjs.org/request/-/request-2.88.0.tgz", 1686 | "integrity": "sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==", 1687 | "requires": { 1688 | "aws-sign2": "~0.7.0", 1689 | "aws4": "^1.8.0", 1690 | "caseless": "~0.12.0", 1691 | "combined-stream": "~1.0.6", 1692 | "extend": "~3.0.2", 1693 | "forever-agent": "~0.6.1", 1694 | "form-data": "~2.3.2", 1695 | "har-validator": "~5.1.0", 1696 | "http-signature": "~1.2.0", 1697 | "is-typedarray": "~1.0.0", 1698 | "isstream": "~0.1.2", 1699 | "json-stringify-safe": "~5.0.1", 1700 | "mime-types": "~2.1.19", 1701 | "oauth-sign": "~0.9.0", 1702 | "performance-now": "^2.1.0", 1703 | "qs": "~6.5.2", 1704 | "safe-buffer": "^5.1.2", 1705 | "tough-cookie": "~2.4.3", 1706 | "tunnel-agent": "^0.6.0", 1707 | "uuid": "^3.3.2" 1708 | } 1709 | }, 1710 | "require_optional": { 1711 | "version": "1.0.1", 1712 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz", 1713 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==", 1714 | "requires": { 1715 | "resolve-from": "^2.0.0", 1716 | "semver": "^5.1.0" 1717 | } 1718 | }, 1719 | "requires-port": { 1720 | "version": "1.0.0", 1721 | "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", 1722 | "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=" 1723 | }, 1724 | "resolve-from": { 1725 | "version": "2.0.0", 1726 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz", 1727 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=" 1728 | }, 1729 | "right-align": { 1730 | "version": "0.1.3", 1731 | "resolved": "https://registry.npmjs.org/right-align/-/right-align-0.1.3.tgz", 1732 | "integrity": "sha1-YTObci/mo1FWiSENJOFMlhSGE+8=", 1733 | "requires": { 1734 | "align-text": "^0.1.1" 1735 | } 1736 | }, 1737 | "safe-buffer": { 1738 | "version": "5.1.2", 1739 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 1740 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 1741 | }, 1742 | "safer-buffer": { 1743 | "version": "2.1.2", 1744 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1745 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1746 | }, 1747 | "saslprep": { 1748 | "version": "1.0.3", 1749 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 1750 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 1751 | "optional": true, 1752 | "requires": { 1753 | "sparse-bitfield": "^3.0.3" 1754 | } 1755 | }, 1756 | "semver": { 1757 | "version": "5.7.0", 1758 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.0.tgz", 1759 | "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==" 1760 | }, 1761 | "send": { 1762 | "version": "0.16.2", 1763 | "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", 1764 | "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", 1765 | "requires": { 1766 | "debug": "2.6.9", 1767 | "depd": "~1.1.2", 1768 | "destroy": "~1.0.4", 1769 | "encodeurl": "~1.0.2", 1770 | "escape-html": "~1.0.3", 1771 | "etag": "~1.8.1", 1772 | "fresh": "0.5.2", 1773 | "http-errors": "~1.6.2", 1774 | "mime": "1.4.1", 1775 | "ms": "2.0.0", 1776 | "on-finished": "~2.3.0", 1777 | "range-parser": "~1.2.0", 1778 | "statuses": "~1.4.0" 1779 | } 1780 | }, 1781 | "serve-static": { 1782 | "version": "1.13.2", 1783 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz", 1784 | "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==", 1785 | "requires": { 1786 | "encodeurl": "~1.0.2", 1787 | "escape-html": "~1.0.3", 1788 | "parseurl": "~1.3.2", 1789 | "send": "0.16.2" 1790 | } 1791 | }, 1792 | "setprototypeof": { 1793 | "version": "1.1.0", 1794 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", 1795 | "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==" 1796 | }, 1797 | "shortid": { 1798 | "version": "2.2.14", 1799 | "resolved": "https://registry.npmjs.org/shortid/-/shortid-2.2.14.tgz", 1800 | "integrity": "sha512-4UnZgr9gDdA1kaKj/38IiudfC3KHKhDc1zi/HSxd9FQDR0VLwH3/y79tZJLsVYPsJgIjeHjqIWaWVRJUj9qZOQ==", 1801 | "requires": { 1802 | "nanoid": "^2.0.0" 1803 | } 1804 | }, 1805 | "sift": { 1806 | "version": "7.0.1", 1807 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz", 1808 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==" 1809 | }, 1810 | "sliced": { 1811 | "version": "1.0.1", 1812 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", 1813 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" 1814 | }, 1815 | "source-map": { 1816 | "version": "0.4.4", 1817 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.4.4.tgz", 1818 | "integrity": "sha1-66T12pwNyZneaAMti092FzZSA2s=", 1819 | "requires": { 1820 | "amdefine": ">=0.0.4" 1821 | } 1822 | }, 1823 | "sparse-bitfield": { 1824 | "version": "3.0.3", 1825 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 1826 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", 1827 | "optional": true, 1828 | "requires": { 1829 | "memory-pager": "^1.0.2" 1830 | } 1831 | }, 1832 | "split2": { 1833 | "version": "3.1.1", 1834 | "resolved": "https://registry.npmjs.org/split2/-/split2-3.1.1.tgz", 1835 | "integrity": "sha512-emNzr1s7ruq4N+1993yht631/JH+jaj0NYBosuKmLcq+JkGQ9MmTw1RB1fGaTCzUuseRIClrlSLHRNYGwWQ58Q==", 1836 | "requires": { 1837 | "readable-stream": "^3.0.0" 1838 | }, 1839 | "dependencies": { 1840 | "readable-stream": { 1841 | "version": "3.4.0", 1842 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.4.0.tgz", 1843 | "integrity": "sha512-jItXPLmrSR8jmTRmRWJXCnGJsfy85mB3Wd/uINMXA65yrnFo0cPClFIUWzo2najVNSl+mx7/4W8ttlLWJe99pQ==", 1844 | "requires": { 1845 | "inherits": "^2.0.3", 1846 | "string_decoder": "^1.1.1", 1847 | "util-deprecate": "^1.0.1" 1848 | } 1849 | }, 1850 | "string_decoder": { 1851 | "version": "1.2.0", 1852 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.2.0.tgz", 1853 | "integrity": "sha512-6YqyX6ZWEYguAxgZzHGL7SsCeGx3V2TtOTqZz1xSTSWnqsbWwbptafNyvf/ACquZUXV3DANr5BDIwNYe1mN42w==", 1854 | "requires": { 1855 | "safe-buffer": "~5.1.0" 1856 | } 1857 | } 1858 | } 1859 | }, 1860 | "sshpk": { 1861 | "version": "1.16.1", 1862 | "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", 1863 | "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", 1864 | "requires": { 1865 | "asn1": "~0.2.3", 1866 | "assert-plus": "^1.0.0", 1867 | "bcrypt-pbkdf": "^1.0.0", 1868 | "dashdash": "^1.12.0", 1869 | "ecc-jsbn": "~0.1.1", 1870 | "getpass": "^0.1.1", 1871 | "jsbn": "~0.1.0", 1872 | "safer-buffer": "^2.0.2", 1873 | "tweetnacl": "~0.14.0" 1874 | } 1875 | }, 1876 | "statuses": { 1877 | "version": "1.4.0", 1878 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", 1879 | "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==" 1880 | }, 1881 | "stream-shift": { 1882 | "version": "1.0.0", 1883 | "resolved": "https://registry.npmjs.org/stream-shift/-/stream-shift-1.0.0.tgz", 1884 | "integrity": "sha1-1cdSgl5TZ+eG944Y5EXqIjoVWVI=" 1885 | }, 1886 | "string_decoder": { 1887 | "version": "0.10.31", 1888 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 1889 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 1890 | }, 1891 | "through2": { 1892 | "version": "2.0.5", 1893 | "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", 1894 | "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", 1895 | "requires": { 1896 | "readable-stream": "~2.3.6", 1897 | "xtend": "~4.0.1" 1898 | }, 1899 | "dependencies": { 1900 | "isarray": { 1901 | "version": "1.0.0", 1902 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1903 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1904 | }, 1905 | "readable-stream": { 1906 | "version": "2.3.6", 1907 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 1908 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 1909 | "requires": { 1910 | "core-util-is": "~1.0.0", 1911 | "inherits": "~2.0.3", 1912 | "isarray": "~1.0.0", 1913 | "process-nextick-args": "~2.0.0", 1914 | "safe-buffer": "~5.1.1", 1915 | "string_decoder": "~1.1.1", 1916 | "util-deprecate": "~1.0.1" 1917 | } 1918 | }, 1919 | "string_decoder": { 1920 | "version": "1.1.1", 1921 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 1922 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 1923 | "requires": { 1924 | "safe-buffer": "~5.1.0" 1925 | } 1926 | } 1927 | } 1928 | }, 1929 | "through2-filter": { 1930 | "version": "3.0.0", 1931 | "resolved": "https://registry.npmjs.org/through2-filter/-/through2-filter-3.0.0.tgz", 1932 | "integrity": "sha512-jaRjI2WxN3W1V8/FMZ9HKIBXixtiqs3SQSX4/YGIiP3gL6djW48VoZq9tDqeCWs3MT8YY5wb/zli8VW8snY1CA==", 1933 | "requires": { 1934 | "through2": "~2.0.0", 1935 | "xtend": "~4.0.0" 1936 | } 1937 | }, 1938 | "to-absolute-glob": { 1939 | "version": "2.0.2", 1940 | "resolved": "https://registry.npmjs.org/to-absolute-glob/-/to-absolute-glob-2.0.2.tgz", 1941 | "integrity": "sha1-GGX0PZ50sIItufFFt4z/fQ98hJs=", 1942 | "requires": { 1943 | "is-absolute": "^1.0.0", 1944 | "is-negated-glob": "^1.0.0" 1945 | } 1946 | }, 1947 | "tough-cookie": { 1948 | "version": "2.4.3", 1949 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.4.3.tgz", 1950 | "integrity": "sha512-Q5srk/4vDM54WJsJio3XNn6K2sCG+CQ8G5Wz6bZhRZoAe/+TxjWB/GlFAnYEbkYVlON9FMk/fE3h2RLpPXo4lQ==", 1951 | "requires": { 1952 | "psl": "^1.1.24", 1953 | "punycode": "^1.4.1" 1954 | }, 1955 | "dependencies": { 1956 | "punycode": { 1957 | "version": "1.4.1", 1958 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", 1959 | "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" 1960 | } 1961 | } 1962 | }, 1963 | "transformers": { 1964 | "version": "2.1.0", 1965 | "resolved": "https://registry.npmjs.org/transformers/-/transformers-2.1.0.tgz", 1966 | "integrity": "sha1-XSPLNVYd2F3Gf7hIIwm0fVPM6ac=", 1967 | "requires": { 1968 | "css": "~1.0.8", 1969 | "promise": "~2.0", 1970 | "uglify-js": "~2.2.5" 1971 | }, 1972 | "dependencies": { 1973 | "is-promise": { 1974 | "version": "1.0.1", 1975 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-1.0.1.tgz", 1976 | "integrity": "sha1-MVc3YcBX4zwukaq56W2gjO++duU=" 1977 | }, 1978 | "promise": { 1979 | "version": "2.0.0", 1980 | "resolved": "https://registry.npmjs.org/promise/-/promise-2.0.0.tgz", 1981 | "integrity": "sha1-RmSKqdYFr10ucMMCS/WUNtoCuA4=", 1982 | "requires": { 1983 | "is-promise": "~1" 1984 | } 1985 | }, 1986 | "source-map": { 1987 | "version": "0.1.43", 1988 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.1.43.tgz", 1989 | "integrity": "sha1-wkvBRspRfBRx9drL4lcbK3+eM0Y=", 1990 | "requires": { 1991 | "amdefine": ">=0.0.4" 1992 | } 1993 | }, 1994 | "uglify-js": { 1995 | "version": "2.2.5", 1996 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.2.5.tgz", 1997 | "integrity": "sha1-puAqcNg5eSuXgEiLe4sYTAlcmcc=", 1998 | "requires": { 1999 | "optimist": "~0.3.5", 2000 | "source-map": "~0.1.7" 2001 | } 2002 | } 2003 | } 2004 | }, 2005 | "tunnel-agent": { 2006 | "version": "0.6.0", 2007 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 2008 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 2009 | "requires": { 2010 | "safe-buffer": "^5.0.1" 2011 | } 2012 | }, 2013 | "tweetnacl": { 2014 | "version": "0.14.5", 2015 | "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", 2016 | "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=" 2017 | }, 2018 | "type-is": { 2019 | "version": "1.6.18", 2020 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 2021 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 2022 | "requires": { 2023 | "media-typer": "0.3.0", 2024 | "mime-types": "~2.1.24" 2025 | } 2026 | }, 2027 | "typedarray": { 2028 | "version": "0.0.6", 2029 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 2030 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" 2031 | }, 2032 | "uglify-js": { 2033 | "version": "2.8.29", 2034 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-2.8.29.tgz", 2035 | "integrity": "sha1-KcVzMUgFe7Th913zW3qcty5qWd0=", 2036 | "requires": { 2037 | "source-map": "~0.5.1", 2038 | "uglify-to-browserify": "~1.0.0", 2039 | "yargs": "~3.10.0" 2040 | }, 2041 | "dependencies": { 2042 | "source-map": { 2043 | "version": "0.5.7", 2044 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", 2045 | "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" 2046 | } 2047 | } 2048 | }, 2049 | "uglify-to-browserify": { 2050 | "version": "1.0.2", 2051 | "resolved": "https://registry.npmjs.org/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz", 2052 | "integrity": "sha1-bgkk1r2mta/jSeOabWMoUKD4grc=", 2053 | "optional": true 2054 | }, 2055 | "ultron": { 2056 | "version": "1.1.1", 2057 | "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.1.tgz", 2058 | "integrity": "sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og==" 2059 | }, 2060 | "unc-path-regex": { 2061 | "version": "0.1.2", 2062 | "resolved": "https://registry.npmjs.org/unc-path-regex/-/unc-path-regex-0.1.2.tgz", 2063 | "integrity": "sha1-5z3T17DXxe2G+6xrCufYxqadUPo=" 2064 | }, 2065 | "unique-stream": { 2066 | "version": "2.3.1", 2067 | "resolved": "https://registry.npmjs.org/unique-stream/-/unique-stream-2.3.1.tgz", 2068 | "integrity": "sha512-2nY4TnBE70yoxHkDli7DMazpWiP7xMdCYqU2nBRO0UB+ZpEkGsSija7MvmvnZFUeC+mrgiUfcHSr3LmRFIg4+A==", 2069 | "requires": { 2070 | "json-stable-stringify-without-jsonify": "^1.0.1", 2071 | "through2-filter": "^3.0.0" 2072 | } 2073 | }, 2074 | "unpipe": { 2075 | "version": "1.0.0", 2076 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 2077 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 2078 | }, 2079 | "uri-js": { 2080 | "version": "4.2.2", 2081 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", 2082 | "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", 2083 | "requires": { 2084 | "punycode": "^2.1.0" 2085 | } 2086 | }, 2087 | "url-parse": { 2088 | "version": "1.4.7", 2089 | "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.4.7.tgz", 2090 | "integrity": "sha512-d3uaVyzDB9tQoSXFvuSUNFibTd9zxd2bkVrDRvF5TmvWWQwqE4lgYJ5m+x1DbecWkw+LK4RNl2CU1hHuOKPVlg==", 2091 | "requires": { 2092 | "querystringify": "^2.1.1", 2093 | "requires-port": "^1.0.0" 2094 | } 2095 | }, 2096 | "util-deprecate": { 2097 | "version": "1.0.2", 2098 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 2099 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 2100 | }, 2101 | "utils-merge": { 2102 | "version": "1.0.1", 2103 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2104 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 2105 | }, 2106 | "uuid": { 2107 | "version": "3.3.2", 2108 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.2.tgz", 2109 | "integrity": "sha512-yXJmeNaw3DnnKAOKJE51sL/ZaYfWJRl1pK9dr19YFCu0ObS231AB1/LbqTKRAQ5kw8A90rA6fr4riOUpTZvQZA==" 2110 | }, 2111 | "vary": { 2112 | "version": "1.1.2", 2113 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2114 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 2115 | }, 2116 | "verror": { 2117 | "version": "1.10.0", 2118 | "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", 2119 | "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", 2120 | "requires": { 2121 | "assert-plus": "^1.0.0", 2122 | "core-util-is": "1.0.2", 2123 | "extsprintf": "^1.2.0" 2124 | } 2125 | }, 2126 | "void-elements": { 2127 | "version": "2.0.1", 2128 | "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz", 2129 | "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w=" 2130 | }, 2131 | "websocket-stream": { 2132 | "version": "5.5.0", 2133 | "resolved": "https://registry.npmjs.org/websocket-stream/-/websocket-stream-5.5.0.tgz", 2134 | "integrity": "sha512-EXy/zXb9kNHI07TIMz1oIUIrPZxQRA8aeJ5XYg5ihV8K4kD1DuA+FY6R96HfdIHzlSzS8HiISAfrm+vVQkZBug==", 2135 | "requires": { 2136 | "duplexify": "^3.5.1", 2137 | "inherits": "^2.0.1", 2138 | "readable-stream": "^2.3.3", 2139 | "safe-buffer": "^5.1.2", 2140 | "ws": "^3.2.0", 2141 | "xtend": "^4.0.0" 2142 | }, 2143 | "dependencies": { 2144 | "isarray": { 2145 | "version": "1.0.0", 2146 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 2147 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 2148 | }, 2149 | "readable-stream": { 2150 | "version": "2.3.6", 2151 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", 2152 | "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", 2153 | "requires": { 2154 | "core-util-is": "~1.0.0", 2155 | "inherits": "~2.0.3", 2156 | "isarray": "~1.0.0", 2157 | "process-nextick-args": "~2.0.0", 2158 | "safe-buffer": "~5.1.1", 2159 | "string_decoder": "~1.1.1", 2160 | "util-deprecate": "~1.0.1" 2161 | } 2162 | }, 2163 | "string_decoder": { 2164 | "version": "1.1.1", 2165 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 2166 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 2167 | "requires": { 2168 | "safe-buffer": "~5.1.0" 2169 | } 2170 | } 2171 | } 2172 | }, 2173 | "window-size": { 2174 | "version": "0.1.0", 2175 | "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.1.0.tgz", 2176 | "integrity": "sha1-VDjNLqk7IC76Ohn+iIeu58lPnJ0=" 2177 | }, 2178 | "with": { 2179 | "version": "4.0.3", 2180 | "resolved": "https://registry.npmjs.org/with/-/with-4.0.3.tgz", 2181 | "integrity": "sha1-7v0VTp550sjTQXtkeo8U2f7M4U4=", 2182 | "requires": { 2183 | "acorn": "^1.0.1", 2184 | "acorn-globals": "^1.0.3" 2185 | }, 2186 | "dependencies": { 2187 | "acorn": { 2188 | "version": "1.2.2", 2189 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-1.2.2.tgz", 2190 | "integrity": "sha1-yM4n3grMdtiW0rH6099YjZ6C8BQ=" 2191 | } 2192 | } 2193 | }, 2194 | "wordwrap": { 2195 | "version": "0.0.3", 2196 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", 2197 | "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=" 2198 | }, 2199 | "wrappy": { 2200 | "version": "1.0.2", 2201 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2202 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" 2203 | }, 2204 | "ws": { 2205 | "version": "3.3.3", 2206 | "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz", 2207 | "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==", 2208 | "requires": { 2209 | "async-limiter": "~1.0.0", 2210 | "safe-buffer": "~5.1.0", 2211 | "ultron": "~1.1.0" 2212 | } 2213 | }, 2214 | "xtend": { 2215 | "version": "4.0.1", 2216 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", 2217 | "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=" 2218 | }, 2219 | "yargs": { 2220 | "version": "3.10.0", 2221 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-3.10.0.tgz", 2222 | "integrity": "sha1-9+572FfdfB0tOMDnTvvWgdFDH9E=", 2223 | "requires": { 2224 | "camelcase": "^1.0.2", 2225 | "cliui": "^2.1.0", 2226 | "decamelize": "^1.0.0", 2227 | "window-size": "0.1.0" 2228 | } 2229 | } 2230 | } 2231 | } 2232 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "iothub-server", 3 | "version": "0.0.0", 4 | "private": true, 5 | "scripts": { 6 | "start": "node ./bin/www" 7 | }, 8 | "dependencies": { 9 | "cookie-parser": "~1.4.4", 10 | "debug": "~2.6.9", 11 | "express": "~4.16.1", 12 | "http-errors": "~1.6.3", 13 | "jade": "~1.11.0", 14 | "morgan": "~1.9.1", 15 | "mongoose": "^5.5.9", 16 | "shortid": "^2.2.14", 17 | "jsonwebtoken": "^8.5.1", 18 | "dotenv": "^8.0.0", 19 | "request": "^2.88.0", 20 | "path-to-regexp": "^3.0.0", 21 | "redis": "^2.8.0", 22 | "amqplib": "^0.5.3", 23 | "bson": "^4.0.2", 24 | "influx": "^5.0.7", 25 | "mqtt": "^3.0.0" 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /public/stylesheets/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 50px; 3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif; 4 | } 5 | 6 | a { 7 | color: #00B7FF; 8 | } 9 | -------------------------------------------------------------------------------- /routes/devices.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var Device = require("../models/device") 3 | var shortid = require("shortid") 4 | var router = express.Router(); 5 | var Connection = require('../models/connection') 6 | var UtilsService = require('../services/utils_service') 7 | var DeviceACL = require("../models/device_acl") 8 | 9 | 10 | router.post("/", function (req, res) { 11 | var productName = req.body.product_name 12 | var deviceName = shortid.generate(); 13 | var secret = shortid.generate(); 14 | var brokerUsername = `${productName}/${deviceName}` 15 | 16 | 17 | var device = new Device({ 18 | product_name: productName, 19 | device_name: deviceName, 20 | secret: secret, 21 | broker_username: brokerUsername, 22 | status: "active" 23 | }) 24 | 25 | device.save(function (err) { 26 | if (err) { 27 | res.status(500).send(err) 28 | } else { 29 | var aclRule = device.getACLRule() 30 | var deviceACL = new DeviceACL({ 31 | broker_username: device.broker_username, 32 | publish: aclRule.publish, 33 | subscribe: aclRule.subscribe, 34 | pubsub: aclRule.pubsub 35 | }) 36 | deviceACL.save(function () { 37 | res.json({product_name: productName, device_name: deviceName, secret: secret}) 38 | }) 39 | } 40 | }) 41 | }) 42 | 43 | router.get("/:productName/:deviceName", function (req, res) { 44 | var productName = req.params.productName 45 | var deviceName = req.params.deviceName 46 | Device.findOne({"product_name": productName, "device_name": deviceName}).exec(function (err, device) { 47 | if (err) { 48 | res.send(err) 49 | } else { 50 | if (device != null) { 51 | Connection.find({device: device._id}, function (_, connections) { 52 | res.json(Object.assign(device.toJSONObject(), { 53 | connections: connections.map(function (conn) { 54 | return conn.toJSONObject() 55 | }) 56 | })) 57 | }) 58 | } else { 59 | res.status(404).json({error: "Not Found"}) 60 | } 61 | } 62 | }) 63 | }) 64 | 65 | router.delete("/:productName/:deviceName", function (req, res) { 66 | var productName = req.params.productName 67 | var deviceName = req.params.deviceName 68 | Device.findOne({"product_name": productName, "device_name": deviceName}).exec(function (err, device) { 69 | if (err) { 70 | res.send(err) 71 | } else { 72 | if (device != null) { 73 | device.disconnect() 74 | device.remove() 75 | res.status(200).send("ok") 76 | } else { 77 | res.status(404).json({error: "Not Found"}) 78 | } 79 | } 80 | }) 81 | }) 82 | 83 | router.get("/:productName", function (req, res) { 84 | var productName = req.params.productName 85 | Device.find({"product_name": productName}, function (err, devices) { 86 | if (err) { 87 | res.send(err) 88 | } else { 89 | res.json(devices.map(function (device) { 90 | return device.toJSONObject() 91 | })) 92 | 93 | } 94 | }) 95 | }) 96 | 97 | router.put("/:productName/:deviceName/suspend", function (req, res) { 98 | var productName = req.params.productName 99 | var deviceName = req.params.deviceName 100 | Device.findOneAndUpdate({"product_name": productName, "device_name": deviceName}, 101 | {status: "suspended"}, {useFindAndModify: false}).exec(function (err, device) { 102 | if (err) { 103 | res.send(err) 104 | } else { 105 | if (device != null) { 106 | device.disconnect() 107 | } 108 | res.status(200).send("ok") 109 | } 110 | }) 111 | }) 112 | 113 | router.put("/:productName/:deviceName/resume", function (req, res) { 114 | var productName = req.params.productName 115 | var deviceName = req.params.deviceName 116 | Device.findOneAndUpdate({"product_name": productName, "device_name": deviceName}, 117 | {status: "active"}, {useFindAndModify: false}).exec(function (err) { 118 | if (err) { 119 | res.send(err) 120 | } else { 121 | res.status(200).send("ok") 122 | } 123 | }) 124 | }) 125 | 126 | router.post("/:productName/:deviceName/command", function (req, res) { 127 | var productName = req.params.productName 128 | var deviceName = req.params.deviceName 129 | var useRpc = (req.body.use_rpc == "true") 130 | Device.findOne({"product_name": productName, "device_name": deviceName}, function (err, device) { 131 | if (err) { 132 | res.send(err) 133 | } else if (device != null) { 134 | var ttl = req.body.ttl != null ? parseInt(req.body.ttl) : null 135 | if (useRpc) { 136 | ttl = 5 137 | } 138 | var requestId = device.sendCommand({ 139 | commandName: req.body.command, 140 | data: req.body.data, 141 | encoding: req.body.encoding || "plain", 142 | ttl: ttl, 143 | commandType: useRpc ? "rpc" : "cmd" 144 | }) 145 | if (useRpc) { 146 | UtilsService.waitKey(`cmd_resp/${requestId}`, ttl, function (val) { 147 | if (val == null) { 148 | res.status(200).json({error: "device timeout"}) 149 | } else { 150 | res.status(200).json({response: val.toString("base64")}) 151 | } 152 | }) 153 | } else { 154 | res.status(200).json({request_id: requestId}) 155 | } 156 | } else { 157 | res.status(404).send("device not found") 158 | } 159 | }) 160 | }) 161 | router.put("/:productName/:deviceName/tags", function (req, res) { 162 | var productName = req.params.productName 163 | var deviceName = req.params.deviceName 164 | var tags = req.body.tags.split(",") 165 | Device.findOne({"product_name": productName, "device_name": deviceName}, function (err, device) { 166 | if (err != null) { 167 | res.send(err) 168 | } else if (device != null) { 169 | device.tags = tags 170 | device.tags_version += 1 171 | device.save() 172 | device.sendTags() 173 | res.status(200).send("ok") 174 | } else { 175 | res.status(404).send("device not found") 176 | } 177 | 178 | }) 179 | }) 180 | 181 | router.put("/:productName/:deviceName/shadow", function (req, res) { 182 | var productName = req.params.productName 183 | var deviceName = req.params.deviceName 184 | Device.findOne({"product_name": productName, "device_name": deviceName}, function (err, device) { 185 | if (err != null) { 186 | res.send(err) 187 | } else if (device != null) { 188 | if(device.updateShadowDesired(req.body.desired, req.body.version)){ 189 | res.status(200).send("ok") 190 | }else{ 191 | res.status(409).send("version out of date") 192 | } 193 | } else { 194 | res.status(404).send("device not found") 195 | } 196 | 197 | }) 198 | }) 199 | 200 | 201 | 202 | module.exports = router -------------------------------------------------------------------------------- /routes/emqx_web_hook.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | var Device = require("../models/device") 4 | var messageService = require("../services/message_service") 5 | 6 | router.post("/", function (req, res) { 7 | switch (req.body.action){ 8 | case "client_connected": 9 | Device.addConnection(req.body) 10 | break 11 | case "client_disconnected": 12 | Device.removeConnection(req.body) 13 | break; 14 | case "message_publish": 15 | messageService.dispatchMessage({ 16 | topic: req.body.topic, 17 | payload: new Buffer(req.body.payload, 'base64'), 18 | ts: req.body.ts 19 | }) 20 | } 21 | res.status(200).send("ok") 22 | }) 23 | 24 | module.exports = router 25 | -------------------------------------------------------------------------------- /routes/messages.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | var Message = require('../models/message') 4 | 5 | router.get("/:productName", function (req, res) { 6 | var messageId = req.query.message_id 7 | var deviceName = req.query.device_name 8 | var productName = req.params.productName 9 | var query = {product_name: productName} 10 | if (messageId != null) { 11 | query.message_id = messageId 12 | } 13 | if (deviceName != null) { 14 | query.device_name = deviceName 15 | } 16 | Message.find(query, function (error, messages) { 17 | res.json({ 18 | messages: messages.map(function (message) { 19 | return message.toJSONObject() 20 | }) 21 | }) 22 | }) 23 | }) 24 | module.exports = router -------------------------------------------------------------------------------- /routes/ota.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | var Device = require("../models/device") 4 | var OTAService = require("../services/ota_service") 5 | router.post("/:productName/:deviceName", function (req, res) { 6 | var productName = req.params.productName 7 | var deviceName = req.params.deviceName 8 | Device.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 9 | if (err) { 10 | res.send(err) 11 | } else if (device != null) { 12 | OTAService.sendOTA({ 13 | productName: device.product_name, 14 | deviceName: device.device_name, 15 | fileUrl: req.body.url, 16 | size: parseInt(req.body.size), 17 | md5: req.body.md5, 18 | version: req.body.version, 19 | type: req.body.type 20 | }) 21 | res.status(200).send("ok") 22 | } else { 23 | res.status(400).send("device not found") 24 | } 25 | }) 26 | }) 27 | 28 | router.get("/:productName/:deviceName", function (req, res) { 29 | var productName = req.params.productName 30 | var deviceName = req.params.deviceName 31 | OTAService.getProgress(productName, deviceName, function (progress) { 32 | res.status(200).json(progress) 33 | }) 34 | }) 35 | 36 | module.exports = router -------------------------------------------------------------------------------- /routes/tags.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | const emqxService = require("../services/emqx_service") 4 | const ObjectId = require('bson').ObjectID; 5 | 6 | router.post("/:productName/:tag/command", function (req, res) { 7 | var productName = req.params.productName 8 | var ttl = req.body.ttl != null ? parseInt(req.body.ttl) : null 9 | var commandName = req.body.command 10 | var encoding = req.body.encoding || "plain" 11 | var data = req.body.data 12 | var requestId = new ObjectId().toHexString() 13 | var topic = `tags/${productName}/${req.params.tag}/cmd/${commandName}/${encoding}/${requestId}` 14 | if (ttl != null) { 15 | topic = `${topic}/${Math.floor(Date.now() / 1000) + ttl}` 16 | } 17 | emqxService.publishTo({topic: topic, payload: data}) 18 | res.status(200).json({request_id: requestId}) 19 | }) 20 | module.exports = router -------------------------------------------------------------------------------- /routes/tokens.js: -------------------------------------------------------------------------------- 1 | var express = require('express'); 2 | var router = express.Router(); 3 | var shortid = require("shortid") 4 | var jwt = require('jsonwebtoken') 5 | 6 | //这个值应该和EMQX etc/plugins/emqx_auth_jwt.conf中的保存一致 7 | const jwtSecret = process.env.JWT_SECRET 8 | 9 | router.post("/", function (_, res) { 10 | var username = shortid.generate() 11 | var password = jwt.sign({ 12 | username: username, 13 | exp: Math.floor(Date.now() / 1000) + 10 14 | }, jwtSecret) 15 | res.json({username: username, password: password}) 16 | }) 17 | 18 | module.exports = router 19 | -------------------------------------------------------------------------------- /samples/business_sim.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config({path: "../.env"}) 2 | 3 | const bson = require('bson') 4 | var amqp = require('amqplib/callback_api'); 5 | var uploadDataExchange = "iothub.events.upload_data" 6 | var updateStatusExchange = "iothub.events.update_status" 7 | amqp.connect(process.env.RABBITMQ_URL, function (error0, connection) { 8 | if (error0) { 9 | console.log(error0); 10 | } else { 11 | connection.createChannel(function (error1, channel) { 12 | if (error1) { 13 | console.log(error1) 14 | } else { 15 | channel.assertExchange(uploadDataExchange, 'direct', {durable: true}) 16 | var queue = "iotapp_upload_data"; 17 | channel.assertQueue(queue, { 18 | durable: true 19 | }) 20 | channel.bindQueue(queue, uploadDataExchange, "IotApp") 21 | channel.consume(queue, function (msg) { 22 | var data = bson.deserialize(msg.content) 23 | console.log(`received from ${data.device_name}, messageId: ${data.message_id},payload: ${data.payload.toString()}`) 24 | channel.ack(msg) 25 | }) 26 | 27 | channel.assertExchange(updateStatusExchange, 'direct', {durable: true}) 28 | var queue = "iotapp_update_status"; 29 | channel.assertQueue(queue, { 30 | durable: true 31 | }) 32 | channel.bindQueue(queue, updateStatusExchange, "IotApp") 33 | channel.consume(queue, function (msg) { 34 | var data = bson.deserialize(msg.content) 35 | console.log(`received from ${data.device_name}, status: ${data.device_status}`) 36 | channel.ack(msg) 37 | }) 38 | } 39 | }); 40 | } 41 | }); -------------------------------------------------------------------------------- /samples/perform_ota.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config({path: "../.env"}) 2 | const request = require("request") 3 | var otaData = { 4 | type: "firmware", 5 | url: "http://test.com/firmware/1.1.pkg", 6 | version: "1.1", 7 | size: 1000, 8 | md5: "abcd" 9 | } 10 | var progress = 0 11 | var checkUpgradeProgress = function () { 12 | if (progress < 100) { 13 | request.get(`http://127.0.0.1:3000/ota/${process.env.TARGET_PRODUCT_NAME}/${process.env.TARGET_DEVICE_NAME}`, function (err, res, body) { 14 | if (!err && res.statusCode == 200) { 15 | var info = JSON.parse(body); 16 | if(info.version == otaData.version) { 17 | progress = info.progress 18 | console.log(`current progress:${progress}%`); 19 | } 20 | setTimeout(checkUpgradeProgress, 1000) 21 | } 22 | }) 23 | } else { 24 | request.get(`http://127.0.0.1:3000/devices/${process.env.TARGET_PRODUCT_NAME}/${process.env.TARGET_DEVICE_NAME}`, function (err, res, body) { 25 | if (!err && res.statusCode == 200) { 26 | var info = JSON.parse(body); 27 | console.log(`current version:${info.device_status.firmware_ver}`); 28 | if (info.device_status.firmware_ver == otaData.version) { 29 | console.log(`upgrade completed`); 30 | } else { 31 | setTimeout(checkUpgradeProgress, 1000) 32 | } 33 | } 34 | }) 35 | } 36 | } 37 | 38 | console.log("perform upgrade") 39 | request.post(`http://127.0.0.1:3000/ota/${process.env.TARGET_PRODUCT_NAME}/${process.env.TARGET_DEVICE_NAME}`, { 40 | form: otaData 41 | }, function (error, response) { 42 | if (error) { 43 | console.log(error) 44 | } else { 45 | console.log('statusCode:', response && response.statusCode); 46 | checkUpgradeProgress() 47 | } 48 | }) 49 | 50 | 51 | -------------------------------------------------------------------------------- /samples/ping.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config({path: "../.env"}) 2 | const bson = require('bson') 3 | const request = require("request") 4 | var amqp = require('amqplib/callback_api'); 5 | var exchange = "iothub.events.cmd_resp" 6 | amqp.connect(process.env.RABBITMQ_URL, function (error0, connection) { 7 | if (error0) { 8 | console.log(error0); 9 | } else { 10 | connection.createChannel(function (error1, channel) { 11 | if (error1) { 12 | console.log(error1) 13 | } else { 14 | channel.assertExchange(exchange, 'direct', {durable: true}) 15 | var queue = "iotapp_cmd_resp"; 16 | channel.assertQueue(queue, { 17 | durable: true 18 | }) 19 | channel.bindQueue(queue, exchange, process.env.TARGET_PRODUCT_NAME) 20 | channel.consume(queue, function (msg) { 21 | var data = bson.deserialize(msg.content) 22 | if(data.command == "ping") { 23 | console.log(`received from ${data.device_name}, requestId: ${data.request_id},payload: ${data.payload.buffer.readUInt32BE(0)}`) 24 | } 25 | channel.ack(msg) 26 | }) 27 | } 28 | }); 29 | } 30 | }); 31 | 32 | const buf = Buffer.alloc(4); 33 | buf.writeUInt32BE(Math.floor(Date.now())/1000, 0); 34 | var formData = { 35 | command: "ping", 36 | data: buf.toString("base64"), 37 | encoding: "base64" 38 | } 39 | if(process.argv[2] != null){ 40 | formData.ttl = process.argv[2] 41 | } 42 | request.post(`http://127.0.0.1:3000/devices/${process.env.TARGET_PRODUCT_NAME}/${process.env.TARGET_DEVICE_NAME}/command`, { 43 | form: formData 44 | }, function (error, response, body) { 45 | if (error) { 46 | console.log(error) 47 | } else { 48 | console.log('statusCode:', response && response.statusCode); 49 | console.log('body:', body); 50 | } 51 | }) 52 | 53 | 54 | -------------------------------------------------------------------------------- /samples/resp_to_data_request.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config({path: "../.env"}) 2 | const bson = require('bson') 3 | const request = require("request") 4 | var amqp = require('amqplib/callback_api'); 5 | var exchange = "iothub.events.data_request" 6 | amqp.connect(process.env.RABBITMQ_URL, function (error0, connection) { 7 | if (error0) { 8 | console.log(error0); 9 | } else { 10 | connection.createChannel(function (error1, channel) { 11 | if (error1) { 12 | console.log(error1) 13 | } else { 14 | channel.assertExchange(exchange, 'direct', {durable: true}) 15 | var queue = "iotapp_data_request"; 16 | channel.assertQueue(queue, { 17 | durable: true 18 | }) 19 | channel.bindQueue(queue, exchange, process.env.TARGET_PRODUCT_NAME) 20 | channel.consume(queue, function (msg) { 21 | var data = bson.deserialize(msg.content) 22 | if (data.resource == "weather") { 23 | console.log(`received request for weather from ${data.device_name}`) 24 | request.post(`http://127.0.0.1:3000/devices/${process.env.TARGET_PRODUCT_NAME}/${data.device_name}/command`, { 25 | form: { 26 | command: "weather", 27 | data: JSON.stringify({temp: 25, wind: 4}), 28 | } 29 | }, function (error, response, body) { 30 | if (error) { 31 | console.log(error) 32 | } else { 33 | console.log('statusCode:', response && response.statusCode); 34 | console.log('body:', body); 35 | } 36 | }) 37 | } 38 | channel.ack(msg) 39 | }) 40 | } 41 | }); 42 | } 43 | }); 44 | 45 | 46 | 47 | -------------------------------------------------------------------------------- /samples/rpc_ping.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config({path: "../.env"}) 2 | const request = require("request") 3 | const buf = Buffer.alloc(4); 4 | buf.writeUInt32BE(Math.floor(Date.now())/1000, 0); 5 | var formData = { 6 | command: "ping", 7 | data: buf.toString("base64"), 8 | encoding: "base64", 9 | use_rpc: true 10 | } 11 | request.post(`http://127.0.0.1:3000/devices/${process.env.TARGET_PRODUCT_NAME}/${process.env.TARGET_DEVICE_NAME}/command`, { 12 | form: formData 13 | }, function (error, response, body) { 14 | if (error) { 15 | console.log(error) 16 | } else { 17 | console.log('statusCode:', response && response.statusCode); 18 | var result = JSON.parse(body) 19 | if(result.error != null){ 20 | console.log(result.error) 21 | }else{ 22 | console.log('response:', Buffer.from(result.response, "base64").readUInt32BE(0)); 23 | } 24 | } 25 | }) 26 | 27 | 28 | -------------------------------------------------------------------------------- /samples/update_shadow.js: -------------------------------------------------------------------------------- 1 | require('dotenv').config({path: "../.env"}) 2 | const request = require("request") 3 | 4 | var deviceUrl = `http://127.0.0.1:3000/devices/${process.env.TARGET_PRODUCT_NAME}/${process.env.TARGET_DEVICE_NAME}`; 5 | 6 | var checkLights = function () { 7 | request.get(deviceUrl 8 | , function (err, response, body) { 9 | var shadow = JSON.parse(body).shadow 10 | var lightsStatus = "unknown" 11 | if(shadow.state.reported && shadow.state.reported.lights){ 12 | lightsStatus = shadow.state.reported.lights 13 | } 14 | console.log(`current lights status is ${lightsStatus}`) 15 | setTimeout(checkLights, 2000) 16 | }) 17 | } 18 | request.get(deviceUrl 19 | , function (err, response, body) { 20 | var deviceInfo = JSON.parse(body) 21 | request.put(`${deviceUrl}/shadow`, { 22 | json: { 23 | version: deviceInfo.shadow.version + 1, 24 | desired: {lights: "on"} 25 | } 26 | }, function (err, response, body) { 27 | checkLights() 28 | }) 29 | }) -------------------------------------------------------------------------------- /services/emqx_service.js: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | const request = require('request'); 3 | var shortid = require("shortid") 4 | 5 | class EMQXService { 6 | static disconnectClient(clientId) { 7 | const apiUrl = `${process.env.EMQX_API_URL}/connections/${clientId}` 8 | request.delete(apiUrl, { 9 | "auth": { 10 | 'user': process.env.EMQX_APP_ID, 11 | 'pass': process.env.EMQX_APP_SECRET, 12 | 'sendImmediately': true 13 | } 14 | }, function (error, response, body) { 15 | console.log('statusCode:', response && response.statusCode); 16 | console.log('body:', body); 17 | }) 18 | } 19 | 20 | static publishTo({topic, payload, qos = 1, retained = false}) { 21 | const apiUrl = `${process.env.EMQX_API_URL}/mqtt/publish` 22 | request.post(apiUrl, { 23 | "auth": { 24 | 'user': process.env.EMQX_APP_ID, 25 | 'pass': process.env.EMQX_APP_SECRET, 26 | 'sendImmediately': true 27 | }, 28 | json:{ 29 | topic: topic, 30 | payload: payload, 31 | qos: qos, 32 | retained: retained, 33 | client_id: shortid.generate() 34 | } 35 | }, function (error, response, body) { 36 | console.log(`published to ${topic}`) 37 | console.log('statusCode:', response && response.statusCode); 38 | console.log('body:', body); 39 | }) 40 | } 41 | } 42 | 43 | module.exports = EMQXService -------------------------------------------------------------------------------- /services/influxdb_service.js: -------------------------------------------------------------------------------- 1 | const Influx = require('influx') 2 | const influx = new Influx.InfluxDB({ 3 | host: process.env.INFLUXDB, 4 | database: 'iothub', 5 | schema: [ 6 | { 7 | measurement: 'device_connections', 8 | fields: { 9 | connected: Influx.FieldType.BOOLEAN 10 | }, 11 | tags: [ 12 | 'product_name', 'device_name' 13 | ] 14 | }, 15 | { 16 | measurement: 'connection_count', 17 | fields: { 18 | count: Influx.FieldType.INTEGER 19 | }, 20 | tags: ["node_name"] 21 | } 22 | ] 23 | }) 24 | 25 | class InfluxDBService { 26 | static writeConnectionData({productName, deviceName, connected, ts}) { 27 | var timestamp = ts == null ? Math.floor(Date.now() / 1000) : ts 28 | influx.writePoints([ 29 | { 30 | measurement: 'device_connections', 31 | tags: {product_name: productName, device_name: deviceName}, 32 | fields: {connected: connected}, 33 | timestamp: timestamp 34 | } 35 | ], { 36 | precision: 's', 37 | }).catch(err => { 38 | console.error(`Error saving data to InfluxDB! ${err.stack}`) 39 | }) 40 | } 41 | 42 | static writeConnectionCount(nodeName, count) { 43 | influx.writePoints([ 44 | { 45 | measurement: 'connection_count', 46 | tags: {node_name: nodeName}, 47 | fields: {count: count}, 48 | timestamp: Math.floor(Date.now() / 1000) 49 | } 50 | ], { 51 | precision: 's', 52 | }).catch(err => { 53 | console.error(`Error saving data to InfluxDB! ${err.stack}`) 54 | }) 55 | } 56 | } 57 | 58 | module.exports = InfluxDBService -------------------------------------------------------------------------------- /services/message_service.js: -------------------------------------------------------------------------------- 1 | const redisClient = require("../models/redis") 2 | const pathToRegexp = require('path-to-regexp') 3 | const Message = require("../models/message") 4 | const NotifyService = require("./notify_service") 5 | const Device = require("../models/device") 6 | const OTAService = require("../services/ota_service") 7 | 8 | class MessageService { 9 | static checkMessageDuplication(messageId, callback) { 10 | var key = `/messageIDs/${messageId}` 11 | redisClient.setnx(key, "", function (err, res) { 12 | if (res == 1) { 13 | redisClient.expire(key, 60 * 60 * 6) 14 | callback.call(this, false) 15 | } else { 16 | callback.call(this, true) 17 | } 18 | }) 19 | } 20 | 21 | static dispatchMessage({topic, payload, ts} = {}) { 22 | var dataTopicRule = "upload_data/:productName/:deviceName/:dataType/:messageId"; 23 | var statusTopicRule = "(update_status|update_ota_status)/:productName/:deviceName/:messageId" 24 | var cmdRespRule = "(cmd_resp|rpc_resp)/:productName/:deviceName/:commandName/:requestId/:messageId" 25 | var dataRequestTopicRule = "get/:productName/:deviceName/:resource/:messageId" 26 | const topicRegx = pathToRegexp(dataTopicRule) 27 | const statusRegx = pathToRegexp(statusTopicRule) 28 | const cmdRespRegx = pathToRegexp(cmdRespRule) 29 | const dataRequestRegx = pathToRegexp(dataRequestTopicRule) 30 | var result = null; 31 | if ((result = topicRegx.exec(topic)) != null) { 32 | this.checkMessageDuplication(result[4], function (isDup) { 33 | if (!isDup) { 34 | MessageService.handleUploadData({ 35 | productName: result[1], 36 | deviceName: result[2], 37 | dataType: result[3], 38 | messageId: result[4], 39 | ts: ts, 40 | payload: payload 41 | }) 42 | } 43 | }) 44 | } else if ((result = statusRegx.exec(topic)) != null) { 45 | this.checkMessageDuplication(result[4], function (isDup) { 46 | if (!isDup) { 47 | if (result[1] == "update_status") { 48 | MessageService.handleUpdateStatus({ 49 | productName: result[2], 50 | deviceName: result[3], 51 | deviceStatus: payload.toString(), 52 | ts: ts 53 | }) 54 | } else if (result[1] == "update_ota_status") { 55 | var progress = JSON.parse(payload.toString()) 56 | progress.ts = ts 57 | OTAService.updateProgress(result[2], result[3], progress) 58 | } 59 | } 60 | }) 61 | } else if ((result = cmdRespRegx.exec(topic)) != null) { 62 | this.checkMessageDuplication(result[6], function (isDup) { 63 | if (!isDup) { 64 | if (result[1] == "rpc_resp") { 65 | var key = `cmd_resp/${result[5]}`; 66 | redisClient.set(key, payload) 67 | redisClient.expire(key, 5) 68 | } else { 69 | MessageService.handleCommandResp({ 70 | productName: result[2], 71 | deviceName: result[3], 72 | ts: ts, 73 | command: result[4], 74 | requestId: result[5], 75 | payload: payload 76 | }) 77 | } 78 | } 79 | }) 80 | } else if ((result = dataRequestRegx.exec(topic)) != null) { 81 | this.checkMessageDuplication(result[4], function (isDup) { 82 | if (!isDup) { 83 | MessageService.handleDataRequest({ 84 | productName: result[1], 85 | deviceName: result[2], 86 | resource: result[3], 87 | payload: payload, 88 | ts: ts 89 | }) 90 | } 91 | }) 92 | } 93 | } 94 | 95 | static handleDataRequest({productName, deviceName, resource, payload, ts}) { 96 | if (resource.startsWith("$")) { 97 | if (resource == "$ntp") { 98 | this.handleNTP({ 99 | payload: JSON.parse(payload.toString()), 100 | ts: ts, 101 | productName: productName, 102 | deviceName: deviceName 103 | }) 104 | } else if (resource == "$tags") { 105 | Device.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 106 | if (device != null) { 107 | var data = JSON.parse(payload.toString()) 108 | if (data.tags_version < device.tags_version) { 109 | device.sendTags() 110 | } 111 | } 112 | }) 113 | } else if (resource == "$shadow") { 114 | Device.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 115 | if (device != null) { 116 | device.sendUpdateShadow() 117 | } 118 | }) 119 | } 120 | } else { 121 | NotifyService.notifyDataRequest({ 122 | productName: productName, 123 | deviceName: deviceName, 124 | resource: resource, 125 | payload: payload 126 | }) 127 | } 128 | } 129 | 130 | static handleNTP({payload, ts, productName, deviceName}) { 131 | var data = { 132 | device_time: payload.device_time, 133 | iothub_recv: ts * 1000, 134 | iothub_send: Date.now() 135 | } 136 | Device.sendCommand({ 137 | productName: productName, 138 | deviceName: deviceName, 139 | data: JSON.stringify(data), 140 | commandName: "$set_ntp" 141 | }) 142 | } 143 | 144 | static handleUploadData({productName, deviceName, ts, payload, messageId, dataType} = {}) { 145 | if (dataType.startsWith("$")) { 146 | if (dataType == "$shadow_updated") { 147 | Device.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 148 | if (device != null) { 149 | device.updateShadow(JSON.parse(payload.toString())) 150 | } 151 | }) 152 | }else if("$shadow_updated"){ 153 | Device.findOne({product_name: productName, device_name: deviceName}, function (err, device) { 154 | if (device != null) { 155 | device.reportShadow(JSON.parse(payload.toString())) 156 | } 157 | }) 158 | } 159 | } else { 160 | var message = new Message({ 161 | product_name: productName, 162 | device_name: deviceName, 163 | payload: payload, 164 | message_id: messageId, 165 | data_type: dataType, 166 | sent_at: ts 167 | }) 168 | message.save() 169 | NotifyService.notifyUploadData(message) 170 | } 171 | } 172 | 173 | static handleUpdateStatus({productName, deviceName, deviceStatus, ts}) { 174 | Device.findOneAndUpdate({ 175 | product_name: productName, device_name: deviceName, 176 | "$or": [{last_status_update: {"$exists": false}}, {last_status_update: {"$lt": ts}}] 177 | }, 178 | { 179 | device_status: deviceStatus, 180 | last_status_update: ts 181 | }, {useFindAndModify: false}).exec(function (error, device) { 182 | if (device != null) { 183 | NotifyService.notifyUpdateStatus({ 184 | productName: productName, 185 | deviceName: deviceName, 186 | deviceStatus: deviceStatus 187 | }) 188 | } 189 | }) 190 | } 191 | 192 | static handleCommandResp({productName, deviceName, command, requestId, ts, payload}) { 193 | NotifyService.notifyCommandResp({ 194 | productName: productName, 195 | deviceName: deviceName, 196 | command: command, 197 | requestId: requestId, 198 | ts: ts, 199 | payload: payload 200 | }) 201 | } 202 | } 203 | 204 | module.exports = MessageService -------------------------------------------------------------------------------- /services/notify_service.js: -------------------------------------------------------------------------------- 1 | const bson = require('bson') 2 | var amqp = require('amqplib/callback_api'); 3 | var uploadDataExchange = "iothub.events.upload_data" 4 | var updateStatusExchange = "iothub.events.update_status" 5 | var commandRespExchange = "iothub.events.cmd_resp" 6 | var dataRequestRespExchange = "iothub.events.data_request" 7 | var currentChannel = null; 8 | amqp.connect(process.env.RABBITMQ_URL, function (error0, connection) { 9 | if (error0) { 10 | console.log(error0); 11 | } else { 12 | connection.createChannel(function (error1, channel) { 13 | if (error1) { 14 | console.log(error1) 15 | } else { 16 | currentChannel = channel; 17 | channel.assertExchange(uploadDataExchange, 'direct', {durable: true}) 18 | channel.assertExchange(updateStatusExchange, 'direct', {durable: true}) 19 | channel.assertExchange(commandRespExchange, 'direct', {durable: true}) 20 | channel.assertExchange(dataRequestRespExchange, 'direct', {durable: true}) 21 | } 22 | }); 23 | } 24 | }); 25 | 26 | class NotifyService { 27 | static notifyUploadData(message) { 28 | var data = bson.serialize({ 29 | device_name: message.device_name, 30 | payload: message.payload, 31 | send_at: message.sendAt, 32 | data_type: message.dataType, 33 | message_id: message.message_id 34 | }) 35 | if(currentChannel != null) { 36 | currentChannel.publish(uploadDataExchange, message.product_name, data, { 37 | persistent: true 38 | }) 39 | } 40 | } 41 | 42 | static notifyUpdateStatus({productName, deviceName, deviceStatus}){ 43 | var data = bson.serialize({ 44 | device_name: deviceName, 45 | device_status: deviceStatus 46 | }) 47 | if(currentChannel != null) { 48 | currentChannel.publish(updateStatusExchange, productName, data, { 49 | persistent: true 50 | }) 51 | } 52 | } 53 | 54 | static notifyCommandResp({productName, deviceName, command, requestId, ts, payload}){ 55 | var data = bson.serialize({ 56 | device_name: deviceName, 57 | command: command, 58 | request_id: requestId, 59 | send_at: ts, 60 | payload: payload 61 | }) 62 | if(currentChannel != null){ 63 | currentChannel.publish(commandRespExchange, productName, data) 64 | } 65 | } 66 | 67 | static notifyDataRequest({productName, deviceName, resource, payload}){ 68 | var data = bson.serialize({ 69 | device_name: deviceName, 70 | resource: resource, 71 | payload: payload 72 | }) 73 | if(currentChannel != null){ 74 | currentChannel.publish(dataRequestRespExchange, productName, data) 75 | } 76 | } 77 | } 78 | 79 | module.exports = NotifyService 80 | -------------------------------------------------------------------------------- /services/ota_service.js: -------------------------------------------------------------------------------- 1 | const redisClient = require("../models/redis") 2 | const Device = require("../models/device") 3 | 4 | class OTAService { 5 | static updateProgress(productName, deviceName, progress) { 6 | redisClient.set(`ota_progress/${productName}/${deviceName}`, JSON.stringify(progress)) 7 | } 8 | 9 | static sendOTA({productName, deviceName = null, tag = null, fileUrl, version, size, md5, type}) { 10 | var data = JSON.stringify({ 11 | url: fileUrl, 12 | version: version, 13 | size: size, 14 | md5: md5, 15 | type: type 16 | }) 17 | if (deviceName != null) { 18 | Device.sendCommand({ 19 | productName: productName, 20 | deviceName: deviceName, 21 | commandName: "$ota_upgrade", 22 | data: data 23 | }) 24 | } else if (tag != null) { 25 | Device.sendCommandByTag({ 26 | productName: productName, 27 | tag: tag, 28 | commandName: "$ota_upgrade", 29 | data: data 30 | }) 31 | } 32 | } 33 | 34 | static getProgress(productName, deviceName, callback) { 35 | redisClient.get(`ota_progress/${productName}/${deviceName}`, function (err, value) { 36 | if (value != null) { 37 | callback(JSON.parse(value)) 38 | } else { 39 | callback({}) 40 | } 41 | }) 42 | } 43 | } 44 | 45 | module.exports = OTAService -------------------------------------------------------------------------------- /services/utils_service.js: -------------------------------------------------------------------------------- 1 | const redisClient = require("../models/redis") 2 | class UtilsService { 3 | static waitKey(key, ttl, callback) { 4 | var end = Date.now() + ttl * 1000 5 | function checkKey() { 6 | if (Date.now() < end) { 7 | redisClient.get(key, function (err, val) { 8 | if (val != null) { 9 | callback(val) 10 | } else { 11 | setTimeout(checkKey, 10) 12 | } 13 | }) 14 | } else { 15 | callback(null) 16 | } 17 | 18 | } 19 | checkKey() 20 | } 21 | } 22 | 23 | module.exports = UtilsService -------------------------------------------------------------------------------- /views/error.jade: -------------------------------------------------------------------------------- 1 | extends layout 2 | 3 | block content 4 | h1= message 5 | h2= error.status 6 | pre #{error.stack} 7 | -------------------------------------------------------------------------------- /views/layout.jade: -------------------------------------------------------------------------------- 1 | doctype html 2 | html 3 | head 4 | title= title 5 | link(rel='stylesheet', href='/stylesheets/style.css') 6 | body 7 | block content 8 | --------------------------------------------------------------------------------