├── .gitignore ├── LICENSE ├── README.md ├── app.js ├── bin └── modbus-ws.js ├── cache.js ├── examples └── example.html ├── package-lock.json ├── package.json ├── public └── stylesheets │ └── style.css ├── server.js ├── test └── test.js └── views └── index.ejs /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | 5 | # Runtime data 6 | pids 7 | *.pid 8 | *.seed 9 | 10 | # Directory for instrumented libs generated by jscoverage/JSCover 11 | lib-cov 12 | 13 | # Coverage directory used by tools like istanbul 14 | coverage 15 | 16 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 17 | .grunt 18 | 19 | # node-waf configuration 20 | .lock-wscript 21 | 22 | # Compiled binary addons (http://nodejs.org/api/addons.html) 23 | build/Release 24 | 25 | # Dependency directory 26 | # https://www.npmjs.org/doc/misc/npm-faq.html#should-i-check-my-node_modules-folder-into-git 27 | node_modules 28 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2015, 2 | 3 | Permission to use, copy, modify, and/or distribute this software for any 4 | purpose with or without fee is hereby granted, provided that the above 5 | copyright notice and this permission notice appear in all copies. 6 | 7 | THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 8 | WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 9 | MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 10 | ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 11 | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 12 | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 13 | OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 14 | 15 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # node-modbus-ws 2 | NodeJS Modbus to WebSocket bridge 3 | 4 | [![NPM Version](https://img.shields.io/npm/v/gm.svg?style=flat)](https://www.npmjs.com/package/modbus-ws) 5 | 6 | Control your modbus enabled arduino project, toaster or robot via web browser. 7 | 8 | The modbus-ws server allows a browser to connect to a modbus device, using websockets. 9 | When the server is running, and connected to Serial line or Ethernet, 10 | a web browser can send web socket requests and control a modbus device. 11 | 12 | With this server you can build web pages that will easily monitor and send requests to your modbus project or robot. 13 | 14 | - [node-modbus-ws](#node-modbus-ws) 15 | - [Install](#install) 16 | - [Start the modbus-ws server](#start-the-modbus-ws-server) 17 | - [Server help](#server-help) 18 | - [Client side code](#client-side-code) 19 | - [Examples](#examples) 20 | 21 | #### Install 22 | Install the server: 23 | ``` 24 | npm install modbus-ws -g 25 | ``` 26 | If npm install fails because of root permissions try: 27 | ``` 28 | sudo npm install --unsafe-perm modbus-ws -g 29 | ``` 30 | 31 | This will add the **modbus-ws** command to your path. After install, you can run the server by typing **modbus-ws** on the command line. 32 | 33 | [ If install locally, run the server using the **modbus-ws.js** file in the bin directory. ] 34 | 35 | #### Start the modbus-ws server 36 | 37 | Run the server from the command line: 38 | ``` 39 | # use serial line (modbus-rtu) 40 | modbus-ws -s /dev/ttyUSB0 41 | # or tcp/ip (modbus-tcp) 42 | modbus-ws -i 192.168.1.42 43 | ``` 44 | 45 | After running the server will print out this message to the console: 46 | ``` 47 | ---------------------------------------------------- 48 | Modbus-WS server 1.1.1 49 | Setup serial port: /dev/ttyUSB0 9600 50 | Server is running, http://127.0.0.1:3000 51 | Setup modbus with caching. 52 | Create cache db. 53 | Create cache table 54 | ---------------------------------------------------- 55 | ``` 56 | 57 | Ctrl+C will stop the server. 58 | 59 | #### Server help 60 | ``` 61 | modbus-ws --help 62 | ``` 63 | 64 | ``` 65 | Usage: modbus-ws [options] 66 | 67 | Options: 68 | 69 | -h, --help output usage information 70 | -V, --version output the version number 71 | -s, --serial Use serial port, set port. [false] 72 | -b, --baudrate Set serial port baudrate. [9600] 73 | -i, --ip Use tcp/ip, set slave url or ip address. [false] 74 | -P, --tcpport Server port number [3000] 75 | -c, --nocache Do not use caching for modbus comunication. [false] 76 | -w, --nohttp Run only websocket server, no httpd. [false] 77 | 78 | Examples: 79 | 80 | modbus-ws --ip 192.168.1.24 81 | create a bridge to a modbus slave using tcp/ip. 82 | modbus-ws --serial /dev/ttyUSB0 83 | create a bridge to a modbus slave using a serial port. 84 | modbus-ws 85 | when serial and tcp/ip are not used, default to test. 86 | create a bridge to a modbus simulated slave. 87 | modbus-ws --ip 192.168.1.24 --nocache 88 | create a bridge with modbus without cache. 89 | modbus-ws --ip 192.168.1.24 --nocache --nohttp 90 | create a bridge with modbus without cache and without http web server. 91 | ``` 92 | 93 | #### Client side code 94 | 95 | ###### WebSocket Events 96 | 97 | **From server to client** 98 | 99 | * 'data' - data sent to client. 100 | * 'error' - an error message. 101 | 102 | **From client to server** 103 | 104 | * 'readCoils' 105 | * 'readDiscreteInputs' 106 | * 'readHoldingRegisters' 107 | * 'readInputRegisters' 108 | * 'writeCoil' 109 | * 'writeRegisters' 110 | 111 | ###### Get the socket.io code: 112 | ``` 113 | 114 | ``` 115 | 116 | ###### Use socket.io events, server will replay with a "data" event. 117 | 118 | ```javascript 119 | // connect to sever 120 | var socket = io("ws://127.0.0.1:3000/"); 121 | 122 | // set up socket.on for data received from sever 123 | // server trigger 'data' event when data is received from device. 124 | socket.on('data', function(data){ 125 | console.log('received:', data); 126 | 127 | ... do something fun and interesting with data ... 128 | }); 129 | 130 | // ask server to get registers 131 | // "Read Input Registers" (FC=04) 132 | socket.emit("readInputRegisters", { 133 | "unit": 1, 134 | "address": 0, 135 | "length": 10 136 | }); 137 | 138 | // subscribe to get holding registers every 1000ms 139 | // "Read Holding Registers" (FC=04) 140 | socket.emit("readHoldingRegisters", { 141 | "unit": 1, 142 | "address": 0, 143 | "length": 10, 144 | "interval": 1000 145 | }); 146 | 147 | // ask server to set one coil 148 | // "Force one coil" (FC=5) 149 | socket.emit('writeCoil', { 150 | "unit": 1, 151 | "address": 8, 152 | "state": true 153 | }); 154 | 155 | // ask server to set registers 156 | // "Preset Multiple Registers" (FC=16) 157 | socket.emit('writeRegisters', { 158 | "unit": 1, 159 | "address": 8, 160 | "values": [88,123,47] 161 | }); 162 | 163 | // ask server to get coils 164 | // "Read coils" (FC=01) 165 | socket.emit("readCoils", { 166 | "unit": 1, 167 | "address": 0, 168 | "length": 8 169 | }); 170 | ``` 171 | 172 | #### Examples 173 | 174 | When the server is running it will transfer web socket requests into modbus requests and return the replays received as web socket events. 175 | 176 | See the examples directory in the server's code tree, load an example file to a web browser, and watch for the data received from server. 177 | 178 | -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | /** 2 | * A small example express application 3 | * 4 | */ 5 | 6 | var express = require('express'); 7 | var path = require('path'); 8 | var title = require(path.join(__dirname, 'package.json')).name; 9 | 10 | // create an express server 11 | var app = express(); 12 | 13 | // view engine setup 14 | app.set('views', path.join(__dirname, 'views')); 15 | app.set('view engine', 'ejs'); 16 | 17 | // page router setup 18 | app.use(express.static(path.join(__dirname, 'public'))); 19 | app.get('/', function(req, res) { 20 | res.render('index', { title: title }); 21 | }); 22 | 23 | module.exports = app; 24 | -------------------------------------------------------------------------------- /bin/modbus-ws.js: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * Copyright (c) 2015, Yaacov Zamir 5 | * 6 | * Permission to use, copy, modify, and/or distribute this software for any 7 | * purpose with or without fee is hereby granted, provided that the above 8 | * copyright notice and this permission notice appear in all copies. 9 | * 10 | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 11 | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 12 | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 13 | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 14 | * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 15 | * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 16 | * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 17 | */ 18 | 19 | var program = require('commander'); 20 | 21 | var server = require('../server'); 22 | var version = require('../package.json').version; 23 | var numberRegex = /^[0-9]*$/; 24 | 25 | // get user cli arguments 26 | program 27 | .version(version) 28 | .option('-s, --serial ', 'Use serial port, set port. [false]', false) 29 | .option('-b, --baudrate ', 'Set serial port baudrate. [9600]', numberRegex, 9600) 30 | .option('-i, --ip ', 'Use tcp/ip, set slave url or ip address. [false]', false) 31 | .option('-P, --tcpport ', 'Server port number [3000]', numberRegex, 3000) 32 | .option('-c, --nocache', 'Do not use caching for modbus comunication. [false]', false) 33 | .option('-w, --nohttp', 'Run only websocket server, no httpd. [false]', false) 34 | .option('-m, --maxlength ', 'Max registers/coils to read in one modbus request. [10]', numberRegex, 10) 35 | 36 | .on('--help', function(){ 37 | console.log(' Examples:'); 38 | console.log(''); 39 | console.log(' modbus-ws --ip 192.168.1.24'); 40 | console.log(' create a bridge to a modbus slave using tcp/ip.'); 41 | console.log(' modbus-ws --serial /dev/ttyUSB0'); 42 | console.log(' create a bridge to a modbus slave using a serial port.'); 43 | console.log(' modbus-ws'); 44 | console.log(' when serial and tcp/ip are not used, default to test.'); 45 | console.log(' create a bridge to a modbus simulated slave.'); 46 | console.log(' modbus-ws --ip 192.168.1.24 --nocache'); 47 | console.log(' create a bridge with modbus without cache.'); 48 | console.log(' modbus-ws --ip 192.168.1.24 --nocache --nohttp'); 49 | console.log(' create a bridge with modbus without cache and without http web server.'); 50 | console.log(''); 51 | }) 52 | .parse(process.argv); 53 | 54 | // start the server 55 | server.start(program.opts()); 56 | -------------------------------------------------------------------------------- /cache.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) 2015, Yaacov Zamir 3 | * 4 | * Permission to use, copy, modify, and/or distribute this software for any 5 | * purpose with or without fee is hereby granted, provided that the above 6 | * copyright notice and this permission notice appear in all copies. 7 | * 8 | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 9 | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 10 | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 11 | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 12 | * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 13 | * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 14 | * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 15 | */ 16 | 17 | var sqlite3 = require('sqlite3').verbose(); 18 | var db; 19 | var _io; 20 | var _modbus; 21 | 22 | var VALID_ANS; 23 | var RESEND_WAIT; 24 | var FORGET_ASK; 25 | var MAX_LENGTH; 26 | var POLL_INTERVAL; 27 | 28 | const TYPE_INPUT_REG = 4; 29 | const TYPE_HOLDING_REG = 3; 30 | const TYPE_COIL = 1; 31 | const TYPE_DIGITAL_INPUT = 2; 32 | 33 | /** 34 | * Create the cache database. 35 | */ 36 | var createDb = function() { 37 | console.log(" Create cache db."); 38 | db = new sqlite3.Database(':memory:'); 39 | } 40 | 41 | /** 42 | * Create the cache table. 43 | */ 44 | var createTable = function() { 45 | /* unit - unit id 46 | * reg - register number 47 | * type - register type [0 input, 1 holding, 2 coil, 3 digital input] 48 | * val - register value 49 | * ask - first time browser send request 50 | * ans - last time device replied and answer sent to browser 51 | * snd - last time io event was triggered 52 | */ 53 | const CREATE_CACHE = "CREATE TABLE IF NOT EXISTS cache (\ 54 | unit NUMBER, \ 55 | type NUMBER, \ 56 | reg NUMBER, \ 57 | val NUMBER, \ 58 | ask NUMBER, \ 59 | ans NUMBER, \ 60 | snd NUMBER, \ 61 | PRIMARY KEY (unit, type, reg))"; 62 | 63 | console.log(" Create cache table"); 64 | 65 | db.run(CREATE_CACHE); 66 | } 67 | 68 | /** 69 | * Dump all cache rows to console 70 | */ 71 | var _debugReadAllRows = function() { 72 | const SELECT_ALL = "SELECT unit, type, reg, val, ask, ans, snd \ 73 | FROM cache ORDER BY unit, type, reg"; 74 | 75 | console.log("table: cache"); 76 | 77 | db.serialize(function() { 78 | db.all(SELECT_ALL, function(err, rows) { 79 | rows.forEach(function (row) { 80 | console.log(row.unit, row.type, row.reg, row.ask, row.ans, row.snd); 81 | }); 82 | }); 83 | }); 84 | } 85 | 86 | /** 87 | * Init cache table rows for registers. 88 | * 89 | * @param {number} unit the unit id. 90 | * @param {number} type the register type. 91 | * @param {number} register the first register to create. 92 | * @param {number} length the number of registers to create. 93 | */ 94 | var initRegisters = function(unit, type, register, length) { 95 | const INSERT_NEW = "INSERT OR IGNORE INTO cache VALUES (?, ?, ?, ?, ?, ?, ?)"; 96 | 97 | // make sure each register has a row 98 | var stmt = db.prepare(INSERT_NEW); 99 | for (var i = register; i < (register + length); i++) { 100 | stmt.run(unit, type, i, 0, 0, 0, 0); 101 | } 102 | stmt.finalize(); 103 | } 104 | 105 | /** 106 | * Poll modbus registers 107 | * 108 | * find the registers with the oldest ask time 109 | * and request data from device. 110 | */ 111 | var pollNextGroup = function() { 112 | const SELECT_NEXT_REG = "SELECT unit, type, reg FROM cache \ 113 | WHERE ask > ? \ 114 | ORDER BY ask DESC, reg ASC LIMIT 1"; 115 | const SELECT_LAST_REG = "SELECT reg FROM cache \ 116 | WHERE unit = ? AND type = ? AND reg < ? AND ask > ? \ 117 | ORDER BY reg DESC LIMIT 1"; 118 | 119 | var now = Date.now(); 120 | var minAskTime = now - FORGET_ASK; 121 | 122 | db.serialize(function() { 123 | // start sqlite serialize 124 | 125 | // find the register with oldest ask time 126 | db.get(SELECT_NEXT_REG, now - FORGET_ASK, function(err, row) { 127 | if (err) { 128 | console.log(err); 129 | } else if (row) { 130 | var unit = row.unit; 131 | var type = row.type; 132 | var firstReg = row.reg; 133 | var lastReg; 134 | 135 | //console.log('pollNextGroup first line:', row); 136 | 137 | db.get(SELECT_LAST_REG, unit, type, firstReg + MAX_LENGTH, minAskTime, 138 | function(err, row) { 139 | if (err) { 140 | console.log(err); 141 | } else if (row) { 142 | lastReg = row.reg; 143 | var length = lastReg - firstReg + 1; 144 | 145 | //console.log(' last line:', row); 146 | 147 | // ask from modbus and triger io data get event 148 | // and update cache value 149 | switch (type) { 150 | case TYPE_COIL: 151 | _getFC1(unit, firstReg, length); 152 | break; 153 | case TYPE_DIGITAL_INPUT: 154 | _getFC2(unit, firstReg, length); 155 | break; 156 | case TYPE_HOLDING_REG: 157 | _getFC3(unit, firstReg, length); 158 | break; 159 | case TYPE_INPUT_REG: 160 | _getFC4(unit, firstReg, length); 161 | break; 162 | } 163 | } 164 | } 165 | ); 166 | } 167 | }); 168 | 169 | }); // end sqlite serialize 170 | } 171 | 172 | /** 173 | * emit data get event to browser 174 | * 175 | * @param {number} unit the unit id. 176 | * @param {number} type the register type. 177 | * @param {number} address the first register to set. 178 | * @param {array} data the new values to set into registers 179 | */ 180 | var _emitDataGetEvent = function(unit, type, address, data) { 181 | const UPDATE_SND = "UPDATE cache SET snd = ?, ask = 0 \ 182 | WHERE unit = ? AND type = ? AND reg >= ? AND reg < ?"; 183 | 184 | var now = Date.now(); 185 | var length = data.length; 186 | 187 | // update cache 188 | db.run(UPDATE_SND, now, unit, type, address, address + length); 189 | // triger data-get event 190 | _io.emit('data', { 191 | 'unit': unit, 192 | 'type': type, 193 | 'address': address, 194 | 'data': data, 195 | 'flag': 'get' 196 | }); 197 | 198 | return; 199 | } 200 | 201 | /** 202 | * Get coils using cache, and flag to get new data from device 203 | * 204 | * @param {number} unit the unit id. 205 | * @param {number} address the first coil to get. 206 | * @param {number} length the number of coils to get. 207 | */ 208 | var getCoils = function(unit, address, length) { 209 | getRegisters(unit, TYPE_COIL, address, length) 210 | } 211 | 212 | /** 213 | * Get modbus coils 214 | * 215 | * @param {number} unit the unit id. 216 | * @param {number} address the first register to get. 217 | * @param {number} length the number of registers to get 218 | */ 219 | var _getFC1 = function(unit, address, length) { 220 | const UPDATE_REG = "UPDATE cache SET val= ?, ans = ?, ask = 0 \ 221 | WHERE unit = ? AND type = ? AND reg = ?"; 222 | 223 | var now = Date.now(); 224 | var type = TYPE_COIL; 225 | 226 | _modbus.writeFC1(unit, address, length, 227 | function(err, msg) { 228 | if (err) { 229 | _io.emit('error', {'err': err}); 230 | } else { 231 | // update data in cache, and clear ask flag 232 | var stmt = db.prepare(UPDATE_REG); 233 | for (i = 0; i < length; i++) { 234 | stmt.run(msg.data[i], now, unit, type, address + i); 235 | } 236 | stmt.finalize(); 237 | 238 | // emit data get event 239 | _emitDataGetEvent(unit, type, address, msg.data); 240 | } 241 | } 242 | ); 243 | } 244 | 245 | /** 246 | * Get input status using cache, and flag to get new data from device 247 | * 248 | * @param {number} unit the unit id. 249 | * @param {number} address the first input to get. 250 | * @param {number} length the number of input to get. 251 | */ 252 | var getInputStatus = function(unit, address, length) { 253 | getRegisters(unit, TYPE_DIGITAL_INPUT, address, length) 254 | } 255 | 256 | /** 257 | * Get modbus digital inputs 258 | * 259 | * @param {number} unit the unit id. 260 | * @param {number} address the first register to get. 261 | * @param {number} length the number of registers to get 262 | */ 263 | var _getFC2 = function(unit, address, length) { 264 | const UPDATE_REG = "UPDATE cache SET val= ?, ans = ?, ask = 0 \ 265 | WHERE unit = ? AND type = ? AND reg = ?"; 266 | 267 | var now = Date.now(); 268 | var type = TYPE_DIGITAL_INPUT; 269 | 270 | _modbus.writeFC2(unit, address, length, 271 | function(err, msg) { 272 | if (err) { 273 | _io.emit('error', {'err': err}); 274 | } else { 275 | // update data in cache, and clear ask flag 276 | var stmt = db.prepare(UPDATE_REG); 277 | for (i = 0; i < length; i++) { 278 | stmt.run(msg.data[i], now, unit, type, address + i); 279 | } 280 | stmt.finalize(); 281 | 282 | // emit data get event 283 | _emitDataGetEvent(unit, type, address, msg.data); 284 | } 285 | } 286 | ); 287 | } 288 | 289 | /** 290 | * Get holding registers using cache, and flag to get new data from device 291 | * 292 | * @param {number} unit the unit id. 293 | * @param {number} address the first register to get. 294 | * @param {number} length the number of registers to get. 295 | */ 296 | var getHoldingRegisters = function(unit, address, length) { 297 | getRegisters(unit, TYPE_HOLDING_REG, address, length) 298 | } 299 | 300 | /** 301 | * Get modbus holding registers 302 | * 303 | * @param {number} unit the unit id. 304 | * @param {number} address the first register to get. 305 | * @param {number} length the number of registers to get 306 | */ 307 | var _getFC3 = function(unit, address, length) { 308 | const UPDATE_REG = "UPDATE cache SET val= ?, ans = ?, ask = 0 \ 309 | WHERE unit = ? AND type = ? AND reg = ?"; 310 | 311 | var now = Date.now(); 312 | var type = TYPE_HOLDING_REG; 313 | 314 | _modbus.writeFC3(unit, address, length, 315 | function(err, msg) { 316 | if (err) { 317 | _io.emit('error', {'err': err}); 318 | } else { 319 | // update data in cache, and clear ask flag 320 | var stmt = db.prepare(UPDATE_REG); 321 | for (i = 0; i < length; i++) { 322 | stmt.run(msg.data[i], now, unit, type, address + i); 323 | } 324 | stmt.finalize(); 325 | 326 | // emit data get event 327 | _emitDataGetEvent(unit, type, address, msg.data); 328 | } 329 | } 330 | ); 331 | } 332 | 333 | /** 334 | * Get input registers using cache, and flag to get new data from device 335 | * 336 | * @param {number} unit the unit id. 337 | * @param {number} address the first register to get. 338 | * @param {number} length the number of registers to get. 339 | */ 340 | var getInputRegisters = function(unit, address, length) { 341 | getRegisters(unit, TYPE_INPUT_REG, address, length) 342 | } 343 | 344 | /** 345 | * Get modbus input registers 346 | * 347 | * @param {number} unit the unit id. 348 | * @param {number} address the first register to get. 349 | * @param {number} length the number of registers to get 350 | */ 351 | var _getFC4 = function(unit, address, length) { 352 | const UPDATE_REG = "UPDATE cache SET val= ?, ans = ?, ask = 0 \ 353 | WHERE unit = ? AND type = ? AND reg = ?"; 354 | 355 | var now = Date.now(); 356 | var type = TYPE_INPUT_REG; 357 | 358 | _modbus.writeFC4(unit, address, length, 359 | function(err, msg) { 360 | if (err) { 361 | _io.emit('error', {'err': err}); 362 | } else { 363 | // update data in cache, and clear ask flag 364 | var stmt = db.prepare(UPDATE_REG); 365 | for (i = 0; i < length; i++) { 366 | stmt.run(msg.data[i], now, unit, type, address + i); 367 | } 368 | stmt.finalize(); 369 | 370 | // emit data get event 371 | _emitDataGetEvent(unit, type, address, msg.data); 372 | } 373 | } 374 | ); 375 | } 376 | 377 | /** 378 | * Get registers using cache, and flag to get new data from device 379 | * 380 | * @param {number} unit the unit id. 381 | * @param {number} type the register type. 382 | * @param {number} address the first register to get. 383 | * @param {number} length the number of registers to get. 384 | */ 385 | var getRegisters = function(unit, type, address, length) { 386 | const UPDATE_ASK = "UPDATE cache SET ask = ? \ 387 | WHERE unit = ? AND type = ? AND reg >= ? AND reg < ? AND ask < ?"; 388 | const SELECT_GET = "SELECT unit, reg, val FROM cache \ 389 | WHERE unit = ? AND type = ? AND reg >= ? AND reg < ? \ 390 | AND ans > ? AND snd < ?"; 391 | 392 | initRegisters(unit, type, address, length); 393 | 394 | var now = Date.now(); 395 | var data = new Array(length); 396 | 397 | db.serialize(function() { 398 | // start sqlite serialize 399 | 400 | // set ask signal 401 | db.run(UPDATE_ASK, now, unit, type, address, address + length, 402 | now - FORGET_ASK); 403 | 404 | // check for data in cache 405 | db.all(SELECT_GET, unit, type, address, address + length, 406 | now - VALID_ANS, now - RESEND_WAIT, 407 | function(err, rows) { 408 | if (err) { 409 | _io.emit('error', {'err': err}); 410 | } else { 411 | // if we have valid data in cache 412 | if (rows.length == length) { 413 | // fill the data arry 414 | rows.forEach(function(row, i) {data[i] = row.val;}); 415 | 416 | // on discreet types we expect boolean data values 417 | if (type == TYPE_COIL || type == TYPE_DIGITAL_INPUT) { 418 | data = data.map(function(d) {return (d == 1);}); 419 | } 420 | 421 | // emit data get event 422 | _emitDataGetEvent(unit, type, address, data); 423 | } 424 | } 425 | } 426 | ); 427 | 428 | }); // end sqlite serialize 429 | } 430 | 431 | /** 432 | * force one coil 433 | * 434 | * @param {number} unit the slave unit address. 435 | * @param {number} address the Data Address of the coil. 436 | * @param {number} state the state to set into coil. 437 | */ 438 | var forceCoil = function(unit, address, state) { 439 | const UPDATE_REG = "UPDATE cache SET ans = 0, snd = 0, ask = 0 \ 440 | WHERE unit = ? AND type = ? AND reg >= ? AND reg < ?"; 441 | 442 | var length = 1; 443 | var type = TYPE_COIL; 444 | 445 | initRegisters(unit, type, address, length); 446 | 447 | _modbus.writeFC5(unit, address, state, 448 | function(err, msg) { 449 | if (err) { 450 | _io.emit('error', {'err': err}); 451 | } else { 452 | // invalidate the current value in cache 453 | db.run(UPDATE_REG, unit, type, address, address + length); 454 | 455 | // triger data-set event 456 | _io.emit('data', { 457 | 'unit': unit, 458 | 'type': type, 459 | 'address': address, 460 | 'data': state, 461 | 'flag': 'set' 462 | }); 463 | } 464 | } 465 | ); 466 | } 467 | 468 | /** 469 | * Set modbus registers 470 | * 471 | * @param {number} unit the unit id. 472 | * @param {number} address the first register to set. 473 | * @param {array} data the new values to set into registers 474 | */ 475 | var setRegisters = function(unit, address, data) { 476 | const UPDATE_REG = "UPDATE cache SET ans = 0, snd = 0, ask = 0 \ 477 | WHERE unit = ? AND type = ? AND reg >= ? AND reg < ?"; 478 | 479 | var length = data.length; 480 | var type = TYPE_HOLDING_REG; 481 | 482 | initRegisters(unit, type, address, length); 483 | 484 | _modbus.writeFC16(unit, address, data, 485 | function(err, msg) { 486 | if (err) { 487 | _io.emit('error', {'err': err}); 488 | } else { 489 | // invalidate the current value in cache 490 | db.run(UPDATE_REG, unit, type, address, address + length); 491 | 492 | // triger data-set event 493 | _io.emit('data', { 494 | 'unit': unit, 495 | 'type': type, 496 | 'address': address, 497 | 'data': data, 498 | 'flag': 'set' 499 | }); 500 | } 501 | } 502 | ); 503 | } 504 | 505 | /** 506 | * Close data base 507 | */ 508 | var closeDb = function() { 509 | console.log(" Close cache db"); 510 | db.close(); 511 | } 512 | 513 | /** 514 | * Init the cache and Run polling cycle 515 | * 516 | * @param {socket.io} io the socket io object to comunicate with browser. 517 | * @param {modbus} modbus the modbus object to comunicate with devices. 518 | * @param {object} set options options wile running. 519 | */ 520 | var run = function(io, modbus, options) { 521 | /* check for options 522 | */ 523 | if (!options) options = {}; 524 | 525 | // answers are valid for N-ms. 526 | VALID_ANS = options.validans || 5000; 527 | // do not trigger new io-get event for N-ms. 528 | RESEND_WAIT = options.resendwait || 100; 529 | // if not answered after N-ms, forget ask request. 530 | FORGET_ASK = options.forgetask || 10000; 531 | // max registers to ask in one modbus request. 532 | MAX_LENGTH = parseInt(options.maxlength) || 10; 533 | // wait N-ms between modbus polls. 534 | POLL_INTERVAL = options.pollinterval || 500; 535 | 536 | // print out cache options line 537 | console.log(' Cache options: ', VALID_ANS, RESEND_WAIT, 538 | FORGET_ASK, MAX_LENGTH, POLL_INTERVAL); 539 | 540 | createDb(); 541 | createTable(); 542 | 543 | _io = io; 544 | _modbus = modbus; 545 | 546 | var poll = function() { 547 | //_debugReadAllRows(); 548 | //console.log('polling'); 549 | pollNextGroup(); 550 | 551 | // wait a little and then do poll again 552 | setTimeout(poll, POLL_INTERVAL); 553 | } 554 | 555 | setTimeout(poll, 200); 556 | } 557 | 558 | module.exports = {}; 559 | module.exports.run = run; 560 | module.exports.getCoils = getCoils; 561 | module.exports.getInputStatus = getInputStatus; 562 | module.exports.forceCoil = forceCoil; 563 | module.exports.setRegisters = setRegisters; 564 | module.exports.getHoldingRegisters = getHoldingRegisters; 565 | module.exports.getInputRegisters = getInputRegisters; 566 | -------------------------------------------------------------------------------- /examples/example.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | test 5 | 34 | 35 | 36 |

Modbus-WS - Modbus to Websocket

37 |

Welcome to Modbus-WS data monitor page

38 | 39 |

Data received:

40 | 41 |
    42 |
43 | 44 | 45 | 93 | 94 | 95 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "modbus-ws", 3 | "version": "2.2.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@gar/promisify": { 8 | "version": "1.1.3", 9 | "resolved": "https://registry.npmjs.org/@gar/promisify/-/promisify-1.1.3.tgz", 10 | "integrity": "sha512-k2Ty1JcVojjJFwrg/ThKi2ujJ7XNLYaFGNB/bWT9wGR+oSMJHMa5w+CUq6p/pVrKeNNgA7pCqEcjSnHVoqJQFw==", 11 | "optional": true 12 | }, 13 | "@npmcli/fs": { 14 | "version": "1.1.1", 15 | "resolved": "https://registry.npmjs.org/@npmcli/fs/-/fs-1.1.1.tgz", 16 | "integrity": "sha512-8KG5RD0GVP4ydEzRn/I4BNDuxDtqVbOdm8675T49OIG/NGhaK0pjPX7ZcDlvKYbA+ulvVK3ztfcF4uBdOxuJbQ==", 17 | "optional": true, 18 | "requires": { 19 | "@gar/promisify": "^1.0.1", 20 | "semver": "^7.3.5" 21 | }, 22 | "dependencies": { 23 | "semver": { 24 | "version": "7.6.0", 25 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", 26 | "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", 27 | "optional": true, 28 | "requires": { 29 | "lru-cache": "^6.0.0" 30 | } 31 | } 32 | } 33 | }, 34 | "@npmcli/move-file": { 35 | "version": "1.1.2", 36 | "resolved": "https://registry.npmjs.org/@npmcli/move-file/-/move-file-1.1.2.tgz", 37 | "integrity": "sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg==", 38 | "optional": true, 39 | "requires": { 40 | "mkdirp": "^1.0.4", 41 | "rimraf": "^3.0.2" 42 | } 43 | }, 44 | "@serialport/binding-abstract": { 45 | "version": "9.0.2", 46 | "resolved": "https://registry.npmjs.org/@serialport/binding-abstract/-/binding-abstract-9.0.2.tgz", 47 | "integrity": "sha512-kyMX6usn+VLpidt0YsDq5JwztIan9TPCX6skr0XcalOxI8I7w+/2qVZJzjgo2fSqDnPRcU2jMWTytwzEXFODvQ==", 48 | "requires": { 49 | "debug": "^4.1.1" 50 | }, 51 | "dependencies": { 52 | "debug": { 53 | "version": "4.3.1", 54 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", 55 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", 56 | "requires": { 57 | "ms": "2.1.2" 58 | } 59 | }, 60 | "ms": { 61 | "version": "2.1.2", 62 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 63 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 64 | } 65 | } 66 | }, 67 | "@serialport/binding-mock": { 68 | "version": "9.0.2", 69 | "resolved": "https://registry.npmjs.org/@serialport/binding-mock/-/binding-mock-9.0.2.tgz", 70 | "integrity": "sha512-HfrvJ/LXULHk8w63CGxwDNiDidFgDX8BnadY+cgVS6yHMHikbhLCLjCmUKsKBWaGKRqOznl0w+iUl7TMi1lkXQ==", 71 | "requires": { 72 | "@serialport/binding-abstract": "^9.0.2", 73 | "debug": "^4.1.1" 74 | }, 75 | "dependencies": { 76 | "debug": { 77 | "version": "4.3.1", 78 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", 79 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", 80 | "requires": { 81 | "ms": "2.1.2" 82 | } 83 | }, 84 | "ms": { 85 | "version": "2.1.2", 86 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 87 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 88 | } 89 | } 90 | }, 91 | "@serialport/bindings": { 92 | "version": "9.0.4", 93 | "resolved": "https://registry.npmjs.org/@serialport/bindings/-/bindings-9.0.4.tgz", 94 | "integrity": "sha512-6dlE1vm5c1xk667f1Zm7D+msbHJ9jdnUr9l8DResKpj2iCBzbCNsW+yCYq26WxzXWc1L2HUaS3/aL+k0wm5amg==", 95 | "requires": { 96 | "@serialport/binding-abstract": "^9.0.2", 97 | "@serialport/parser-readline": "^9.0.1", 98 | "bindings": "^1.5.0", 99 | "debug": "^4.3.1", 100 | "nan": "^2.14.2", 101 | "prebuild-install": "^6.0.0" 102 | }, 103 | "dependencies": { 104 | "debug": { 105 | "version": "4.3.1", 106 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", 107 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", 108 | "requires": { 109 | "ms": "2.1.2" 110 | } 111 | }, 112 | "ms": { 113 | "version": "2.1.2", 114 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 115 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 116 | } 117 | } 118 | }, 119 | "@serialport/parser-byte-length": { 120 | "version": "9.0.1", 121 | "resolved": "https://registry.npmjs.org/@serialport/parser-byte-length/-/parser-byte-length-9.0.1.tgz", 122 | "integrity": "sha512-1Ikv4lgCNw8OMf35yCpgzjHwkpgBEkhBuXFXIdWZk+ixaHFLlAtp03QxGPZBmzHMK58WDmEQoBHC1V5BkkAKSQ==" 123 | }, 124 | "@serialport/parser-cctalk": { 125 | "version": "9.0.1", 126 | "resolved": "https://registry.npmjs.org/@serialport/parser-cctalk/-/parser-cctalk-9.0.1.tgz", 127 | "integrity": "sha512-GtMda2DeJ+23bNqOc79JYV06dax2n3FLLFM3zA7nfReCOi98QbuDj4TUbFESMOnp4DB0oMO0GYHCR9gHOedTkg==" 128 | }, 129 | "@serialport/parser-delimiter": { 130 | "version": "9.0.1", 131 | "resolved": "https://registry.npmjs.org/@serialport/parser-delimiter/-/parser-delimiter-9.0.1.tgz", 132 | "integrity": "sha512-+oaSl5zEu47OlrRiF5p5tn2qgGqYuhVcE+NI+Pv4E1xsNB/A0fFxxMv/8XUw466CRLEJ5IESIB9qbFvKE6ltaQ==" 133 | }, 134 | "@serialport/parser-inter-byte-timeout": { 135 | "version": "9.0.1", 136 | "resolved": "https://registry.npmjs.org/@serialport/parser-inter-byte-timeout/-/parser-inter-byte-timeout-9.0.1.tgz", 137 | "integrity": "sha512-lFflcUflcP5SF4vLIixAKs1xUI/wfOzCv1Xq78VbPOBlIjZ6ny9lQ6g7cMPR/sB/M1BHwGcdX7CEr90pe3kkog==" 138 | }, 139 | "@serialport/parser-readline": { 140 | "version": "9.0.1", 141 | "resolved": "https://registry.npmjs.org/@serialport/parser-readline/-/parser-readline-9.0.1.tgz", 142 | "integrity": "sha512-38058gxvyfgdeLpg3aUyD98NuWkVB9yyTLpcSdeQ3GYiupivwH6Tdy/SKPmxlHIw3Ml2qil5MR2mtW2fLPB5CQ==", 143 | "requires": { 144 | "@serialport/parser-delimiter": "^9.0.1" 145 | } 146 | }, 147 | "@serialport/parser-ready": { 148 | "version": "9.0.1", 149 | "resolved": "https://registry.npmjs.org/@serialport/parser-ready/-/parser-ready-9.0.1.tgz", 150 | "integrity": "sha512-lgzGkVJaaV1rJVx26WwI2UKyPxc0vu1rsOeldzA3VVbF+ABrblUQA06+cRPpT6k96GY+X4+1fB1rWuPpt8HbgQ==" 151 | }, 152 | "@serialport/parser-regex": { 153 | "version": "9.0.1", 154 | "resolved": "https://registry.npmjs.org/@serialport/parser-regex/-/parser-regex-9.0.1.tgz", 155 | "integrity": "sha512-BHTV+Lkl+J8hSecFtDRENaR4fgA6tw44J+dmA1vEKEyum0iDN4bihbu8yvztYyo4PhBGUKDfm/PnD5EkJm0dPA==" 156 | }, 157 | "@serialport/stream": { 158 | "version": "9.0.2", 159 | "resolved": "https://registry.npmjs.org/@serialport/stream/-/stream-9.0.2.tgz", 160 | "integrity": "sha512-0RkVe+gvwZu/PPfbb7ExQ+euGoCTGKD/B8TQ5fuhe+eKk1sh73RwjKmu9gp6veSNqx9Zljnh1dF6mhdEKWZpSA==", 161 | "requires": { 162 | "debug": "^4.1.1" 163 | }, 164 | "dependencies": { 165 | "debug": { 166 | "version": "4.3.1", 167 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", 168 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", 169 | "requires": { 170 | "ms": "2.1.2" 171 | } 172 | }, 173 | "ms": { 174 | "version": "2.1.2", 175 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 176 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 177 | } 178 | } 179 | }, 180 | "@socket.io/component-emitter": { 181 | "version": "3.1.0", 182 | "resolved": "https://registry.npmjs.org/@socket.io/component-emitter/-/component-emitter-3.1.0.tgz", 183 | "integrity": "sha512-+9jVqKhRSpsc591z5vX+X5Yyw+he/HCB4iQ/RYxw35CEPaY1gnsNE43nf9n9AaYjAQrTiI/mOwKUKdUs9vf7Xg==" 184 | }, 185 | "@tootallnate/once": { 186 | "version": "1.1.2", 187 | "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", 188 | "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", 189 | "optional": true 190 | }, 191 | "@types/cookie": { 192 | "version": "0.4.1", 193 | "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.1.tgz", 194 | "integrity": "sha512-XW/Aa8APYr6jSVVA1y/DEIZX0/GMKLEVekNG727R8cs56ahETkRAy/3DR7+fJyh7oUgGwNQaRfXCun0+KbWY7Q==" 195 | }, 196 | "@types/cors": { 197 | "version": "2.8.13", 198 | "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.13.tgz", 199 | "integrity": "sha512-RG8AStHlUiV5ysZQKq97copd2UmVYw3/pRMLefISZ3S1hK104Cwm7iLQ3fTKx+lsUH2CE8FlLaYeEA2LSeqYUA==", 200 | "requires": { 201 | "@types/node": "*" 202 | } 203 | }, 204 | "@types/node": { 205 | "version": "18.16.3", 206 | "resolved": "https://registry.npmjs.org/@types/node/-/node-18.16.3.tgz", 207 | "integrity": "sha512-OPs5WnnT1xkCBiuQrZA4+YAV4HEJejmHneyraIaxsbev5yCEr6KMwINNFP9wQeFIw8FWcoTqF3vQsa5CDaI+8Q==" 208 | }, 209 | "abbrev": { 210 | "version": "1.1.1", 211 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 212 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 213 | "optional": true 214 | }, 215 | "accepts": { 216 | "version": "1.3.8", 217 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 218 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 219 | "requires": { 220 | "mime-types": "~2.1.34", 221 | "negotiator": "0.6.3" 222 | }, 223 | "dependencies": { 224 | "mime-db": { 225 | "version": "1.52.0", 226 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 227 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" 228 | }, 229 | "mime-types": { 230 | "version": "2.1.35", 231 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 232 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 233 | "requires": { 234 | "mime-db": "1.52.0" 235 | } 236 | } 237 | } 238 | }, 239 | "agent-base": { 240 | "version": "6.0.2", 241 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", 242 | "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", 243 | "optional": true, 244 | "requires": { 245 | "debug": "4" 246 | }, 247 | "dependencies": { 248 | "debug": { 249 | "version": "4.3.4", 250 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 251 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 252 | "optional": true, 253 | "requires": { 254 | "ms": "2.1.2" 255 | } 256 | }, 257 | "ms": { 258 | "version": "2.1.2", 259 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 260 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 261 | "optional": true 262 | } 263 | } 264 | }, 265 | "agentkeepalive": { 266 | "version": "4.5.0", 267 | "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-4.5.0.tgz", 268 | "integrity": "sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew==", 269 | "optional": true, 270 | "requires": { 271 | "humanize-ms": "^1.2.1" 272 | } 273 | }, 274 | "aggregate-error": { 275 | "version": "3.1.0", 276 | "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", 277 | "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", 278 | "optional": true, 279 | "requires": { 280 | "clean-stack": "^2.0.0", 281 | "indent-string": "^4.0.0" 282 | } 283 | }, 284 | "ansi-colors": { 285 | "version": "4.1.1", 286 | "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", 287 | "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", 288 | "dev": true 289 | }, 290 | "ansi-regex": { 291 | "version": "2.1.1", 292 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", 293 | "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" 294 | }, 295 | "ansi-styles": { 296 | "version": "4.3.0", 297 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 298 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 299 | "requires": { 300 | "color-convert": "^2.0.1" 301 | } 302 | }, 303 | "anymatch": { 304 | "version": "3.1.2", 305 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", 306 | "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", 307 | "dev": true, 308 | "requires": { 309 | "normalize-path": "^3.0.0", 310 | "picomatch": "^2.0.4" 311 | } 312 | }, 313 | "aproba": { 314 | "version": "1.2.0", 315 | "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", 316 | "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==" 317 | }, 318 | "are-we-there-yet": { 319 | "version": "1.1.5", 320 | "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz", 321 | "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", 322 | "requires": { 323 | "delegates": "^1.0.0", 324 | "readable-stream": "^2.0.6" 325 | } 326 | }, 327 | "argparse": { 328 | "version": "2.0.1", 329 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", 330 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", 331 | "dev": true 332 | }, 333 | "array-flatten": { 334 | "version": "1.1.1", 335 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 336 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" 337 | }, 338 | "assertion-error": { 339 | "version": "1.1.0", 340 | "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz", 341 | "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==", 342 | "dev": true 343 | }, 344 | "async": { 345 | "version": "3.2.4", 346 | "resolved": "https://registry.npmjs.org/async/-/async-3.2.4.tgz", 347 | "integrity": "sha512-iAB+JbDEGXhyIUavoDl9WP/Jj106Kz9DEn1DPgYw5ruDn0e3Wgi3sKFm55sASdGBNOQB8F59d9qQ7deqrHA8wQ==" 348 | }, 349 | "balanced-match": { 350 | "version": "1.0.0", 351 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 352 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" 353 | }, 354 | "base64-js": { 355 | "version": "1.5.1", 356 | "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", 357 | "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" 358 | }, 359 | "base64id": { 360 | "version": "2.0.0", 361 | "resolved": "https://registry.npmjs.org/base64id/-/base64id-2.0.0.tgz", 362 | "integrity": "sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog==" 363 | }, 364 | "binary-extensions": { 365 | "version": "2.2.0", 366 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", 367 | "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", 368 | "dev": true 369 | }, 370 | "bindings": { 371 | "version": "1.5.0", 372 | "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", 373 | "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", 374 | "requires": { 375 | "file-uri-to-path": "1.0.0" 376 | } 377 | }, 378 | "bl": { 379 | "version": "4.0.3", 380 | "resolved": "https://registry.npmjs.org/bl/-/bl-4.0.3.tgz", 381 | "integrity": "sha512-fs4G6/Hu4/EE+F75J8DuN/0IpQqNjAdC7aEQv7Qt8MHGUH7Ckv2MwTEEeN9QehD0pfIDkMI1bkHYkKy7xHyKIg==", 382 | "requires": { 383 | "buffer": "^5.5.0", 384 | "inherits": "^2.0.4", 385 | "readable-stream": "^3.4.0" 386 | }, 387 | "dependencies": { 388 | "inherits": { 389 | "version": "2.0.4", 390 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 391 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 392 | }, 393 | "readable-stream": { 394 | "version": "3.6.0", 395 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", 396 | "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", 397 | "requires": { 398 | "inherits": "^2.0.3", 399 | "string_decoder": "^1.1.1", 400 | "util-deprecate": "^1.0.1" 401 | } 402 | } 403 | } 404 | }, 405 | "body-parser": { 406 | "version": "1.20.2", 407 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.2.tgz", 408 | "integrity": "sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==", 409 | "requires": { 410 | "bytes": "3.1.2", 411 | "content-type": "~1.0.5", 412 | "debug": "2.6.9", 413 | "depd": "2.0.0", 414 | "destroy": "1.2.0", 415 | "http-errors": "2.0.0", 416 | "iconv-lite": "0.4.24", 417 | "on-finished": "2.4.1", 418 | "qs": "6.11.0", 419 | "raw-body": "2.5.2", 420 | "type-is": "~1.6.18", 421 | "unpipe": "1.0.0" 422 | } 423 | }, 424 | "brace-expansion": { 425 | "version": "1.1.11", 426 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 427 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 428 | "requires": { 429 | "balanced-match": "^1.0.0", 430 | "concat-map": "0.0.1" 431 | } 432 | }, 433 | "braces": { 434 | "version": "3.0.2", 435 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", 436 | "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", 437 | "dev": true, 438 | "requires": { 439 | "fill-range": "^7.0.1" 440 | } 441 | }, 442 | "browser-stdout": { 443 | "version": "1.3.1", 444 | "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", 445 | "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", 446 | "dev": true 447 | }, 448 | "buffer": { 449 | "version": "5.7.1", 450 | "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", 451 | "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", 452 | "requires": { 453 | "base64-js": "^1.3.1", 454 | "ieee754": "^1.1.13" 455 | } 456 | }, 457 | "bytes": { 458 | "version": "3.1.2", 459 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 460 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==" 461 | }, 462 | "cacache": { 463 | "version": "15.3.0", 464 | "resolved": "https://registry.npmjs.org/cacache/-/cacache-15.3.0.tgz", 465 | "integrity": "sha512-VVdYzXEn+cnbXpFgWs5hTT7OScegHVmLhJIR8Ufqk3iFD6A6j5iSX1KuBTfNEv4tdJWE2PzA6IVFtcLC7fN9wQ==", 466 | "optional": true, 467 | "requires": { 468 | "@npmcli/fs": "^1.0.0", 469 | "@npmcli/move-file": "^1.0.1", 470 | "chownr": "^2.0.0", 471 | "fs-minipass": "^2.0.0", 472 | "glob": "^7.1.4", 473 | "infer-owner": "^1.0.4", 474 | "lru-cache": "^6.0.0", 475 | "minipass": "^3.1.1", 476 | "minipass-collect": "^1.0.2", 477 | "minipass-flush": "^1.0.5", 478 | "minipass-pipeline": "^1.2.2", 479 | "mkdirp": "^1.0.3", 480 | "p-map": "^4.0.0", 481 | "promise-inflight": "^1.0.1", 482 | "rimraf": "^3.0.2", 483 | "ssri": "^8.0.1", 484 | "tar": "^6.0.2", 485 | "unique-filename": "^1.1.1" 486 | }, 487 | "dependencies": { 488 | "chownr": { 489 | "version": "2.0.0", 490 | "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", 491 | "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", 492 | "optional": true 493 | } 494 | } 495 | }, 496 | "call-bind": { 497 | "version": "1.0.7", 498 | "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", 499 | "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", 500 | "requires": { 501 | "es-define-property": "^1.0.0", 502 | "es-errors": "^1.3.0", 503 | "function-bind": "^1.1.2", 504 | "get-intrinsic": "^1.2.4", 505 | "set-function-length": "^1.2.1" 506 | } 507 | }, 508 | "camelcase": { 509 | "version": "6.3.0", 510 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", 511 | "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", 512 | "dev": true 513 | }, 514 | "chai": { 515 | "version": "4.2.0", 516 | "resolved": "https://registry.npmjs.org/chai/-/chai-4.2.0.tgz", 517 | "integrity": "sha512-XQU3bhBukrOsQCuwZndwGcCVQHyZi53fQ6Ys1Fym7E4olpIqqZZhhoFJoaKVvV17lWQoXYwgWN2nF5crA8J2jw==", 518 | "dev": true, 519 | "requires": { 520 | "assertion-error": "^1.1.0", 521 | "check-error": "^1.0.2", 522 | "deep-eql": "^3.0.1", 523 | "get-func-name": "^2.0.0", 524 | "pathval": "^1.1.0", 525 | "type-detect": "^4.0.5" 526 | } 527 | }, 528 | "chalk": { 529 | "version": "4.1.2", 530 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 531 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 532 | "requires": { 533 | "ansi-styles": "^4.1.0", 534 | "supports-color": "^7.1.0" 535 | } 536 | }, 537 | "check-error": { 538 | "version": "1.0.2", 539 | "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz", 540 | "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=", 541 | "dev": true 542 | }, 543 | "chokidar": { 544 | "version": "3.5.3", 545 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", 546 | "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", 547 | "dev": true, 548 | "requires": { 549 | "anymatch": "~3.1.2", 550 | "braces": "~3.0.2", 551 | "fsevents": "~2.3.2", 552 | "glob-parent": "~5.1.2", 553 | "is-binary-path": "~2.1.0", 554 | "is-glob": "~4.0.1", 555 | "normalize-path": "~3.0.0", 556 | "readdirp": "~3.6.0" 557 | } 558 | }, 559 | "chownr": { 560 | "version": "1.1.4", 561 | "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", 562 | "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==" 563 | }, 564 | "clean-stack": { 565 | "version": "2.2.0", 566 | "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", 567 | "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", 568 | "optional": true 569 | }, 570 | "cliui": { 571 | "version": "7.0.4", 572 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", 573 | "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", 574 | "dev": true, 575 | "requires": { 576 | "string-width": "^4.2.0", 577 | "strip-ansi": "^6.0.0", 578 | "wrap-ansi": "^7.0.0" 579 | }, 580 | "dependencies": { 581 | "ansi-regex": { 582 | "version": "5.0.1", 583 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 584 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 585 | "dev": true 586 | }, 587 | "is-fullwidth-code-point": { 588 | "version": "3.0.0", 589 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 590 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 591 | "dev": true 592 | }, 593 | "string-width": { 594 | "version": "4.2.3", 595 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 596 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 597 | "dev": true, 598 | "requires": { 599 | "emoji-regex": "^8.0.0", 600 | "is-fullwidth-code-point": "^3.0.0", 601 | "strip-ansi": "^6.0.1" 602 | } 603 | }, 604 | "strip-ansi": { 605 | "version": "6.0.1", 606 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 607 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 608 | "dev": true, 609 | "requires": { 610 | "ansi-regex": "^5.0.1" 611 | } 612 | } 613 | } 614 | }, 615 | "code-point-at": { 616 | "version": "1.1.0", 617 | "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", 618 | "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=" 619 | }, 620 | "color-convert": { 621 | "version": "2.0.1", 622 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 623 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 624 | "requires": { 625 | "color-name": "~1.1.4" 626 | } 627 | }, 628 | "color-name": { 629 | "version": "1.1.4", 630 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 631 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" 632 | }, 633 | "color-support": { 634 | "version": "1.1.3", 635 | "resolved": "https://registry.npmjs.org/color-support/-/color-support-1.1.3.tgz", 636 | "integrity": "sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==", 637 | "optional": true 638 | }, 639 | "commander": { 640 | "version": "7.0.0", 641 | "resolved": "https://registry.npmjs.org/commander/-/commander-7.0.0.tgz", 642 | "integrity": "sha512-ovx/7NkTrnPuIV8sqk/GjUIIM1+iUQeqA3ye2VNpq9sVoiZsooObWlQy+OPWGI17GDaEoybuAGJm6U8yC077BA==" 643 | }, 644 | "concat-map": { 645 | "version": "0.0.1", 646 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 647 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" 648 | }, 649 | "console-control-strings": { 650 | "version": "1.1.0", 651 | "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", 652 | "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=" 653 | }, 654 | "content-disposition": { 655 | "version": "0.5.4", 656 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 657 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 658 | "requires": { 659 | "safe-buffer": "5.2.1" 660 | }, 661 | "dependencies": { 662 | "safe-buffer": { 663 | "version": "5.2.1", 664 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 665 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" 666 | } 667 | } 668 | }, 669 | "content-type": { 670 | "version": "1.0.5", 671 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", 672 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==" 673 | }, 674 | "cookie": { 675 | "version": "0.4.2", 676 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.2.tgz", 677 | "integrity": "sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==" 678 | }, 679 | "cookie-signature": { 680 | "version": "1.0.6", 681 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 682 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" 683 | }, 684 | "core-util-is": { 685 | "version": "1.0.2", 686 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 687 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 688 | }, 689 | "cors": { 690 | "version": "2.8.5", 691 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 692 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 693 | "requires": { 694 | "object-assign": "^4", 695 | "vary": "^1" 696 | } 697 | }, 698 | "debug": { 699 | "version": "2.6.9", 700 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 701 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 702 | "requires": { 703 | "ms": "2.0.0" 704 | } 705 | }, 706 | "decamelize": { 707 | "version": "4.0.0", 708 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz", 709 | "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==", 710 | "dev": true 711 | }, 712 | "decompress-response": { 713 | "version": "4.2.1", 714 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-4.2.1.tgz", 715 | "integrity": "sha512-jOSne2qbyE+/r8G1VU+G/82LBs2Fs4LAsTiLSHOCOMZQl2OKZ6i8i4IyHemTe+/yIXOtTcRQMzPcgyhoFlqPkw==", 716 | "requires": { 717 | "mimic-response": "^2.0.0" 718 | } 719 | }, 720 | "deep-eql": { 721 | "version": "3.0.1", 722 | "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz", 723 | "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==", 724 | "dev": true, 725 | "requires": { 726 | "type-detect": "^4.0.0" 727 | } 728 | }, 729 | "deep-extend": { 730 | "version": "0.6.0", 731 | "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", 732 | "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==" 733 | }, 734 | "define-data-property": { 735 | "version": "1.1.4", 736 | "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", 737 | "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", 738 | "requires": { 739 | "es-define-property": "^1.0.0", 740 | "es-errors": "^1.3.0", 741 | "gopd": "^1.0.1" 742 | } 743 | }, 744 | "delegates": { 745 | "version": "1.0.0", 746 | "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", 747 | "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=" 748 | }, 749 | "depd": { 750 | "version": "2.0.0", 751 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 752 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 753 | }, 754 | "destroy": { 755 | "version": "1.2.0", 756 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 757 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==" 758 | }, 759 | "detect-libc": { 760 | "version": "1.0.3", 761 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz", 762 | "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups=" 763 | }, 764 | "diff": { 765 | "version": "5.0.0", 766 | "resolved": "https://registry.npmjs.org/diff/-/diff-5.0.0.tgz", 767 | "integrity": "sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==", 768 | "dev": true 769 | }, 770 | "ee-first": { 771 | "version": "1.1.1", 772 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 773 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" 774 | }, 775 | "ejs": { 776 | "version": "3.1.7", 777 | "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.7.tgz", 778 | "integrity": "sha512-BIar7R6abbUxDA3bfXrO4DSgwo8I+fB5/1zgujl3HLLjwd6+9iOnrT+t3grn2qbk9vOgBubXOFwX2m9axoFaGw==", 779 | "requires": { 780 | "jake": "^10.8.5" 781 | } 782 | }, 783 | "emoji-regex": { 784 | "version": "8.0.0", 785 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 786 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" 787 | }, 788 | "encodeurl": { 789 | "version": "1.0.2", 790 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 791 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==" 792 | }, 793 | "encoding": { 794 | "version": "0.1.13", 795 | "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.13.tgz", 796 | "integrity": "sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==", 797 | "optional": true, 798 | "requires": { 799 | "iconv-lite": "^0.6.2" 800 | }, 801 | "dependencies": { 802 | "iconv-lite": { 803 | "version": "0.6.3", 804 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", 805 | "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", 806 | "optional": true, 807 | "requires": { 808 | "safer-buffer": ">= 2.1.2 < 3.0.0" 809 | } 810 | } 811 | } 812 | }, 813 | "end-of-stream": { 814 | "version": "1.4.4", 815 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 816 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 817 | "requires": { 818 | "once": "^1.4.0" 819 | } 820 | }, 821 | "engine.io": { 822 | "version": "6.4.2", 823 | "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-6.4.2.tgz", 824 | "integrity": "sha512-FKn/3oMiJjrOEOeUub2WCox6JhxBXq/Zn3fZOMCBxKnNYtsdKjxhl7yR3fZhM9PV+rdE75SU5SYMc+2PGzo+Tg==", 825 | "requires": { 826 | "@types/cookie": "^0.4.1", 827 | "@types/cors": "^2.8.12", 828 | "@types/node": ">=10.0.0", 829 | "accepts": "~1.3.4", 830 | "base64id": "2.0.0", 831 | "cookie": "~0.4.1", 832 | "cors": "~2.8.5", 833 | "debug": "~4.3.1", 834 | "engine.io-parser": "~5.0.3", 835 | "ws": "~8.11.0" 836 | }, 837 | "dependencies": { 838 | "debug": { 839 | "version": "4.3.4", 840 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 841 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 842 | "requires": { 843 | "ms": "2.1.2" 844 | } 845 | }, 846 | "engine.io-parser": { 847 | "version": "5.0.6", 848 | "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-5.0.6.tgz", 849 | "integrity": "sha512-tjuoZDMAdEhVnSFleYPCtdL2GXwVTGtNjoeJd9IhIG3C1xs9uwxqRNEu5WpnDZCaozwVlK/nuQhpodhXSIMaxw==" 850 | }, 851 | "ms": { 852 | "version": "2.1.2", 853 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 854 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 855 | }, 856 | "ws": { 857 | "version": "8.11.0", 858 | "resolved": "https://registry.npmjs.org/ws/-/ws-8.11.0.tgz", 859 | "integrity": "sha512-HPG3wQd9sNQoT9xHyNCXoDUa+Xw/VevmY9FoHyQ+g+rrMn4j6FB4np7Z0OhdTgjx6MgQLK7jwSy1YecU1+4Asg==" 860 | } 861 | } 862 | }, 863 | "engine.io-client": { 864 | "version": "6.4.0", 865 | "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-6.4.0.tgz", 866 | "integrity": "sha512-GyKPDyoEha+XZ7iEqam49vz6auPnNJ9ZBfy89f+rMMas8AuiMWOZ9PVzu8xb9ZC6rafUqiGHSCfu22ih66E+1g==", 867 | "dev": true, 868 | "requires": { 869 | "@socket.io/component-emitter": "~3.1.0", 870 | "debug": "~4.3.1", 871 | "engine.io-parser": "~5.0.3", 872 | "ws": "~8.11.0", 873 | "xmlhttprequest-ssl": "~2.0.0" 874 | }, 875 | "dependencies": { 876 | "debug": { 877 | "version": "4.3.4", 878 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 879 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 880 | "dev": true, 881 | "requires": { 882 | "ms": "2.1.2" 883 | } 884 | }, 885 | "ms": { 886 | "version": "2.1.2", 887 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 888 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 889 | "dev": true 890 | } 891 | } 892 | }, 893 | "engine.io-parser": { 894 | "version": "5.0.6", 895 | "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-5.0.6.tgz", 896 | "integrity": "sha512-tjuoZDMAdEhVnSFleYPCtdL2GXwVTGtNjoeJd9IhIG3C1xs9uwxqRNEu5WpnDZCaozwVlK/nuQhpodhXSIMaxw==", 897 | "dev": true 898 | }, 899 | "env-paths": { 900 | "version": "2.2.1", 901 | "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", 902 | "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", 903 | "optional": true 904 | }, 905 | "err-code": { 906 | "version": "2.0.3", 907 | "resolved": "https://registry.npmjs.org/err-code/-/err-code-2.0.3.tgz", 908 | "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", 909 | "optional": true 910 | }, 911 | "es-define-property": { 912 | "version": "1.0.0", 913 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", 914 | "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", 915 | "requires": { 916 | "get-intrinsic": "^1.2.4" 917 | } 918 | }, 919 | "es-errors": { 920 | "version": "1.3.0", 921 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", 922 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==" 923 | }, 924 | "escalade": { 925 | "version": "3.1.1", 926 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 927 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 928 | "dev": true 929 | }, 930 | "escape-html": { 931 | "version": "1.0.3", 932 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 933 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" 934 | }, 935 | "etag": { 936 | "version": "1.8.1", 937 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 938 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==" 939 | }, 940 | "expand-template": { 941 | "version": "2.0.3", 942 | "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", 943 | "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==" 944 | }, 945 | "express": { 946 | "version": "4.19.2", 947 | "resolved": "https://registry.npmjs.org/express/-/express-4.19.2.tgz", 948 | "integrity": "sha512-5T6nhjsT+EOMzuck8JjBHARTHfMht0POzlA60WV2pMD3gyXw2LZnZ+ueGdNxG+0calOJcWKbpFcuzLZ91YWq9Q==", 949 | "requires": { 950 | "accepts": "~1.3.8", 951 | "array-flatten": "1.1.1", 952 | "body-parser": "1.20.2", 953 | "content-disposition": "0.5.4", 954 | "content-type": "~1.0.4", 955 | "cookie": "0.6.0", 956 | "cookie-signature": "1.0.6", 957 | "debug": "2.6.9", 958 | "depd": "2.0.0", 959 | "encodeurl": "~1.0.2", 960 | "escape-html": "~1.0.3", 961 | "etag": "~1.8.1", 962 | "finalhandler": "1.2.0", 963 | "fresh": "0.5.2", 964 | "http-errors": "2.0.0", 965 | "merge-descriptors": "1.0.1", 966 | "methods": "~1.1.2", 967 | "on-finished": "2.4.1", 968 | "parseurl": "~1.3.3", 969 | "path-to-regexp": "0.1.7", 970 | "proxy-addr": "~2.0.7", 971 | "qs": "6.11.0", 972 | "range-parser": "~1.2.1", 973 | "safe-buffer": "5.2.1", 974 | "send": "0.18.0", 975 | "serve-static": "1.15.0", 976 | "setprototypeof": "1.2.0", 977 | "statuses": "2.0.1", 978 | "type-is": "~1.6.18", 979 | "utils-merge": "1.0.1", 980 | "vary": "~1.1.2" 981 | }, 982 | "dependencies": { 983 | "cookie": { 984 | "version": "0.6.0", 985 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.6.0.tgz", 986 | "integrity": "sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==" 987 | }, 988 | "safe-buffer": { 989 | "version": "5.2.1", 990 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 991 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" 992 | } 993 | } 994 | }, 995 | "file-uri-to-path": { 996 | "version": "1.0.0", 997 | "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", 998 | "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==" 999 | }, 1000 | "filelist": { 1001 | "version": "1.0.4", 1002 | "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.4.tgz", 1003 | "integrity": "sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==", 1004 | "requires": { 1005 | "minimatch": "^5.0.1" 1006 | }, 1007 | "dependencies": { 1008 | "brace-expansion": { 1009 | "version": "2.0.1", 1010 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", 1011 | "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", 1012 | "requires": { 1013 | "balanced-match": "^1.0.0" 1014 | } 1015 | }, 1016 | "minimatch": { 1017 | "version": "5.1.0", 1018 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.0.tgz", 1019 | "integrity": "sha512-9TPBGGak4nHfGZsPBohm9AWg6NoT7QTCehS3BIJABslyZbzxfV78QM2Y6+i741OPZIafFAaiiEMh5OyIrJPgtg==", 1020 | "requires": { 1021 | "brace-expansion": "^2.0.1" 1022 | } 1023 | } 1024 | } 1025 | }, 1026 | "fill-range": { 1027 | "version": "7.0.1", 1028 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", 1029 | "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", 1030 | "dev": true, 1031 | "requires": { 1032 | "to-regex-range": "^5.0.1" 1033 | } 1034 | }, 1035 | "finalhandler": { 1036 | "version": "1.2.0", 1037 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz", 1038 | "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", 1039 | "requires": { 1040 | "debug": "2.6.9", 1041 | "encodeurl": "~1.0.2", 1042 | "escape-html": "~1.0.3", 1043 | "on-finished": "2.4.1", 1044 | "parseurl": "~1.3.3", 1045 | "statuses": "2.0.1", 1046 | "unpipe": "~1.0.0" 1047 | } 1048 | }, 1049 | "find-up": { 1050 | "version": "5.0.0", 1051 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 1052 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 1053 | "dev": true, 1054 | "requires": { 1055 | "locate-path": "^6.0.0", 1056 | "path-exists": "^4.0.0" 1057 | } 1058 | }, 1059 | "flat": { 1060 | "version": "5.0.2", 1061 | "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", 1062 | "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", 1063 | "dev": true 1064 | }, 1065 | "forwarded": { 1066 | "version": "0.2.0", 1067 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 1068 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" 1069 | }, 1070 | "fresh": { 1071 | "version": "0.5.2", 1072 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 1073 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==" 1074 | }, 1075 | "fs-constants": { 1076 | "version": "1.0.0", 1077 | "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", 1078 | "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==" 1079 | }, 1080 | "fs-minipass": { 1081 | "version": "2.1.0", 1082 | "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", 1083 | "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", 1084 | "requires": { 1085 | "minipass": "^3.0.0" 1086 | } 1087 | }, 1088 | "fs.realpath": { 1089 | "version": "1.0.0", 1090 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 1091 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" 1092 | }, 1093 | "fsevents": { 1094 | "version": "2.3.2", 1095 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 1096 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 1097 | "dev": true, 1098 | "optional": true 1099 | }, 1100 | "function-bind": { 1101 | "version": "1.1.2", 1102 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 1103 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==" 1104 | }, 1105 | "gauge": { 1106 | "version": "2.7.4", 1107 | "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz", 1108 | "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", 1109 | "requires": { 1110 | "aproba": "^1.0.3", 1111 | "console-control-strings": "^1.0.0", 1112 | "has-unicode": "^2.0.0", 1113 | "object-assign": "^4.1.0", 1114 | "signal-exit": "^3.0.0", 1115 | "string-width": "^1.0.1", 1116 | "strip-ansi": "^3.0.1", 1117 | "wide-align": "^1.1.0" 1118 | } 1119 | }, 1120 | "get-caller-file": { 1121 | "version": "2.0.5", 1122 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 1123 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 1124 | "dev": true 1125 | }, 1126 | "get-func-name": { 1127 | "version": "2.0.0", 1128 | "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz", 1129 | "integrity": "sha1-6td0q+5y4gQJQzoGY2YCPdaIekE=", 1130 | "dev": true 1131 | }, 1132 | "get-intrinsic": { 1133 | "version": "1.2.4", 1134 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", 1135 | "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", 1136 | "requires": { 1137 | "es-errors": "^1.3.0", 1138 | "function-bind": "^1.1.2", 1139 | "has-proto": "^1.0.1", 1140 | "has-symbols": "^1.0.3", 1141 | "hasown": "^2.0.0" 1142 | } 1143 | }, 1144 | "github-from-package": { 1145 | "version": "0.0.0", 1146 | "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", 1147 | "integrity": "sha1-l/tdlr/eiXMxPyDoKI75oWf6ZM4=" 1148 | }, 1149 | "glob": { 1150 | "version": "7.2.3", 1151 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 1152 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 1153 | "optional": true, 1154 | "requires": { 1155 | "fs.realpath": "^1.0.0", 1156 | "inflight": "^1.0.4", 1157 | "inherits": "2", 1158 | "minimatch": "^3.1.1", 1159 | "once": "^1.3.0", 1160 | "path-is-absolute": "^1.0.0" 1161 | } 1162 | }, 1163 | "glob-parent": { 1164 | "version": "5.1.2", 1165 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 1166 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 1167 | "dev": true, 1168 | "requires": { 1169 | "is-glob": "^4.0.1" 1170 | } 1171 | }, 1172 | "gopd": { 1173 | "version": "1.0.1", 1174 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", 1175 | "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", 1176 | "requires": { 1177 | "get-intrinsic": "^1.1.3" 1178 | } 1179 | }, 1180 | "graceful-fs": { 1181 | "version": "4.2.11", 1182 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", 1183 | "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", 1184 | "optional": true 1185 | }, 1186 | "has-property-descriptors": { 1187 | "version": "1.0.2", 1188 | "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", 1189 | "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", 1190 | "requires": { 1191 | "es-define-property": "^1.0.0" 1192 | } 1193 | }, 1194 | "has-proto": { 1195 | "version": "1.0.3", 1196 | "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", 1197 | "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==" 1198 | }, 1199 | "has-symbols": { 1200 | "version": "1.0.3", 1201 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", 1202 | "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==" 1203 | }, 1204 | "has-unicode": { 1205 | "version": "2.0.1", 1206 | "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", 1207 | "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=" 1208 | }, 1209 | "hasown": { 1210 | "version": "2.0.2", 1211 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 1212 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 1213 | "requires": { 1214 | "function-bind": "^1.1.2" 1215 | } 1216 | }, 1217 | "he": { 1218 | "version": "1.2.0", 1219 | "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", 1220 | "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", 1221 | "dev": true 1222 | }, 1223 | "http-cache-semantics": { 1224 | "version": "4.1.1", 1225 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz", 1226 | "integrity": "sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==", 1227 | "optional": true 1228 | }, 1229 | "http-errors": { 1230 | "version": "2.0.0", 1231 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 1232 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 1233 | "requires": { 1234 | "depd": "2.0.0", 1235 | "inherits": "2.0.4", 1236 | "setprototypeof": "1.2.0", 1237 | "statuses": "2.0.1", 1238 | "toidentifier": "1.0.1" 1239 | }, 1240 | "dependencies": { 1241 | "inherits": { 1242 | "version": "2.0.4", 1243 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1244 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 1245 | } 1246 | } 1247 | }, 1248 | "http-proxy-agent": { 1249 | "version": "4.0.1", 1250 | "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", 1251 | "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", 1252 | "optional": true, 1253 | "requires": { 1254 | "@tootallnate/once": "1", 1255 | "agent-base": "6", 1256 | "debug": "4" 1257 | }, 1258 | "dependencies": { 1259 | "debug": { 1260 | "version": "4.3.4", 1261 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1262 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1263 | "optional": true, 1264 | "requires": { 1265 | "ms": "2.1.2" 1266 | } 1267 | }, 1268 | "ms": { 1269 | "version": "2.1.2", 1270 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1271 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1272 | "optional": true 1273 | } 1274 | } 1275 | }, 1276 | "https-proxy-agent": { 1277 | "version": "5.0.1", 1278 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", 1279 | "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", 1280 | "optional": true, 1281 | "requires": { 1282 | "agent-base": "6", 1283 | "debug": "4" 1284 | }, 1285 | "dependencies": { 1286 | "debug": { 1287 | "version": "4.3.4", 1288 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1289 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1290 | "optional": true, 1291 | "requires": { 1292 | "ms": "2.1.2" 1293 | } 1294 | }, 1295 | "ms": { 1296 | "version": "2.1.2", 1297 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1298 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1299 | "optional": true 1300 | } 1301 | } 1302 | }, 1303 | "humanize-ms": { 1304 | "version": "1.2.1", 1305 | "resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz", 1306 | "integrity": "sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==", 1307 | "optional": true, 1308 | "requires": { 1309 | "ms": "^2.0.0" 1310 | } 1311 | }, 1312 | "iconv-lite": { 1313 | "version": "0.4.24", 1314 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 1315 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 1316 | "requires": { 1317 | "safer-buffer": ">= 2.1.2 < 3" 1318 | } 1319 | }, 1320 | "ieee754": { 1321 | "version": "1.2.1", 1322 | "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", 1323 | "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" 1324 | }, 1325 | "imurmurhash": { 1326 | "version": "0.1.4", 1327 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 1328 | "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", 1329 | "optional": true 1330 | }, 1331 | "indent-string": { 1332 | "version": "4.0.0", 1333 | "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", 1334 | "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", 1335 | "optional": true 1336 | }, 1337 | "infer-owner": { 1338 | "version": "1.0.4", 1339 | "resolved": "https://registry.npmjs.org/infer-owner/-/infer-owner-1.0.4.tgz", 1340 | "integrity": "sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==", 1341 | "optional": true 1342 | }, 1343 | "inflight": { 1344 | "version": "1.0.6", 1345 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 1346 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 1347 | "requires": { 1348 | "once": "^1.3.0", 1349 | "wrappy": "1" 1350 | } 1351 | }, 1352 | "inherits": { 1353 | "version": "2.0.3", 1354 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 1355 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 1356 | }, 1357 | "ini": { 1358 | "version": "1.3.8", 1359 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", 1360 | "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==" 1361 | }, 1362 | "ip-address": { 1363 | "version": "9.0.5", 1364 | "resolved": "https://registry.npmjs.org/ip-address/-/ip-address-9.0.5.tgz", 1365 | "integrity": "sha512-zHtQzGojZXTwZTHQqra+ETKd4Sn3vgi7uBmlPoXVWZqYvuKmtI0l/VZTjqGmJY9x88GGOaZ9+G9ES8hC4T4X8g==", 1366 | "optional": true, 1367 | "requires": { 1368 | "jsbn": "1.1.0", 1369 | "sprintf-js": "^1.1.3" 1370 | } 1371 | }, 1372 | "ipaddr.js": { 1373 | "version": "1.9.1", 1374 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 1375 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 1376 | }, 1377 | "is-binary-path": { 1378 | "version": "2.1.0", 1379 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 1380 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 1381 | "dev": true, 1382 | "requires": { 1383 | "binary-extensions": "^2.0.0" 1384 | } 1385 | }, 1386 | "is-extglob": { 1387 | "version": "2.1.1", 1388 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1389 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 1390 | "dev": true 1391 | }, 1392 | "is-fullwidth-code-point": { 1393 | "version": "1.0.0", 1394 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", 1395 | "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", 1396 | "requires": { 1397 | "number-is-nan": "^1.0.0" 1398 | } 1399 | }, 1400 | "is-glob": { 1401 | "version": "4.0.3", 1402 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 1403 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 1404 | "dev": true, 1405 | "requires": { 1406 | "is-extglob": "^2.1.1" 1407 | } 1408 | }, 1409 | "is-lambda": { 1410 | "version": "1.0.1", 1411 | "resolved": "https://registry.npmjs.org/is-lambda/-/is-lambda-1.0.1.tgz", 1412 | "integrity": "sha512-z7CMFGNrENq5iFB9Bqo64Xk6Y9sg+epq1myIcdHaGnbMTYOxvzsEtdYqQUylB7LxfkvgrrjP32T6Ywciio9UIQ==", 1413 | "optional": true 1414 | }, 1415 | "is-number": { 1416 | "version": "7.0.0", 1417 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1418 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 1419 | "dev": true 1420 | }, 1421 | "is-plain-obj": { 1422 | "version": "2.1.0", 1423 | "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz", 1424 | "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==", 1425 | "dev": true 1426 | }, 1427 | "is-unicode-supported": { 1428 | "version": "0.1.0", 1429 | "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", 1430 | "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", 1431 | "dev": true 1432 | }, 1433 | "isarray": { 1434 | "version": "1.0.0", 1435 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1436 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 1437 | }, 1438 | "isexe": { 1439 | "version": "2.0.0", 1440 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 1441 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", 1442 | "optional": true 1443 | }, 1444 | "jake": { 1445 | "version": "10.8.5", 1446 | "resolved": "https://registry.npmjs.org/jake/-/jake-10.8.5.tgz", 1447 | "integrity": "sha512-sVpxYeuAhWt0OTWITwT98oyV0GsXyMlXCF+3L1SuafBVUIr/uILGRB+NqwkzhgXKvoJpDIpQvqkUALgdmQsQxw==", 1448 | "requires": { 1449 | "async": "^3.2.3", 1450 | "chalk": "^4.0.2", 1451 | "filelist": "^1.0.1", 1452 | "minimatch": "^3.0.4" 1453 | } 1454 | }, 1455 | "js-yaml": { 1456 | "version": "4.1.0", 1457 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", 1458 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", 1459 | "dev": true, 1460 | "requires": { 1461 | "argparse": "^2.0.1" 1462 | } 1463 | }, 1464 | "jsbn": { 1465 | "version": "1.1.0", 1466 | "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-1.1.0.tgz", 1467 | "integrity": "sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A==", 1468 | "optional": true 1469 | }, 1470 | "locate-path": { 1471 | "version": "6.0.0", 1472 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 1473 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 1474 | "dev": true, 1475 | "requires": { 1476 | "p-locate": "^5.0.0" 1477 | } 1478 | }, 1479 | "log-symbols": { 1480 | "version": "4.1.0", 1481 | "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", 1482 | "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", 1483 | "dev": true, 1484 | "requires": { 1485 | "chalk": "^4.1.0", 1486 | "is-unicode-supported": "^0.1.0" 1487 | }, 1488 | "dependencies": { 1489 | "ansi-styles": { 1490 | "version": "4.3.0", 1491 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1492 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1493 | "dev": true, 1494 | "requires": { 1495 | "color-convert": "^2.0.1" 1496 | } 1497 | }, 1498 | "chalk": { 1499 | "version": "4.1.2", 1500 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 1501 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 1502 | "dev": true, 1503 | "requires": { 1504 | "ansi-styles": "^4.1.0", 1505 | "supports-color": "^7.1.0" 1506 | } 1507 | }, 1508 | "color-convert": { 1509 | "version": "2.0.1", 1510 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 1511 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 1512 | "dev": true, 1513 | "requires": { 1514 | "color-name": "~1.1.4" 1515 | } 1516 | }, 1517 | "color-name": { 1518 | "version": "1.1.4", 1519 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1520 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1521 | "dev": true 1522 | }, 1523 | "has-flag": { 1524 | "version": "4.0.0", 1525 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1526 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1527 | "dev": true 1528 | }, 1529 | "supports-color": { 1530 | "version": "7.2.0", 1531 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 1532 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 1533 | "dev": true, 1534 | "requires": { 1535 | "has-flag": "^4.0.0" 1536 | } 1537 | } 1538 | } 1539 | }, 1540 | "lru-cache": { 1541 | "version": "6.0.0", 1542 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 1543 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 1544 | "requires": { 1545 | "yallist": "^4.0.0" 1546 | } 1547 | }, 1548 | "make-fetch-happen": { 1549 | "version": "9.1.0", 1550 | "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-9.1.0.tgz", 1551 | "integrity": "sha512-+zopwDy7DNknmwPQplem5lAZX/eCOzSvSNNcSKm5eVwTkOBzoktEfXsa9L23J/GIRhxRsaxzkPEhrJEpE2F4Gg==", 1552 | "optional": true, 1553 | "requires": { 1554 | "agentkeepalive": "^4.1.3", 1555 | "cacache": "^15.2.0", 1556 | "http-cache-semantics": "^4.1.0", 1557 | "http-proxy-agent": "^4.0.1", 1558 | "https-proxy-agent": "^5.0.0", 1559 | "is-lambda": "^1.0.1", 1560 | "lru-cache": "^6.0.0", 1561 | "minipass": "^3.1.3", 1562 | "minipass-collect": "^1.0.2", 1563 | "minipass-fetch": "^1.3.2", 1564 | "minipass-flush": "^1.0.5", 1565 | "minipass-pipeline": "^1.2.4", 1566 | "negotiator": "^0.6.2", 1567 | "promise-retry": "^2.0.1", 1568 | "socks-proxy-agent": "^6.0.0", 1569 | "ssri": "^8.0.0" 1570 | } 1571 | }, 1572 | "media-typer": { 1573 | "version": "0.3.0", 1574 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1575 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==" 1576 | }, 1577 | "merge-descriptors": { 1578 | "version": "1.0.1", 1579 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 1580 | "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==" 1581 | }, 1582 | "methods": { 1583 | "version": "1.1.2", 1584 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1585 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==" 1586 | }, 1587 | "mime": { 1588 | "version": "1.6.0", 1589 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1590 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 1591 | }, 1592 | "mime-db": { 1593 | "version": "1.52.0", 1594 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 1595 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" 1596 | }, 1597 | "mime-types": { 1598 | "version": "2.1.35", 1599 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 1600 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 1601 | "requires": { 1602 | "mime-db": "1.52.0" 1603 | } 1604 | }, 1605 | "mimic-response": { 1606 | "version": "2.1.0", 1607 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-2.1.0.tgz", 1608 | "integrity": "sha512-wXqjST+SLt7R009ySCglWBCFpjUygmCIfD790/kVbiGmUgfYGuB14PiTd5DwVxSV4NcYHjzMkoj5LjQZwTQLEA==" 1609 | }, 1610 | "minimatch": { 1611 | "version": "3.1.2", 1612 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1613 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1614 | "requires": { 1615 | "brace-expansion": "^1.1.7" 1616 | } 1617 | }, 1618 | "minimist": { 1619 | "version": "1.2.7", 1620 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.7.tgz", 1621 | "integrity": "sha512-bzfL1YUZsP41gmu/qjrEk0Q6i2ix/cVeAhbCbqH9u3zYutS1cLg00qhrD0M2MVdCcx4Sc0UpP2eBWo9rotpq6g==" 1622 | }, 1623 | "minipass": { 1624 | "version": "3.3.6", 1625 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.3.6.tgz", 1626 | "integrity": "sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==", 1627 | "requires": { 1628 | "yallist": "^4.0.0" 1629 | } 1630 | }, 1631 | "minipass-collect": { 1632 | "version": "1.0.2", 1633 | "resolved": "https://registry.npmjs.org/minipass-collect/-/minipass-collect-1.0.2.tgz", 1634 | "integrity": "sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA==", 1635 | "optional": true, 1636 | "requires": { 1637 | "minipass": "^3.0.0" 1638 | } 1639 | }, 1640 | "minipass-fetch": { 1641 | "version": "1.4.1", 1642 | "resolved": "https://registry.npmjs.org/minipass-fetch/-/minipass-fetch-1.4.1.tgz", 1643 | "integrity": "sha512-CGH1eblLq26Y15+Azk7ey4xh0J/XfJfrCox5LDJiKqI2Q2iwOLOKrlmIaODiSQS8d18jalF6y2K2ePUm0CmShw==", 1644 | "optional": true, 1645 | "requires": { 1646 | "encoding": "^0.1.12", 1647 | "minipass": "^3.1.0", 1648 | "minipass-sized": "^1.0.3", 1649 | "minizlib": "^2.0.0" 1650 | } 1651 | }, 1652 | "minipass-flush": { 1653 | "version": "1.0.5", 1654 | "resolved": "https://registry.npmjs.org/minipass-flush/-/minipass-flush-1.0.5.tgz", 1655 | "integrity": "sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==", 1656 | "optional": true, 1657 | "requires": { 1658 | "minipass": "^3.0.0" 1659 | } 1660 | }, 1661 | "minipass-pipeline": { 1662 | "version": "1.2.4", 1663 | "resolved": "https://registry.npmjs.org/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz", 1664 | "integrity": "sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A==", 1665 | "optional": true, 1666 | "requires": { 1667 | "minipass": "^3.0.0" 1668 | } 1669 | }, 1670 | "minipass-sized": { 1671 | "version": "1.0.3", 1672 | "resolved": "https://registry.npmjs.org/minipass-sized/-/minipass-sized-1.0.3.tgz", 1673 | "integrity": "sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g==", 1674 | "optional": true, 1675 | "requires": { 1676 | "minipass": "^3.0.0" 1677 | } 1678 | }, 1679 | "minizlib": { 1680 | "version": "2.1.2", 1681 | "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", 1682 | "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", 1683 | "requires": { 1684 | "minipass": "^3.0.0", 1685 | "yallist": "^4.0.0" 1686 | } 1687 | }, 1688 | "mkdirp": { 1689 | "version": "1.0.4", 1690 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", 1691 | "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==" 1692 | }, 1693 | "mkdirp-classic": { 1694 | "version": "0.5.3", 1695 | "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", 1696 | "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==" 1697 | }, 1698 | "mocha": { 1699 | "version": "10.1.0", 1700 | "resolved": "https://registry.npmjs.org/mocha/-/mocha-10.1.0.tgz", 1701 | "integrity": "sha512-vUF7IYxEoN7XhQpFLxQAEMtE4W91acW4B6En9l97MwE9stL1A9gusXfoHZCLVHDUJ/7V5+lbCM6yMqzo5vNymg==", 1702 | "dev": true, 1703 | "requires": { 1704 | "ansi-colors": "4.1.1", 1705 | "browser-stdout": "1.3.1", 1706 | "chokidar": "3.5.3", 1707 | "debug": "4.3.4", 1708 | "diff": "5.0.0", 1709 | "escape-string-regexp": "4.0.0", 1710 | "find-up": "5.0.0", 1711 | "glob": "7.2.0", 1712 | "he": "1.2.0", 1713 | "js-yaml": "4.1.0", 1714 | "log-symbols": "4.1.0", 1715 | "minimatch": "5.0.1", 1716 | "ms": "2.1.3", 1717 | "nanoid": "3.3.3", 1718 | "serialize-javascript": "6.0.0", 1719 | "strip-json-comments": "3.1.1", 1720 | "supports-color": "8.1.1", 1721 | "workerpool": "6.2.1", 1722 | "yargs": "16.2.0", 1723 | "yargs-parser": "20.2.4", 1724 | "yargs-unparser": "2.0.0" 1725 | }, 1726 | "dependencies": { 1727 | "debug": { 1728 | "version": "4.3.4", 1729 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 1730 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 1731 | "dev": true, 1732 | "requires": { 1733 | "ms": "2.1.2" 1734 | }, 1735 | "dependencies": { 1736 | "ms": { 1737 | "version": "2.1.2", 1738 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1739 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 1740 | "dev": true 1741 | } 1742 | } 1743 | }, 1744 | "escape-string-regexp": { 1745 | "version": "4.0.0", 1746 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 1747 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 1748 | "dev": true 1749 | }, 1750 | "glob": { 1751 | "version": "7.2.0", 1752 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", 1753 | "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", 1754 | "dev": true, 1755 | "requires": { 1756 | "fs.realpath": "^1.0.0", 1757 | "inflight": "^1.0.4", 1758 | "inherits": "2", 1759 | "minimatch": "^3.0.4", 1760 | "once": "^1.3.0", 1761 | "path-is-absolute": "^1.0.0" 1762 | }, 1763 | "dependencies": { 1764 | "minimatch": { 1765 | "version": "3.1.2", 1766 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1767 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1768 | "dev": true, 1769 | "requires": { 1770 | "brace-expansion": "^1.1.7" 1771 | } 1772 | } 1773 | } 1774 | }, 1775 | "has-flag": { 1776 | "version": "4.0.0", 1777 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 1778 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 1779 | "dev": true 1780 | }, 1781 | "minimatch": { 1782 | "version": "5.0.1", 1783 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.0.1.tgz", 1784 | "integrity": "sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==", 1785 | "dev": true, 1786 | "requires": { 1787 | "brace-expansion": "^2.0.1" 1788 | }, 1789 | "dependencies": { 1790 | "brace-expansion": { 1791 | "version": "2.0.1", 1792 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", 1793 | "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", 1794 | "dev": true, 1795 | "requires": { 1796 | "balanced-match": "^1.0.0" 1797 | } 1798 | } 1799 | } 1800 | }, 1801 | "ms": { 1802 | "version": "2.1.3", 1803 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1804 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1805 | "dev": true 1806 | }, 1807 | "strip-json-comments": { 1808 | "version": "3.1.1", 1809 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 1810 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 1811 | "dev": true 1812 | }, 1813 | "supports-color": { 1814 | "version": "8.1.1", 1815 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", 1816 | "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", 1817 | "dev": true, 1818 | "requires": { 1819 | "has-flag": "^4.0.0" 1820 | } 1821 | } 1822 | } 1823 | }, 1824 | "modbus-serial": { 1825 | "version": "8.0.1", 1826 | "resolved": "https://registry.npmjs.org/modbus-serial/-/modbus-serial-8.0.1.tgz", 1827 | "integrity": "sha512-WcN5nuQOMxpMcywCa9tABzktXK60t/zrZLHW7lWfqdXCljBMkQ4QWxuljm5+ayDaBeDgh/0iOIhzfDQ0O7i2KA==", 1828 | "requires": { 1829 | "debug": "^4.1.1", 1830 | "serialport": "^9.0.0" 1831 | }, 1832 | "dependencies": { 1833 | "debug": { 1834 | "version": "4.3.1", 1835 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", 1836 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", 1837 | "requires": { 1838 | "ms": "2.1.2" 1839 | } 1840 | }, 1841 | "ms": { 1842 | "version": "2.1.2", 1843 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 1844 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 1845 | } 1846 | } 1847 | }, 1848 | "ms": { 1849 | "version": "2.0.0", 1850 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1851 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" 1852 | }, 1853 | "nan": { 1854 | "version": "2.14.2", 1855 | "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz", 1856 | "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==" 1857 | }, 1858 | "nanoid": { 1859 | "version": "3.3.3", 1860 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.3.tgz", 1861 | "integrity": "sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w==", 1862 | "dev": true 1863 | }, 1864 | "napi-build-utils": { 1865 | "version": "1.0.2", 1866 | "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-1.0.2.tgz", 1867 | "integrity": "sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==" 1868 | }, 1869 | "negotiator": { 1870 | "version": "0.6.3", 1871 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 1872 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==" 1873 | }, 1874 | "node-abi": { 1875 | "version": "2.19.3", 1876 | "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-2.19.3.tgz", 1877 | "integrity": "sha512-9xZrlyfvKhWme2EXFKQhZRp1yNWT/uI1luYPr3sFl+H4keYY4xR+1jO7mvTTijIsHf1M+QDe9uWuKeEpLInIlg==", 1878 | "requires": { 1879 | "semver": "^5.4.1" 1880 | } 1881 | }, 1882 | "node-addon-api": { 1883 | "version": "7.1.0", 1884 | "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-7.1.0.tgz", 1885 | "integrity": "sha512-mNcltoe1R8o7STTegSOHdnJNN7s5EUvhoS7ShnTHDyOSd+8H+UdWODq6qSv67PjC8Zc5JRT8+oLAMCr0SIXw7g==" 1886 | }, 1887 | "node-gyp": { 1888 | "version": "8.4.1", 1889 | "resolved": "https://registry.npmjs.org/node-gyp/-/node-gyp-8.4.1.tgz", 1890 | "integrity": "sha512-olTJRgUtAb/hOXG0E93wZDs5YiJlgbXxTwQAFHyNlRsXQnYzUaF2aGgujZbw+hR8aF4ZG/rST57bWMWD16jr9w==", 1891 | "optional": true, 1892 | "requires": { 1893 | "env-paths": "^2.2.0", 1894 | "glob": "^7.1.4", 1895 | "graceful-fs": "^4.2.6", 1896 | "make-fetch-happen": "^9.1.0", 1897 | "nopt": "^5.0.0", 1898 | "npmlog": "^6.0.0", 1899 | "rimraf": "^3.0.2", 1900 | "semver": "^7.3.5", 1901 | "tar": "^6.1.2", 1902 | "which": "^2.0.2" 1903 | }, 1904 | "dependencies": { 1905 | "ansi-regex": { 1906 | "version": "5.0.1", 1907 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 1908 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 1909 | "optional": true 1910 | }, 1911 | "are-we-there-yet": { 1912 | "version": "3.0.1", 1913 | "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-3.0.1.tgz", 1914 | "integrity": "sha512-QZW4EDmGwlYur0Yyf/b2uGucHQMa8aFUP7eu9ddR73vvhFyt4V0Vl3QHPcTNJ8l6qYOBdxgXdnBXQrHilfRQBg==", 1915 | "optional": true, 1916 | "requires": { 1917 | "delegates": "^1.0.0", 1918 | "readable-stream": "^3.6.0" 1919 | } 1920 | }, 1921 | "gauge": { 1922 | "version": "4.0.4", 1923 | "resolved": "https://registry.npmjs.org/gauge/-/gauge-4.0.4.tgz", 1924 | "integrity": "sha512-f9m+BEN5jkg6a0fZjleidjN51VE1X+mPFQ2DJ0uv1V39oCLCbsGe6yjbBnp7eK7z/+GAon99a3nHuqbuuthyPg==", 1925 | "optional": true, 1926 | "requires": { 1927 | "aproba": "^1.0.3 || ^2.0.0", 1928 | "color-support": "^1.1.3", 1929 | "console-control-strings": "^1.1.0", 1930 | "has-unicode": "^2.0.1", 1931 | "signal-exit": "^3.0.7", 1932 | "string-width": "^4.2.3", 1933 | "strip-ansi": "^6.0.1", 1934 | "wide-align": "^1.1.5" 1935 | } 1936 | }, 1937 | "is-fullwidth-code-point": { 1938 | "version": "3.0.0", 1939 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1940 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1941 | "optional": true 1942 | }, 1943 | "npmlog": { 1944 | "version": "6.0.2", 1945 | "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-6.0.2.tgz", 1946 | "integrity": "sha512-/vBvz5Jfr9dT/aFWd0FIRf+T/Q2WBsLENygUaFUqstqsycmZAP/t5BvFJTK0viFmSUxiUKTUplWy5vt+rvKIxg==", 1947 | "optional": true, 1948 | "requires": { 1949 | "are-we-there-yet": "^3.0.0", 1950 | "console-control-strings": "^1.1.0", 1951 | "gauge": "^4.0.3", 1952 | "set-blocking": "^2.0.0" 1953 | } 1954 | }, 1955 | "readable-stream": { 1956 | "version": "3.6.2", 1957 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", 1958 | "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", 1959 | "optional": true, 1960 | "requires": { 1961 | "inherits": "^2.0.3", 1962 | "string_decoder": "^1.1.1", 1963 | "util-deprecate": "^1.0.1" 1964 | } 1965 | }, 1966 | "semver": { 1967 | "version": "7.6.0", 1968 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", 1969 | "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", 1970 | "optional": true, 1971 | "requires": { 1972 | "lru-cache": "^6.0.0" 1973 | } 1974 | }, 1975 | "signal-exit": { 1976 | "version": "3.0.7", 1977 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", 1978 | "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", 1979 | "optional": true 1980 | }, 1981 | "string-width": { 1982 | "version": "4.2.3", 1983 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 1984 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 1985 | "optional": true, 1986 | "requires": { 1987 | "emoji-regex": "^8.0.0", 1988 | "is-fullwidth-code-point": "^3.0.0", 1989 | "strip-ansi": "^6.0.1" 1990 | } 1991 | }, 1992 | "strip-ansi": { 1993 | "version": "6.0.1", 1994 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 1995 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 1996 | "optional": true, 1997 | "requires": { 1998 | "ansi-regex": "^5.0.1" 1999 | } 2000 | }, 2001 | "wide-align": { 2002 | "version": "1.1.5", 2003 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.5.tgz", 2004 | "integrity": "sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg==", 2005 | "optional": true, 2006 | "requires": { 2007 | "string-width": "^1.0.2 || 2 || 3 || 4" 2008 | } 2009 | } 2010 | } 2011 | }, 2012 | "noop-logger": { 2013 | "version": "0.1.1", 2014 | "resolved": "https://registry.npmjs.org/noop-logger/-/noop-logger-0.1.1.tgz", 2015 | "integrity": "sha1-lKKxYzxPExdVMAfYlm/Q6EG2pMI=" 2016 | }, 2017 | "nopt": { 2018 | "version": "5.0.0", 2019 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-5.0.0.tgz", 2020 | "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", 2021 | "optional": true, 2022 | "requires": { 2023 | "abbrev": "1" 2024 | } 2025 | }, 2026 | "normalize-path": { 2027 | "version": "3.0.0", 2028 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 2029 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 2030 | "dev": true 2031 | }, 2032 | "npmlog": { 2033 | "version": "4.1.2", 2034 | "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz", 2035 | "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", 2036 | "requires": { 2037 | "are-we-there-yet": "~1.1.2", 2038 | "console-control-strings": "~1.1.0", 2039 | "gauge": "~2.7.3", 2040 | "set-blocking": "~2.0.0" 2041 | } 2042 | }, 2043 | "number-is-nan": { 2044 | "version": "1.0.1", 2045 | "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", 2046 | "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" 2047 | }, 2048 | "object-assign": { 2049 | "version": "4.1.1", 2050 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 2051 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 2052 | }, 2053 | "object-inspect": { 2054 | "version": "1.13.1", 2055 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.1.tgz", 2056 | "integrity": "sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==" 2057 | }, 2058 | "on-finished": { 2059 | "version": "2.4.1", 2060 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 2061 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 2062 | "requires": { 2063 | "ee-first": "1.1.1" 2064 | } 2065 | }, 2066 | "once": { 2067 | "version": "1.4.0", 2068 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 2069 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 2070 | "requires": { 2071 | "wrappy": "1" 2072 | } 2073 | }, 2074 | "p-limit": { 2075 | "version": "3.1.0", 2076 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 2077 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 2078 | "dev": true, 2079 | "requires": { 2080 | "yocto-queue": "^0.1.0" 2081 | } 2082 | }, 2083 | "p-locate": { 2084 | "version": "5.0.0", 2085 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 2086 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 2087 | "dev": true, 2088 | "requires": { 2089 | "p-limit": "^3.0.2" 2090 | } 2091 | }, 2092 | "p-map": { 2093 | "version": "4.0.0", 2094 | "resolved": "https://registry.npmjs.org/p-map/-/p-map-4.0.0.tgz", 2095 | "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", 2096 | "optional": true, 2097 | "requires": { 2098 | "aggregate-error": "^3.0.0" 2099 | } 2100 | }, 2101 | "parseurl": { 2102 | "version": "1.3.3", 2103 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 2104 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 2105 | }, 2106 | "path-exists": { 2107 | "version": "4.0.0", 2108 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 2109 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 2110 | "dev": true 2111 | }, 2112 | "path-is-absolute": { 2113 | "version": "1.0.1", 2114 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 2115 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" 2116 | }, 2117 | "path-to-regexp": { 2118 | "version": "0.1.7", 2119 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 2120 | "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==" 2121 | }, 2122 | "pathval": { 2123 | "version": "1.1.1", 2124 | "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz", 2125 | "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==", 2126 | "dev": true 2127 | }, 2128 | "picomatch": { 2129 | "version": "2.3.1", 2130 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 2131 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 2132 | "dev": true 2133 | }, 2134 | "prebuild-install": { 2135 | "version": "6.0.0", 2136 | "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-6.0.0.tgz", 2137 | "integrity": "sha512-h2ZJ1PXHKWZpp1caLw0oX9sagVpL2YTk+ZwInQbQ3QqNd4J03O6MpFNmMTJlkfgPENWqe5kP0WjQLqz5OjLfsw==", 2138 | "requires": { 2139 | "detect-libc": "^1.0.3", 2140 | "expand-template": "^2.0.3", 2141 | "github-from-package": "0.0.0", 2142 | "minimist": "^1.2.3", 2143 | "mkdirp-classic": "^0.5.3", 2144 | "napi-build-utils": "^1.0.1", 2145 | "node-abi": "^2.7.0", 2146 | "noop-logger": "^0.1.1", 2147 | "npmlog": "^4.0.1", 2148 | "pump": "^3.0.0", 2149 | "rc": "^1.2.7", 2150 | "simple-get": "^3.0.3", 2151 | "tar-fs": "^2.0.0", 2152 | "tunnel-agent": "^0.6.0", 2153 | "which-pm-runs": "^1.0.0" 2154 | } 2155 | }, 2156 | "process-nextick-args": { 2157 | "version": "2.0.1", 2158 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 2159 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 2160 | }, 2161 | "promise-inflight": { 2162 | "version": "1.0.1", 2163 | "resolved": "https://registry.npmjs.org/promise-inflight/-/promise-inflight-1.0.1.tgz", 2164 | "integrity": "sha512-6zWPyEOFaQBJYcGMHBKTKJ3u6TBsnMFOIZSa6ce1e/ZrrsOlnHRHbabMjLiBYKp+n44X9eUI6VUPaukCXHuG4g==", 2165 | "optional": true 2166 | }, 2167 | "promise-retry": { 2168 | "version": "2.0.1", 2169 | "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-2.0.1.tgz", 2170 | "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", 2171 | "optional": true, 2172 | "requires": { 2173 | "err-code": "^2.0.2", 2174 | "retry": "^0.12.0" 2175 | } 2176 | }, 2177 | "proxy-addr": { 2178 | "version": "2.0.7", 2179 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 2180 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 2181 | "requires": { 2182 | "forwarded": "0.2.0", 2183 | "ipaddr.js": "1.9.1" 2184 | } 2185 | }, 2186 | "pump": { 2187 | "version": "3.0.0", 2188 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 2189 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 2190 | "requires": { 2191 | "end-of-stream": "^1.1.0", 2192 | "once": "^1.3.1" 2193 | } 2194 | }, 2195 | "qs": { 2196 | "version": "6.11.0", 2197 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz", 2198 | "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==", 2199 | "requires": { 2200 | "side-channel": "^1.0.4" 2201 | } 2202 | }, 2203 | "randombytes": { 2204 | "version": "2.1.0", 2205 | "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", 2206 | "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", 2207 | "dev": true, 2208 | "requires": { 2209 | "safe-buffer": "^5.1.0" 2210 | } 2211 | }, 2212 | "range-parser": { 2213 | "version": "1.2.1", 2214 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 2215 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 2216 | }, 2217 | "raw-body": { 2218 | "version": "2.5.2", 2219 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", 2220 | "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", 2221 | "requires": { 2222 | "bytes": "3.1.2", 2223 | "http-errors": "2.0.0", 2224 | "iconv-lite": "0.4.24", 2225 | "unpipe": "1.0.0" 2226 | } 2227 | }, 2228 | "rc": { 2229 | "version": "1.2.8", 2230 | "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", 2231 | "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", 2232 | "requires": { 2233 | "deep-extend": "^0.6.0", 2234 | "ini": "~1.3.0", 2235 | "minimist": "^1.2.0", 2236 | "strip-json-comments": "~2.0.1" 2237 | } 2238 | }, 2239 | "readable-stream": { 2240 | "version": "2.3.7", 2241 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 2242 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 2243 | "requires": { 2244 | "core-util-is": "~1.0.0", 2245 | "inherits": "~2.0.3", 2246 | "isarray": "~1.0.0", 2247 | "process-nextick-args": "~2.0.0", 2248 | "safe-buffer": "~5.1.1", 2249 | "string_decoder": "~1.1.1", 2250 | "util-deprecate": "~1.0.1" 2251 | } 2252 | }, 2253 | "readdirp": { 2254 | "version": "3.6.0", 2255 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 2256 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 2257 | "dev": true, 2258 | "requires": { 2259 | "picomatch": "^2.2.1" 2260 | } 2261 | }, 2262 | "require-directory": { 2263 | "version": "2.1.1", 2264 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 2265 | "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", 2266 | "dev": true 2267 | }, 2268 | "retry": { 2269 | "version": "0.12.0", 2270 | "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz", 2271 | "integrity": "sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==", 2272 | "optional": true 2273 | }, 2274 | "rimraf": { 2275 | "version": "3.0.2", 2276 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 2277 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 2278 | "optional": true, 2279 | "requires": { 2280 | "glob": "^7.1.3" 2281 | } 2282 | }, 2283 | "safe-buffer": { 2284 | "version": "5.1.2", 2285 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 2286 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 2287 | }, 2288 | "safer-buffer": { 2289 | "version": "2.1.2", 2290 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 2291 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 2292 | }, 2293 | "semver": { 2294 | "version": "5.7.2", 2295 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", 2296 | "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==" 2297 | }, 2298 | "send": { 2299 | "version": "0.18.0", 2300 | "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", 2301 | "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", 2302 | "requires": { 2303 | "debug": "2.6.9", 2304 | "depd": "2.0.0", 2305 | "destroy": "1.2.0", 2306 | "encodeurl": "~1.0.2", 2307 | "escape-html": "~1.0.3", 2308 | "etag": "~1.8.1", 2309 | "fresh": "0.5.2", 2310 | "http-errors": "2.0.0", 2311 | "mime": "1.6.0", 2312 | "ms": "2.1.3", 2313 | "on-finished": "2.4.1", 2314 | "range-parser": "~1.2.1", 2315 | "statuses": "2.0.1" 2316 | }, 2317 | "dependencies": { 2318 | "ms": { 2319 | "version": "2.1.3", 2320 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 2321 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 2322 | } 2323 | } 2324 | }, 2325 | "serialize-javascript": { 2326 | "version": "6.0.0", 2327 | "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz", 2328 | "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==", 2329 | "dev": true, 2330 | "requires": { 2331 | "randombytes": "^2.1.0" 2332 | } 2333 | }, 2334 | "serialport": { 2335 | "version": "9.0.6", 2336 | "resolved": "https://registry.npmjs.org/serialport/-/serialport-9.0.6.tgz", 2337 | "integrity": "sha512-T9eY4HFzQij0Hd/RsPcZySdeuAqzV5iGICHz8FXUSKVn2SvGT5zjfz/H+pRwI86k+3iFVOyddEyy8gbVNVbW7A==", 2338 | "requires": { 2339 | "@serialport/binding-mock": "^9.0.2", 2340 | "@serialport/bindings": "^9.0.4", 2341 | "@serialport/parser-byte-length": "^9.0.1", 2342 | "@serialport/parser-cctalk": "^9.0.1", 2343 | "@serialport/parser-delimiter": "^9.0.1", 2344 | "@serialport/parser-inter-byte-timeout": "^9.0.1", 2345 | "@serialport/parser-readline": "^9.0.1", 2346 | "@serialport/parser-ready": "^9.0.1", 2347 | "@serialport/parser-regex": "^9.0.1", 2348 | "@serialport/stream": "^9.0.2", 2349 | "debug": "^4.1.1" 2350 | }, 2351 | "dependencies": { 2352 | "debug": { 2353 | "version": "4.3.1", 2354 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", 2355 | "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", 2356 | "requires": { 2357 | "ms": "2.1.2" 2358 | } 2359 | }, 2360 | "ms": { 2361 | "version": "2.1.2", 2362 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2363 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 2364 | } 2365 | } 2366 | }, 2367 | "serve-static": { 2368 | "version": "1.15.0", 2369 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", 2370 | "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", 2371 | "requires": { 2372 | "encodeurl": "~1.0.2", 2373 | "escape-html": "~1.0.3", 2374 | "parseurl": "~1.3.3", 2375 | "send": "0.18.0" 2376 | } 2377 | }, 2378 | "set-blocking": { 2379 | "version": "2.0.0", 2380 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 2381 | "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" 2382 | }, 2383 | "set-function-length": { 2384 | "version": "1.2.2", 2385 | "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", 2386 | "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", 2387 | "requires": { 2388 | "define-data-property": "^1.1.4", 2389 | "es-errors": "^1.3.0", 2390 | "function-bind": "^1.1.2", 2391 | "get-intrinsic": "^1.2.4", 2392 | "gopd": "^1.0.1", 2393 | "has-property-descriptors": "^1.0.2" 2394 | } 2395 | }, 2396 | "setprototypeof": { 2397 | "version": "1.2.0", 2398 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 2399 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" 2400 | }, 2401 | "side-channel": { 2402 | "version": "1.0.6", 2403 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", 2404 | "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", 2405 | "requires": { 2406 | "call-bind": "^1.0.7", 2407 | "es-errors": "^1.3.0", 2408 | "get-intrinsic": "^1.2.4", 2409 | "object-inspect": "^1.13.1" 2410 | } 2411 | }, 2412 | "signal-exit": { 2413 | "version": "3.0.3", 2414 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", 2415 | "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==" 2416 | }, 2417 | "simple-concat": { 2418 | "version": "1.0.1", 2419 | "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", 2420 | "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==" 2421 | }, 2422 | "simple-get": { 2423 | "version": "3.1.1", 2424 | "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-3.1.1.tgz", 2425 | "integrity": "sha512-CQ5LTKGfCpvE1K0n2us+kuMPbk/q0EKl82s4aheV9oXjFEz6W/Y7oQFVJuU6QG77hRT4Ghb5RURteF5vnWjupA==", 2426 | "requires": { 2427 | "decompress-response": "^4.2.0", 2428 | "once": "^1.3.1", 2429 | "simple-concat": "^1.0.0" 2430 | } 2431 | }, 2432 | "smart-buffer": { 2433 | "version": "4.2.0", 2434 | "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", 2435 | "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", 2436 | "optional": true 2437 | }, 2438 | "socket.io": { 2439 | "version": "4.6.1", 2440 | "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.6.1.tgz", 2441 | "integrity": "sha512-KMcaAi4l/8+xEjkRICl6ak8ySoxsYG+gG6/XfRCPJPQ/haCRIJBTL4wIl8YCsmtaBovcAXGLOShyVWQ/FG8GZA==", 2442 | "requires": { 2443 | "accepts": "~1.3.4", 2444 | "base64id": "~2.0.0", 2445 | "debug": "~4.3.2", 2446 | "engine.io": "~6.4.1", 2447 | "socket.io-adapter": "~2.5.2", 2448 | "socket.io-parser": "~4.2.1" 2449 | }, 2450 | "dependencies": { 2451 | "debug": { 2452 | "version": "4.3.4", 2453 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 2454 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 2455 | "requires": { 2456 | "ms": "2.1.2" 2457 | } 2458 | }, 2459 | "ms": { 2460 | "version": "2.1.2", 2461 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2462 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 2463 | } 2464 | } 2465 | }, 2466 | "socket.io-adapter": { 2467 | "version": "2.5.2", 2468 | "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.5.2.tgz", 2469 | "integrity": "sha512-87C3LO/NOMc+eMcpcxUBebGjkpMDkNBS9tf7KJqcDsmL936EChtVva71Dw2q4tQcuVC+hAUy4an2NO/sYXmwRA==", 2470 | "requires": { 2471 | "ws": "~8.11.0" 2472 | }, 2473 | "dependencies": { 2474 | "ws": { 2475 | "version": "8.11.0", 2476 | "resolved": "https://registry.npmjs.org/ws/-/ws-8.11.0.tgz", 2477 | "integrity": "sha512-HPG3wQd9sNQoT9xHyNCXoDUa+Xw/VevmY9FoHyQ+g+rrMn4j6FB4np7Z0OhdTgjx6MgQLK7jwSy1YecU1+4Asg==" 2478 | } 2479 | } 2480 | }, 2481 | "socket.io-client": { 2482 | "version": "4.6.1", 2483 | "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-4.6.1.tgz", 2484 | "integrity": "sha512-5UswCV6hpaRsNg5kkEHVcbBIXEYoVbMQaHJBXJCyEQ+CiFPV1NIOY0XOFWG4XR4GZcB8Kn6AsRs/9cy9TbqVMQ==", 2485 | "dev": true, 2486 | "requires": { 2487 | "@socket.io/component-emitter": "~3.1.0", 2488 | "debug": "~4.3.2", 2489 | "engine.io-client": "~6.4.0", 2490 | "socket.io-parser": "~4.2.1" 2491 | }, 2492 | "dependencies": { 2493 | "debug": { 2494 | "version": "4.3.4", 2495 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 2496 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 2497 | "dev": true, 2498 | "requires": { 2499 | "ms": "2.1.2" 2500 | } 2501 | }, 2502 | "ms": { 2503 | "version": "2.1.2", 2504 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2505 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 2506 | "dev": true 2507 | } 2508 | } 2509 | }, 2510 | "socket.io-parser": { 2511 | "version": "4.2.3", 2512 | "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.2.3.tgz", 2513 | "integrity": "sha512-JMafRntWVO2DCJimKsRTh/wnqVvO4hrfwOqtO7f+uzwsQMuxO6VwImtYxaQ+ieoyshWOTJyV0fA21lccEXRPpQ==", 2514 | "requires": { 2515 | "@socket.io/component-emitter": "~3.1.0", 2516 | "debug": "~4.3.1" 2517 | }, 2518 | "dependencies": { 2519 | "debug": { 2520 | "version": "4.3.4", 2521 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 2522 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 2523 | "requires": { 2524 | "ms": "2.1.2" 2525 | } 2526 | }, 2527 | "ms": { 2528 | "version": "2.1.2", 2529 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2530 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 2531 | } 2532 | } 2533 | }, 2534 | "socks": { 2535 | "version": "2.8.3", 2536 | "resolved": "https://registry.npmjs.org/socks/-/socks-2.8.3.tgz", 2537 | "integrity": "sha512-l5x7VUUWbjVFbafGLxPWkYsHIhEvmF85tbIeFZWc8ZPtoMyybuEhL7Jye/ooC4/d48FgOjSJXgsF/AJPYCW8Zw==", 2538 | "optional": true, 2539 | "requires": { 2540 | "ip-address": "^9.0.5", 2541 | "smart-buffer": "^4.2.0" 2542 | } 2543 | }, 2544 | "socks-proxy-agent": { 2545 | "version": "6.2.1", 2546 | "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-6.2.1.tgz", 2547 | "integrity": "sha512-a6KW9G+6B3nWZ1yB8G7pJwL3ggLy1uTzKAgCb7ttblwqdz9fMGJUuTy3uFzEP48FAs9FLILlmzDlE2JJhVQaXQ==", 2548 | "optional": true, 2549 | "requires": { 2550 | "agent-base": "^6.0.2", 2551 | "debug": "^4.3.3", 2552 | "socks": "^2.6.2" 2553 | }, 2554 | "dependencies": { 2555 | "debug": { 2556 | "version": "4.3.4", 2557 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 2558 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 2559 | "optional": true, 2560 | "requires": { 2561 | "ms": "2.1.2" 2562 | } 2563 | }, 2564 | "ms": { 2565 | "version": "2.1.2", 2566 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2567 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 2568 | "optional": true 2569 | } 2570 | } 2571 | }, 2572 | "sprintf-js": { 2573 | "version": "1.1.3", 2574 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.3.tgz", 2575 | "integrity": "sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==", 2576 | "optional": true 2577 | }, 2578 | "sqlite3": { 2579 | "version": "5.1.7", 2580 | "resolved": "https://registry.npmjs.org/sqlite3/-/sqlite3-5.1.7.tgz", 2581 | "integrity": "sha512-GGIyOiFaG+TUra3JIfkI/zGP8yZYLPQ0pl1bH+ODjiX57sPhrLU5sQJn1y9bDKZUFYkX1crlrPfSYt0BKKdkog==", 2582 | "requires": { 2583 | "bindings": "^1.5.0", 2584 | "node-addon-api": "^7.0.0", 2585 | "node-gyp": "8.x", 2586 | "prebuild-install": "^7.1.1", 2587 | "tar": "^6.1.11" 2588 | }, 2589 | "dependencies": { 2590 | "decompress-response": { 2591 | "version": "6.0.0", 2592 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", 2593 | "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", 2594 | "requires": { 2595 | "mimic-response": "^3.1.0" 2596 | } 2597 | }, 2598 | "detect-libc": { 2599 | "version": "2.0.3", 2600 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", 2601 | "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==" 2602 | }, 2603 | "mimic-response": { 2604 | "version": "3.1.0", 2605 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", 2606 | "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==" 2607 | }, 2608 | "node-abi": { 2609 | "version": "3.57.0", 2610 | "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.57.0.tgz", 2611 | "integrity": "sha512-Dp+A9JWxRaKuHP35H77I4kCKesDy5HUDEmScia2FyncMTOXASMyg251F5PhFoDA5uqBrDDffiLpbqnrZmNXW+g==", 2612 | "requires": { 2613 | "semver": "^7.3.5" 2614 | } 2615 | }, 2616 | "prebuild-install": { 2617 | "version": "7.1.2", 2618 | "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.2.tgz", 2619 | "integrity": "sha512-UnNke3IQb6sgarcZIDU3gbMeTp/9SSU1DAIkil7PrqG1vZlBtY5msYccSKSHDqa3hNg436IXK+SNImReuA1wEQ==", 2620 | "requires": { 2621 | "detect-libc": "^2.0.0", 2622 | "expand-template": "^2.0.3", 2623 | "github-from-package": "0.0.0", 2624 | "minimist": "^1.2.3", 2625 | "mkdirp-classic": "^0.5.3", 2626 | "napi-build-utils": "^1.0.1", 2627 | "node-abi": "^3.3.0", 2628 | "pump": "^3.0.0", 2629 | "rc": "^1.2.7", 2630 | "simple-get": "^4.0.0", 2631 | "tar-fs": "^2.0.0", 2632 | "tunnel-agent": "^0.6.0" 2633 | } 2634 | }, 2635 | "semver": { 2636 | "version": "7.6.0", 2637 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", 2638 | "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", 2639 | "requires": { 2640 | "lru-cache": "^6.0.0" 2641 | } 2642 | }, 2643 | "simple-get": { 2644 | "version": "4.0.1", 2645 | "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz", 2646 | "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==", 2647 | "requires": { 2648 | "decompress-response": "^6.0.0", 2649 | "once": "^1.3.1", 2650 | "simple-concat": "^1.0.0" 2651 | } 2652 | } 2653 | } 2654 | }, 2655 | "ssri": { 2656 | "version": "8.0.1", 2657 | "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.1.tgz", 2658 | "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", 2659 | "optional": true, 2660 | "requires": { 2661 | "minipass": "^3.1.1" 2662 | } 2663 | }, 2664 | "statuses": { 2665 | "version": "2.0.1", 2666 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 2667 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==" 2668 | }, 2669 | "string-width": { 2670 | "version": "1.0.2", 2671 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", 2672 | "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", 2673 | "requires": { 2674 | "code-point-at": "^1.0.0", 2675 | "is-fullwidth-code-point": "^1.0.0", 2676 | "strip-ansi": "^3.0.0" 2677 | } 2678 | }, 2679 | "string_decoder": { 2680 | "version": "1.1.1", 2681 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 2682 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 2683 | "requires": { 2684 | "safe-buffer": "~5.1.0" 2685 | } 2686 | }, 2687 | "strip-ansi": { 2688 | "version": "3.0.1", 2689 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", 2690 | "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", 2691 | "requires": { 2692 | "ansi-regex": "^2.0.0" 2693 | } 2694 | }, 2695 | "strip-json-comments": { 2696 | "version": "2.0.1", 2697 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 2698 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=" 2699 | }, 2700 | "supports-color": { 2701 | "version": "7.2.0", 2702 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 2703 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 2704 | "requires": { 2705 | "has-flag": "^4.0.0" 2706 | }, 2707 | "dependencies": { 2708 | "has-flag": { 2709 | "version": "4.0.0", 2710 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 2711 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" 2712 | } 2713 | } 2714 | }, 2715 | "tar": { 2716 | "version": "6.2.1", 2717 | "resolved": "https://registry.npmjs.org/tar/-/tar-6.2.1.tgz", 2718 | "integrity": "sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A==", 2719 | "requires": { 2720 | "chownr": "^2.0.0", 2721 | "fs-minipass": "^2.0.0", 2722 | "minipass": "^5.0.0", 2723 | "minizlib": "^2.1.1", 2724 | "mkdirp": "^1.0.3", 2725 | "yallist": "^4.0.0" 2726 | }, 2727 | "dependencies": { 2728 | "chownr": { 2729 | "version": "2.0.0", 2730 | "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", 2731 | "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==" 2732 | }, 2733 | "minipass": { 2734 | "version": "5.0.0", 2735 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-5.0.0.tgz", 2736 | "integrity": "sha512-3FnjYuehv9k6ovOEbyOswadCDPX1piCfhV8ncmYtHOjuPwylVWsghTLo7rabjC3Rx5xD4HDx8Wm1xnMF7S5qFQ==" 2737 | } 2738 | } 2739 | }, 2740 | "tar-fs": { 2741 | "version": "2.1.1", 2742 | "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz", 2743 | "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==", 2744 | "requires": { 2745 | "chownr": "^1.1.1", 2746 | "mkdirp-classic": "^0.5.2", 2747 | "pump": "^3.0.0", 2748 | "tar-stream": "^2.1.4" 2749 | } 2750 | }, 2751 | "tar-stream": { 2752 | "version": "2.2.0", 2753 | "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", 2754 | "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", 2755 | "requires": { 2756 | "bl": "^4.0.3", 2757 | "end-of-stream": "^1.4.1", 2758 | "fs-constants": "^1.0.0", 2759 | "inherits": "^2.0.3", 2760 | "readable-stream": "^3.1.1" 2761 | }, 2762 | "dependencies": { 2763 | "readable-stream": { 2764 | "version": "3.6.0", 2765 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", 2766 | "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", 2767 | "requires": { 2768 | "inherits": "^2.0.3", 2769 | "string_decoder": "^1.1.1", 2770 | "util-deprecate": "^1.0.1" 2771 | } 2772 | } 2773 | } 2774 | }, 2775 | "to-regex-range": { 2776 | "version": "5.0.1", 2777 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2778 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2779 | "dev": true, 2780 | "requires": { 2781 | "is-number": "^7.0.0" 2782 | } 2783 | }, 2784 | "toidentifier": { 2785 | "version": "1.0.1", 2786 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 2787 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==" 2788 | }, 2789 | "tunnel-agent": { 2790 | "version": "0.6.0", 2791 | "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", 2792 | "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", 2793 | "requires": { 2794 | "safe-buffer": "^5.0.1" 2795 | } 2796 | }, 2797 | "type-detect": { 2798 | "version": "4.0.8", 2799 | "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", 2800 | "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", 2801 | "dev": true 2802 | }, 2803 | "type-is": { 2804 | "version": "1.6.18", 2805 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 2806 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 2807 | "requires": { 2808 | "media-typer": "0.3.0", 2809 | "mime-types": "~2.1.24" 2810 | } 2811 | }, 2812 | "unique-filename": { 2813 | "version": "1.1.1", 2814 | "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-1.1.1.tgz", 2815 | "integrity": "sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==", 2816 | "optional": true, 2817 | "requires": { 2818 | "unique-slug": "^2.0.0" 2819 | } 2820 | }, 2821 | "unique-slug": { 2822 | "version": "2.0.2", 2823 | "resolved": "https://registry.npmjs.org/unique-slug/-/unique-slug-2.0.2.tgz", 2824 | "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", 2825 | "optional": true, 2826 | "requires": { 2827 | "imurmurhash": "^0.1.4" 2828 | } 2829 | }, 2830 | "unpipe": { 2831 | "version": "1.0.0", 2832 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 2833 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==" 2834 | }, 2835 | "util-deprecate": { 2836 | "version": "1.0.2", 2837 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 2838 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 2839 | }, 2840 | "utils-merge": { 2841 | "version": "1.0.1", 2842 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2843 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==" 2844 | }, 2845 | "vary": { 2846 | "version": "1.1.2", 2847 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2848 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 2849 | }, 2850 | "which": { 2851 | "version": "2.0.2", 2852 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 2853 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 2854 | "optional": true, 2855 | "requires": { 2856 | "isexe": "^2.0.0" 2857 | } 2858 | }, 2859 | "which-pm-runs": { 2860 | "version": "1.0.0", 2861 | "resolved": "https://registry.npmjs.org/which-pm-runs/-/which-pm-runs-1.0.0.tgz", 2862 | "integrity": "sha1-Zws6+8VS4LVd9rd4DKdGFfI60cs=" 2863 | }, 2864 | "wide-align": { 2865 | "version": "1.1.3", 2866 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", 2867 | "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", 2868 | "requires": { 2869 | "string-width": "^1.0.2 || 2" 2870 | } 2871 | }, 2872 | "workerpool": { 2873 | "version": "6.2.1", 2874 | "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.2.1.tgz", 2875 | "integrity": "sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw==", 2876 | "dev": true 2877 | }, 2878 | "wrap-ansi": { 2879 | "version": "7.0.0", 2880 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 2881 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 2882 | "dev": true, 2883 | "requires": { 2884 | "ansi-styles": "^4.0.0", 2885 | "string-width": "^4.1.0", 2886 | "strip-ansi": "^6.0.0" 2887 | }, 2888 | "dependencies": { 2889 | "ansi-regex": { 2890 | "version": "5.0.1", 2891 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 2892 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 2893 | "dev": true 2894 | }, 2895 | "ansi-styles": { 2896 | "version": "4.3.0", 2897 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 2898 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 2899 | "dev": true, 2900 | "requires": { 2901 | "color-convert": "^2.0.1" 2902 | } 2903 | }, 2904 | "color-convert": { 2905 | "version": "2.0.1", 2906 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 2907 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 2908 | "dev": true, 2909 | "requires": { 2910 | "color-name": "~1.1.4" 2911 | } 2912 | }, 2913 | "color-name": { 2914 | "version": "1.1.4", 2915 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 2916 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 2917 | "dev": true 2918 | }, 2919 | "is-fullwidth-code-point": { 2920 | "version": "3.0.0", 2921 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 2922 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 2923 | "dev": true 2924 | }, 2925 | "string-width": { 2926 | "version": "4.2.3", 2927 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 2928 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 2929 | "dev": true, 2930 | "requires": { 2931 | "emoji-regex": "^8.0.0", 2932 | "is-fullwidth-code-point": "^3.0.0", 2933 | "strip-ansi": "^6.0.1" 2934 | } 2935 | }, 2936 | "strip-ansi": { 2937 | "version": "6.0.1", 2938 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 2939 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 2940 | "dev": true, 2941 | "requires": { 2942 | "ansi-regex": "^5.0.1" 2943 | } 2944 | } 2945 | } 2946 | }, 2947 | "wrappy": { 2948 | "version": "1.0.2", 2949 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2950 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" 2951 | }, 2952 | "ws": { 2953 | "version": "8.11.0", 2954 | "resolved": "https://registry.npmjs.org/ws/-/ws-8.11.0.tgz", 2955 | "integrity": "sha512-HPG3wQd9sNQoT9xHyNCXoDUa+Xw/VevmY9FoHyQ+g+rrMn4j6FB4np7Z0OhdTgjx6MgQLK7jwSy1YecU1+4Asg==", 2956 | "dev": true 2957 | }, 2958 | "xmlhttprequest-ssl": { 2959 | "version": "2.0.0", 2960 | "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-2.0.0.tgz", 2961 | "integrity": "sha512-QKxVRxiRACQcVuQEYFsI1hhkrMlrXHPegbbd1yn9UHOmRxY+si12nQYzri3vbzt8VdTTRviqcKxcyllFas5z2A==", 2962 | "dev": true 2963 | }, 2964 | "y18n": { 2965 | "version": "5.0.8", 2966 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 2967 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 2968 | "dev": true 2969 | }, 2970 | "yallist": { 2971 | "version": "4.0.0", 2972 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 2973 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" 2974 | }, 2975 | "yargs": { 2976 | "version": "16.2.0", 2977 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", 2978 | "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", 2979 | "dev": true, 2980 | "requires": { 2981 | "cliui": "^7.0.2", 2982 | "escalade": "^3.1.1", 2983 | "get-caller-file": "^2.0.5", 2984 | "require-directory": "^2.1.1", 2985 | "string-width": "^4.2.0", 2986 | "y18n": "^5.0.5", 2987 | "yargs-parser": "^20.2.2" 2988 | }, 2989 | "dependencies": { 2990 | "ansi-regex": { 2991 | "version": "5.0.1", 2992 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 2993 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 2994 | "dev": true 2995 | }, 2996 | "is-fullwidth-code-point": { 2997 | "version": "3.0.0", 2998 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 2999 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 3000 | "dev": true 3001 | }, 3002 | "string-width": { 3003 | "version": "4.2.3", 3004 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 3005 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 3006 | "dev": true, 3007 | "requires": { 3008 | "emoji-regex": "^8.0.0", 3009 | "is-fullwidth-code-point": "^3.0.0", 3010 | "strip-ansi": "^6.0.1" 3011 | } 3012 | }, 3013 | "strip-ansi": { 3014 | "version": "6.0.1", 3015 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 3016 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 3017 | "dev": true, 3018 | "requires": { 3019 | "ansi-regex": "^5.0.1" 3020 | } 3021 | } 3022 | } 3023 | }, 3024 | "yargs-parser": { 3025 | "version": "20.2.4", 3026 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", 3027 | "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", 3028 | "dev": true 3029 | }, 3030 | "yargs-unparser": { 3031 | "version": "2.0.0", 3032 | "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz", 3033 | "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==", 3034 | "dev": true, 3035 | "requires": { 3036 | "camelcase": "^6.0.0", 3037 | "decamelize": "^4.0.0", 3038 | "flat": "^5.0.2", 3039 | "is-plain-obj": "^2.1.0" 3040 | } 3041 | }, 3042 | "yocto-queue": { 3043 | "version": "0.1.0", 3044 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 3045 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 3046 | "dev": true 3047 | } 3048 | } 3049 | } 3050 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "modbus-ws", 3 | "version": "2.2.0", 4 | "description": "NodeJS Modbus to WebSocket bridge", 5 | "main": "index.js", 6 | "bin": { 7 | "modbus-ws": "./bin/modbus-ws.js" 8 | }, 9 | "scripts": { 10 | "start": "node ./bin/modbus-ws.js", 11 | "start-serial": "node ./bin/modbus-ws.js -s /dev/ttyACM0 -b 19200", 12 | "test": "mocha" 13 | }, 14 | "repository": { 15 | "type": "git", 16 | "url": "git+https://github.com/yaacov/node-modbus-ws.git" 17 | }, 18 | "keywords": [ 19 | "modbus", 20 | "websocket", 21 | "socket.io", 22 | "server" 23 | ], 24 | "author": "Yaacov Zamir ", 25 | "license": "ISC", 26 | "bugs": { 27 | "url": "https://github.com/yaacov/node-modbus-ws/issues" 28 | }, 29 | "homepage": "https://github.com/yaacov/node-modbus-ws#readme", 30 | "dependencies": { 31 | "commander": "^7.0.0", 32 | "ejs": "^3.1.5", 33 | "express": "^4.16.3", 34 | "modbus-serial": "^8.0.1", 35 | "socket.io": "^4.5.3", 36 | "sqlite3": "^5.1.7" 37 | }, 38 | "devDependencies": { 39 | "chai": "^4.1.2", 40 | "mocha": "^10.1.0", 41 | "socket.io-client": "^4.6.1" 42 | } 43 | } 44 | -------------------------------------------------------------------------------- /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 | } -------------------------------------------------------------------------------- /server.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Copyright (c) 2015, Yaacov Zamir 3 | * 4 | * Permission to use, copy, modify, and/or distribute this software for any 5 | * purpose with or without fee is hereby granted, provided that the above 6 | * copyright notice and this permission notice appear in all copies. 7 | * 8 | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 9 | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 10 | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 11 | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 12 | * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 13 | * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 14 | * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 15 | */ 16 | 17 | var io; 18 | var ModbusRTU = require("modbus-serial"); 19 | var version = require('./package.json').version; 20 | 21 | var serialPort; 22 | var modbusRTU; 23 | 24 | var getCoils; 25 | var getInputStatus; 26 | var getInputRegisters; 27 | var getHoldingRegisters; 28 | var forceCoil; 29 | var setRegisters; 30 | var debug; 31 | 32 | /** 33 | * Write a Modbus "Read Coils" (FC=01) to serial port, 34 | * and emit the replay to websocket 35 | * 36 | * @param {number} unit the slave unit address. 37 | * @param {number} address the Data Address of the first coil. 38 | * @param {number} length the total number of coils requested. 39 | */ 40 | var _getCoils = function(unit, address, length) { 41 | modbusRTU.writeFC1(unit, address, length, 42 | function(err, msg) { 43 | if (err) { 44 | console.log(err); 45 | io.emit('data', {'err': err}); 46 | } else { 47 | io.emit('data', { 48 | 'unit': unit, 49 | 'type': 1, 50 | 'address': address, 51 | 'data': msg.data, 52 | 'flag': 'get' 53 | }); 54 | } 55 | } 56 | ); 57 | } 58 | 59 | /** 60 | * Write a Modbus "Read input status" (FC=02) to serial port, 61 | * and emit the replay to websocket 62 | * 63 | * @param {number} unit the slave unit address. 64 | * @param {number} address the Data Address of the first digital input. 65 | * @param {number} length the total number of digital inputs requested. 66 | */ 67 | var _getInputStatus = function(unit, address, length) { 68 | modbusRTU.writeFC2(unit, address, length, 69 | function(err, msg) { 70 | if (err) { 71 | console.log(err); 72 | io.emit('data', {'err': err}); 73 | } else { 74 | io.emit('data', { 75 | 'unit': unit, 76 | 'type': 2, 77 | 'address': address, 78 | 'data': msg.data, 79 | 'flag': 'get' 80 | }); 81 | } 82 | } 83 | ); 84 | } 85 | 86 | /** 87 | * Write a Modbus "Read Holding Registers" (FC=03) to serial port, 88 | * and emit the replay to websocket 89 | * 90 | * @param {number} unit the slave unit address. 91 | * @param {number} address the Data Address of the first register. 92 | * @param {number} length the total number of registers requested. 93 | */ 94 | var _getHoldingRegisters = function(unit, address, length) { 95 | modbusRTU.writeFC3(unit, address, length, 96 | function(err, msg) { 97 | if (err) { 98 | console.log(err); 99 | io.emit('data', {'err': err}); 100 | } else { 101 | io.emit('data', { 102 | 'unit': unit, 103 | 'type': 3, 104 | 'address': address, 105 | 'data': msg.data, 106 | 'flag': 'get' 107 | }); 108 | } 109 | } 110 | ); 111 | } 112 | 113 | /** 114 | * Write a Modbus "Read Input Registers" (FC=04) to serial port, 115 | * and emit the replay to websocket 116 | * 117 | * @param {number} unit the slave unit address. 118 | * @param {number} address the Data Address of the first register. 119 | * @param {number} length the total number of registers requested. 120 | */ 121 | var _getInputRegisters = function(unit, address, length) { 122 | modbusRTU.writeFC4(unit, address, length, 123 | function(err, msg) { 124 | if (err) { 125 | console.log(err); 126 | io.emit('data', {'err': err}); 127 | } else { 128 | io.emit('data', { 129 | 'unit': unit, 130 | 'type': 4, 131 | 'address': address, 132 | 'data': msg.data, 133 | 'flag': 'get' 134 | }); 135 | } 136 | } 137 | ); 138 | } 139 | 140 | /** 141 | * Write a Modbus "Force one coil" (FC=05) to serial port, 142 | * and emit the replay to websocket 143 | * 144 | * @param {number} unit the slave unit address. 145 | * @param {number} address the Data Address of the coil. 146 | * @param {number} state the state to set into coil. 147 | */ 148 | var _forceCoil = function(unit, address, state) { 149 | modbusRTU.writeFC5(unit, address, state, 150 | function(err, msg) { 151 | if (err) { 152 | console.log(err); 153 | io.emit('data', {'err': err}); 154 | } else { 155 | io.emit('data', { 156 | 'unit': unit, 157 | 'type': 5, 158 | 'address': address, 159 | 'data': state, 160 | 'flag': 'set' 161 | }); 162 | } 163 | } 164 | ); 165 | } 166 | 167 | /** 168 | * Write a Modbus "Preset Multiple Registers" (FC=16) to serial port, 169 | * and emit the replay to websocket 170 | * 171 | * @param {number} unit the slave unit address. 172 | * @param {number} address the Data Address of the first register. 173 | * @param {array} values the array of values to write to registers. 174 | */ 175 | var _setRegisters = function(unit, address, values) { 176 | modbusRTU.writeFC16(unit, address, values, 177 | function(err, msg) { 178 | if (err) { 179 | console.log(err); 180 | io.emit('data', {'err': err}); 181 | } else { 182 | io.emit('data', { 183 | 'unit': unit, 184 | 'type': 3, 185 | 'address': address, 186 | 'data': values, 187 | 'flag': 'set' 188 | }); 189 | } 190 | } 191 | ); 192 | } 193 | 194 | /** 195 | * Setup the socket.io events 196 | */ 197 | var setup = function() { 198 | /* register socker io events 199 | */ 200 | io.on('connection', function(socket){ 201 | var intervalIDs = []; 202 | 203 | //console.log('client connected'); 204 | 205 | socket.on('disconnect', function(){ 206 | // clear all periodically requests 207 | intervalIDs.map(clearInterval); 208 | 209 | //console.log('client disconnected'); 210 | }); 211 | 212 | socket.on('readCoils', function(data){ 213 | // check event validity 214 | if (!data) return; 215 | 216 | var unit = data.unit; 217 | var address = data.address; 218 | var length = data.length; 219 | 220 | // check event validity 221 | if (!unit || typeof address == 'undefined' || !length) return; 222 | 223 | /* if client request an interval, 224 | * set a time interval and emit data 225 | * periodically. 226 | */ 227 | var interval = data.interval; 228 | if (interval) { 229 | var id = setInterval(function() { 230 | getCoils(unit, address, length); 231 | }, interval); 232 | 233 | intervalIDs.push(id); 234 | } else { 235 | getCoils(unit, address, length); 236 | } 237 | }); 238 | 239 | socket.on('readDiscreteInputs', function(data){ 240 | // check event validity 241 | if (!data) return; 242 | 243 | var unit = data.unit; 244 | var address = data.address; 245 | var length = data.length; 246 | 247 | // check event validity 248 | if (!unit || typeof address == 'undefined' || !length) return; 249 | 250 | /* if client request an interval, 251 | * set a time interval and emit data 252 | * periodically. 253 | */ 254 | var interval = data.interval; 255 | if (interval) { 256 | var id = setInterval(function() { 257 | getInputStatus(unit, address, length); 258 | }, interval); 259 | 260 | intervalIDs.push(id); 261 | } else { 262 | getInputStatus(unit, address, length); 263 | } 264 | }); 265 | 266 | socket.on('readHoldingRegisters', function(data){ 267 | // check event validity 268 | if (!data) return; 269 | 270 | var unit = data.unit; 271 | var address = data.address; 272 | var length = data.length; 273 | 274 | // check event validity 275 | if (!unit || typeof address == 'undefined' || !length) return; 276 | 277 | /* if client request an interval, 278 | * set a time interval and emit data 279 | * periodically. 280 | */ 281 | var interval = data.interval; 282 | if (interval) { 283 | var id = setInterval(function() { 284 | getHoldingRegisters(unit, address, length); 285 | }, interval); 286 | 287 | intervalIDs.push(id); 288 | } else { 289 | getHoldingRegisters(unit, address, length); 290 | } 291 | }); 292 | 293 | socket.on('readInputRegisters', function(data){ 294 | // check event validity 295 | if (!data) return; 296 | 297 | var unit = data.unit; 298 | var address = data.address; 299 | var length = data.length; 300 | 301 | // check event validity 302 | if (!unit || typeof address == 'undefined' || !length) return; 303 | 304 | /* if client request an interval, 305 | * set a time interval and emit data 306 | * periodically. 307 | */ 308 | var interval = data.interval; 309 | if (interval) { 310 | var id = setInterval(function() { 311 | getInputRegisters(unit, address, length); 312 | }, interval); 313 | 314 | intervalIDs.push(id); 315 | } else { 316 | getInputRegisters(unit, address, length); 317 | } 318 | }); 319 | 320 | socket.on('writeCoil', function(data){ 321 | // check event validity 322 | if (!data) return; 323 | 324 | var unit = data.unit; 325 | var address = data.address; 326 | var state = data.state; 327 | 328 | // check event validity 329 | if (!unit || 330 | typeof address == 'undefined' || 331 | typeof state == 'undefined') return; 332 | 333 | forceCoil(unit, address, state); 334 | }); 335 | 336 | socket.on('writeRegisters', function(data){ 337 | // check event validity 338 | if (!data) return; 339 | 340 | var unit = data.unit; 341 | var address = data.address; 342 | var values = data.values; 343 | 344 | // check event validity 345 | if (!unit || typeof address == 'undefined' || !values) return; 346 | 347 | setRegisters(unit, address, values); 348 | }); 349 | }); 350 | }; 351 | 352 | /** 353 | * stop the modbus-ws server 354 | */ 355 | var stop = function() { 356 | serialPort.close(); 357 | process.exit(); 358 | } 359 | 360 | /** 361 | * Run a websocket only server 362 | * 363 | * @param {number} tcpPort the tcp port to listen on 364 | * @param {function} callback the function to call when done 365 | */ 366 | var run_wsd = function(tcpPort, callback) { 367 | // run ws server 368 | io = require('socket.io')(tcpPort, { 369 | allowEIO3: true // false by default 370 | }); 371 | 372 | /* Setup WebSocket event listener 373 | */ 374 | setup(); 375 | } 376 | 377 | /** 378 | * Run server with web application 379 | * 380 | * @param {number} tcpPort the tcp port to listen on 381 | * @param {function} callback the function to call when done 382 | */ 383 | var run_httpd = function(tcpPort, callback) { 384 | // run express application 385 | // with websockets 386 | var app = require('./app'); 387 | var http = require('http').Server(app); 388 | io = require('socket.io')(http, { 389 | allowEIO3: true // false by default 390 | }); 391 | /* Setup WebSocket event listener 392 | */ 393 | setup(); 394 | 395 | /* Setup http listener 396 | */ 397 | http.listen(tcpPort); 398 | } 399 | 400 | /** 401 | * start the modbus-ws server 402 | */ 403 | var start = function(options, callback) { 404 | /* set up some default options 405 | */ 406 | var title = "Modbus-WS server"; 407 | var tcpPort = options.tcpport || 3000; 408 | var port = options.serial || false; 409 | var baud = options.baudrate || 9600; 410 | var ip = options.ip || false; 411 | var noCache = options.nocache || false; 412 | var noHttp = options.nohttp || false; 413 | 414 | /* log server title and version 415 | */ 416 | console.log(); 417 | console.log('----------------------------------------------------'); 418 | console.log(title, version); 419 | 420 | /* open a serial port and setup modbus master 421 | */ 422 | modbusRTU = new ModbusRTU(); 423 | if (ip) { 424 | console.log(" Setup tcp/ip port:", ip); 425 | // open connection to a tcp line 426 | modbusRTU.connectTCP(ip, { port: 502 }); 427 | } else if (port) { 428 | console.log(" Setup serial port:", port, baud); 429 | modbusRTU.connectRTUBuffered(port, { baudRate: parseInt(baud) }); 430 | } else { 431 | throw new Error("Test mode is no longer supported in modbus-serial"); 432 | } 433 | 434 | /* set up express web application / only web socket server 435 | */ 436 | if (noHttp) { 437 | // run only web sockets server 438 | console.log(" Server is running, ws://127.0.0.1:" + tcpPort); 439 | 440 | run_wsd(tcpPort); 441 | } else { 442 | // run express application 443 | // with web sockets 444 | console.log(" Server is running, http://127.0.0.1:" + tcpPort); 445 | 446 | run_httpd(tcpPort); 447 | } 448 | 449 | /* set up caching 450 | */ 451 | if (noCache) { 452 | console.log(" Setup modbus without caching."); 453 | 454 | getCoils = _getCoils; 455 | getInputStatus = _getInputStatus; 456 | getHoldingRegisters = _getHoldingRegisters; 457 | getInputRegisters = _getInputRegisters; 458 | forceCoil = _forceCoil; 459 | setRegisters = _setRegisters; 460 | } else { 461 | var cache = require("./cache"); 462 | 463 | console.log(" Setup modbus with caching."); 464 | 465 | cache.run(io, modbusRTU, options); 466 | 467 | getCoils = cache.getCoils; 468 | getInputStatus = cache.getInputStatus; 469 | getHoldingRegisters = cache.getHoldingRegisters; 470 | getInputRegisters = cache.getInputRegisters; 471 | forceCoil = cache.forceCoil; 472 | setRegisters = cache.setRegisters; 473 | } 474 | 475 | console.log("----------------------------------------------------"); 476 | console.log(); 477 | 478 | // callback 479 | if (callback) callback(); 480 | } 481 | 482 | module.exports = {}; 483 | module.exports.stop = stop; 484 | module.exports.start = start; 485 | -------------------------------------------------------------------------------- /test/test.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | var io = require('socket.io-client') 4 | var server = require('../server'); 5 | 6 | var socketURL = 'ws://127.0.01:3000'; 7 | var options ={ 8 | transports: ['websocket'], 9 | 'force new connection': true 10 | }; 11 | 12 | var expect = require('chai').expect; 13 | 14 | describe('Modbus-WS server', function() { 15 | before(function(done) { 16 | var serverOptions = { 17 | 'tcpport': 3000, 18 | 'test': true, // use simulated port 19 | 'nohttp': true, // only websockets, no webapp 20 | 'nocache': false, // use cache 21 | 'pollinterval': 200, // short poll interval 22 | 'resendwait': 1 // send without waiting (wait 1 ms) 23 | }; 24 | server.start(serverOptions, done); 25 | }); 26 | 27 | it('Should get input registers', function(done) { 28 | var socket = io.connect(socketURL, options); 29 | 30 | socket.on('connect', function() { 31 | socket.emit('readInputRegisters', { 32 | "unit": 1, 33 | "address": 8, 34 | "length": 3 35 | }); 36 | }); 37 | 38 | socket.on('data', function(data){ 39 | expect(data).to.have.property('flag'); 40 | expect(data.flag).to.equal('get'); 41 | 42 | expect(data).to.have.property('data').with.length(3); 43 | expect(data.data.toString()).to.equal([8, 9, 10].toString()); 44 | 45 | socket.disconnect(); 46 | done() 47 | }); 48 | }); 49 | 50 | it('Should set holding registers', function(done) { 51 | var socket = io.connect(socketURL, options); 52 | 53 | socket.on('connect', function() { 54 | socket.emit('writeRegisters', { 55 | "unit": 1, 56 | "address": 8, 57 | "values": [88,123,47] 58 | }); 59 | }); 60 | 61 | socket.on('data', function(data){ 62 | expect(data).to.have.property('flag'); 63 | expect(data.flag).to.equal('set'); 64 | 65 | socket.disconnect(); 66 | done() 67 | }); 68 | }); 69 | 70 | it('Should get holding registers with the new values', function(done) { 71 | var socket = io.connect(socketURL, options); 72 | 73 | socket.on('connect', function() { 74 | socket.emit('readHoldingRegisters', { 75 | "unit": 1, 76 | "address": 8, 77 | "length": 3 78 | }); 79 | }); 80 | 81 | socket.on('data', function(data){ 82 | expect(data).to.have.property('flag'); 83 | expect(data.flag).to.equal('get'); 84 | 85 | expect(data).to.have.property('data').with.length(3); 86 | expect(data.data.toString()).to.equal([88,123,47].toString()); 87 | 88 | socket.disconnect(); 89 | done() 90 | }); 91 | }); 92 | 93 | it('Should force one coil', function(done) { 94 | var socket = io.connect(socketURL, options); 95 | 96 | socket.on('connect', function() { 97 | socket.emit('writeCoil', { 98 | "unit": 1, 99 | "address": 8, 100 | "state": true 101 | }); 102 | }); 103 | 104 | socket.on('data', function(data){ 105 | expect(data).to.have.property('data'); 106 | expect(data.data).to.equal(true); 107 | 108 | socket.disconnect(); 109 | done() 110 | }); 111 | }); 112 | 113 | it('Should get holding registers from cache', function(done) { 114 | var socket = io.connect(socketURL, options); 115 | 116 | socket.on('connect', function() { 117 | socket.emit('readHoldingRegisters', { 118 | "unit": 1, 119 | "address": 8, 120 | "length": 3 121 | }); 122 | }); 123 | 124 | socket.on('data', function(data){ 125 | expect(data).to.have.property('flag'); 126 | expect(data.flag).to.equal('get'); 127 | 128 | expect(data).to.have.property('data').with.length(3); 129 | expect(data.data.toString()).to.equal([88,123,47].toString()); 130 | 131 | socket.disconnect(); 132 | done() 133 | }); 134 | }); 135 | 136 | it('Should get input status', function(done) { 137 | var socket = io.connect(socketURL, options); 138 | 139 | socket.on('connect', function() { 140 | socket.emit('readDiscreteInputs', { 141 | "unit": 1, 142 | "address": 1, 143 | "length": 8 144 | }); 145 | }); 146 | 147 | socket.on('data', function(data){ 148 | expect(data).to.have.property('flag'); 149 | expect(data.type).to.equal(2); 150 | expect(data.flag).to.equal('get'); 151 | 152 | expect(data).to.have.property('data'); 153 | expect(data.data[7]).to.equal(true); 154 | 155 | socket.disconnect(); 156 | done() 157 | }); 158 | }); 159 | 160 | it('Should get coils after force one coil', function(done) { 161 | var socket = io.connect(socketURL, options); 162 | 163 | socket.on('connect', function() { 164 | socket.emit('readDiscreteInputs', { 165 | "unit": 1, 166 | "address": 8, 167 | "length": 8 168 | }); 169 | }); 170 | 171 | socket.on('data', function(data){ 172 | expect(data).to.have.property('flag'); 173 | expect(data.type).to.equal(2); 174 | expect(data.flag).to.equal('get'); 175 | 176 | expect(data).to.have.property('data'); 177 | expect(data.data[0]).to.equal(true); 178 | expect(data.data[7]).to.equal(false); 179 | 180 | socket.disconnect(); 181 | done() 182 | }); 183 | }); 184 | 185 | it('Should get input status from cache', function(done) { 186 | var socket = io.connect(socketURL, options); 187 | 188 | socket.on('connect', function() { 189 | socket.emit('readDiscreteInputs', { 190 | "unit": 1, 191 | "address": 1, 192 | "length": 8 193 | }); 194 | }); 195 | 196 | socket.on('data', function(data){ 197 | expect(data).to.have.property('flag'); 198 | expect(data.type).to.equal(2); 199 | expect(data.flag).to.equal('get'); 200 | 201 | expect(data).to.have.property('data'); 202 | expect(data.data[0]).to.equal(false); 203 | expect(data.data[7]).to.equal(true); 204 | 205 | socket.disconnect(); 206 | done() 207 | }); 208 | }); 209 | }); 210 | -------------------------------------------------------------------------------- /views/index.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | <%= title %> 5 | 6 | 7 | 8 |

<%= title %>

9 |

Welcome to <%= title %>

10 | 11 | 12 | --------------------------------------------------------------------------------