From 624dbdb1cea02ddf951fce39344ca55fbe9090d1 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 11 Jul 2023 10:46:15 +0200 Subject: [PATCH 01/32] add: pbkdf2 authentication --- src/login.js | 64 +++++++++++++++++++++++++++++++++++++++++++++++----- 1 file changed, 58 insertions(+), 6 deletions(-) diff --git a/src/login.js b/src/login.js index 2c68354..48bc122 100644 --- a/src/login.js +++ b/src/login.js @@ -14,24 +14,42 @@ fritzLogin.getSessionId = async (options) => { if (options.sid) return options.sid // Request a challenge for us to solve. - const response = await fritzRequest.request('/login_sid.lua', 'GET', options) + const response = await fritzRequest.request('/login_sid.lua?version=2', 'GET', options) + console.log(response) // Return the response error if one has presented itself. if (response.error) return response // Solve the challenge. const challenge = response.body.match('(.*?)')[1] - const buffer = Buffer.from(challenge + '-' + options.password, 'UTF-16LE') - const challengeAnswer = challenge + '-' + require('crypto').createHash('md5').update(buffer).digest('hex') + const challengeAnswer = await solveChallenge(challenge, options) // Send our answer to the Fritz!Box. - const path = '/login_sid.lua?username=' + options.username + '&response=' + challengeAnswer - const challengeResponse = await fritzRequest.request(path, 'GET', options) + const path = '/login_sid.lua?version=2&' + //username=' + options.username + '&response=' + challengeAnswer + const form = { + username: options.username, + response: challengeAnswer + } + + const challengeResponse = await fritzRequest.request(path, 'POST', options, true, form) + + fritzRequest.request(path, 'POST', options).then((response) => { + console.log(response) + }, (error) => { + console.log(error) + }) + + console.log("test " + JSON.stringify(challengeResponse)) if (challengeResponse.error) return challengeResponse // Extract the session ID. - const sessionId = challengeResponse.body.match('(.*?)')[1] + const sessionIdMatch = challengeResponse.body.match('(.*?)') + if (!sessionIdMatch || !sessionIdMatch[1]) { + return { error: { message: 'Failed to retrieve session ID from Fritz!Box.' } } + } + const sessionId = sessionIdMatch[1] // Determine if the login worked. if (sessionId === '0000000000000000') { @@ -41,6 +59,40 @@ fritzLogin.getSessionId = async (options) => { return sessionId } +async function solveChallenge (challenge, options) { + const challengeSplit = challenge.split('$') + if (challengeSplit[0] === '2') { + // pbkdf encryption FRITZ!OS 7.24 and later + const iter1 = parseInt(challengeSplit[1]) + const salt1 = parseHexToIntArray(challengeSplit[2]) + const iter2 = parseInt(challengeSplit[3]) + const salt2 = challengeSplit[4] + + let hash1 = crypto.pbkdf2Sync(options.password, salt1, iter1, 32, 'sha256') + let hash2 = crypto.pbkdf2Sync(hash1, parseHexToIntArray(salt2), iter2, 32, 'sha256') + + return `${salt2}$${hash2.toString('hex').trim()}` + } else { + // MD5 encryption + const buffer = Buffer.from(challenge + '-' + options.password, 'UTF-16LE') + return challenge + '-' + require('crypto').createHash('md5').update(buffer).digest('hex') + } +} + +function parseHexToIntArray (hexNumber) { + if (hexNumber.length % 2 !== 0) throw new Error('String has an invalid length for a hex string') + let intArray = [] + for (let iIndex = 0; iIndex < hexNumber.length; iIndex += 2) { + try { + intArray.push(parseInt(hexNumber.substr(iIndex, 2), 16)) + } catch (exception) { + throw new Error('Invalid hex string') + } + } + return new Uint8Array(intArray) +} + // <3 Circular dependencies... // https://stackoverflow.com/a/32428290/1878974 const fritzRequest = require('./request.js') +const crypto = require('crypto') From ba53b38d74d4a87502a68757dde5080a1e75a4c4 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Sat, 15 Jul 2023 02:27:05 +0200 Subject: [PATCH 02/32] add: reworked to node-fetch --- src/login.js | 35 ++++++++++++++++---------- src/request.js | 66 +++++++++++++++++++++++++++++++++++--------------- 2 files changed, 69 insertions(+), 32 deletions(-) diff --git a/src/login.js b/src/login.js index 48bc122..0c5cc87 100644 --- a/src/login.js +++ b/src/login.js @@ -2,7 +2,7 @@ /** @module fritzLogin */ let fritzLogin = {} -module.exports = fritzLogin +export default fritzLogin /** * Login to the Fritz!Box and obtain a sessionId. @@ -16,23 +16,32 @@ fritzLogin.getSessionId = async (options) => { // Request a challenge for us to solve. const response = await fritzRequest.request('/login_sid.lua?version=2', 'GET', options) - console.log(response) + console.log("response: ", response) // Return the response error if one has presented itself. if (response.error) return response // Solve the challenge. - const challenge = response.body.match('(.*?)')[1] + const challenge = response.match('(.*?)')[1] const challengeAnswer = await solveChallenge(challenge, options) // Send our answer to the Fritz!Box. - const path = '/login_sid.lua?version=2&' - //username=' + options.username + '&response=' + challengeAnswer - const form = { + const path = '/login_sid.lua?version=2' + const params = { username: options.username, response: challengeAnswer } + const formBody = [] + for (const property in params) { + const encodedKey = encodeURIComponent(property); + const encodedValue = encodeURIComponent(params[property]); + formBody.push(encodedKey + "=" + encodedValue); + } + + const headers = { + "Content-Type": "application/x-www-form-urlencoded", + } - const challengeResponse = await fritzRequest.request(path, 'POST', options, true, form) + const challengeResponse = await fritzRequest.request(path, 'POST', options, true, false, false, formBody.join("&"), headers) fritzRequest.request(path, 'POST', options).then((response) => { console.log(response) @@ -40,12 +49,12 @@ fritzLogin.getSessionId = async (options) => { console.log(error) }) - console.log("test " + JSON.stringify(challengeResponse)) - + + console.log("test ", challengeResponse) if (challengeResponse.error) return challengeResponse - + // Extract the session ID. - const sessionIdMatch = challengeResponse.body.match('(.*?)') + const sessionIdMatch = challengeResponse.match('(.*?)') if (!sessionIdMatch || !sessionIdMatch[1]) { return { error: { message: 'Failed to retrieve session ID from Fritz!Box.' } } } @@ -94,5 +103,5 @@ function parseHexToIntArray (hexNumber) { // <3 Circular dependencies... // https://stackoverflow.com/a/32428290/1878974 -const fritzRequest = require('./request.js') -const crypto = require('crypto') +import fritzRequest from './request.js' +import crypto from 'crypto' diff --git a/src/request.js b/src/request.js index 06cbb45..f78e62f 100644 --- a/src/request.js +++ b/src/request.js @@ -3,12 +3,19 @@ * @ignore */ -const request = require('request-promise') -const requestNoPromise = require('request') -const fs = require('fs') - +import fetch from 'node-fetch'; +import request from 'request-promise' +import requestNoPromise from 'request' +import fs from 'fs' +import https from'https'; + +const httpsAgent = new https.Agent({ + //Disable SSL verification since the Fritz.box endpoint is not secure enough + //The certificate uses a weak Diffie-Hellman key and is vulnerable to a Logjam Attack + rejectUnauthorized: false, + }); let fritzRequest = {} -module.exports = fritzRequest +export default fritzRequest /** * Send a request to the Fritz!Box. @@ -22,7 +29,8 @@ module.exports = fritzRequest * @param {boolean} formUrlEncoded * @return {Object} Request response object */ -fritzRequest.request = async (path, method, options, pipe = false, formData = false, formUrlEncoded = false) => { +fritzRequest.request = async (path, method, options, pipe = false, formData = false, formUrlEncoded = false, params, headers) => { + console.log("PATH: " + path) options.protocol = options.protocol || 'https' // Make sure we have the required options. @@ -41,10 +49,25 @@ fritzRequest.request = async (path, method, options, pipe = false, formData = fa if (typeof options.removeSidFromUri === 'undefined') { options.removeSidFromUri = false } + if (!params) { + params = {}; + } // Add SID to path if one has been given to us. if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { - path += '&sid=' + options.sid + params['sid'] = options.sid + } + + let body; + if (method === 'POST') { + body = params + } else { + for (const key in params) { + if (!path.endsWith('&') || !path.endsWith('?')) { + path += '&' + } + path += `${encodeURIComponent(key)}=${encodeURIComponent(params[key])}` + }; } // Set the options for the request. @@ -64,7 +87,7 @@ fritzRequest.request = async (path, method, options, pipe = false, formData = fa } // Pipe a file to disk. - if (pipe) { + if (false) { let stream = requestNoPromise(requestOptions).pipe(fs.createWriteStream(pipe)) stream.on('finish', () => { return { message: 'File has been saved to ' + pipe } @@ -72,21 +95,25 @@ fritzRequest.request = async (path, method, options, pipe = false, formData = fa } // Execute HTTP(S) request. - let response = null try { - response = await request(requestOptions) + // response = await request(requestOptions) + console.log(requestOptions.uri) + console.log(method) + console.log(body) + + const response = await fetch(requestOptions.uri, { + headers: headers, + method: method || 'GET', + body: body, + agent: httpsAgent + }) + const xmlResponse = await response.text(); + + return xmlResponse } catch (error) { return fritzRequest.findFailCause(error) } - /* - if (response.statusCode !== 200) { - console.log('is not 200') - return fritzRequest.findFailCause(response) - } - */ - - return response } /** @@ -116,4 +143,5 @@ fritzRequest.findFailCause = (response) => { // <3 Circular dependencies... // https://stackoverflow.com/a/32428290/1878974 -const fritzLogin = require('./login.js') +import fritzLogin from './login.js' + From 29f92f7bdc05552bc12b171e93bbe42f1be8e627 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Sat, 15 Jul 2023 02:28:26 +0200 Subject: [PATCH 03/32] update: node modules naming convention --- index.js | 19 +++++++++---------- src/dect.js | 12 +++++++----- src/fon.js | 14 +++++++------- src/format.js | 6 +++--- src/system.js | 7 +++---- src/wlan.js | 2 ++ test/login.js | 6 +++--- test/monitor.js | 6 +++--- test/smartdevices.js | 6 +++--- 9 files changed, 40 insertions(+), 38 deletions(-) diff --git a/index.js b/index.js index 8ed5fd1..227f3ea 100644 --- a/index.js +++ b/index.js @@ -6,18 +6,17 @@ * Source-code available on GitHub. * https://git.io/fritzbox */ - +import pckJson from './package.json' assert { type: 'json'} const fritzConfig = { - version: require('./package.json').version, - debug: require('./package.json').options.debug + version: pckJson.version, + debug: pckJson.options.debug } -const dir = 'src' -const fritzLogin = require('./' + dir + '/login.js') -const fritzSystem = require('./' + dir + '/system.js') -const fritzFon = require('./' + dir + '/fon.js') -const fritzDect = require('./' + dir + '/dect.js') -const fritzWlan = require('./' + dir + '/wlan.js') +import fritzLogin from './src/login.js' +import fritzSystem from './src/system.js' +import fritzFon from './src/fon.js' +import fritzDect from './src/dect.js' +import fritzWlan from './src/wlan.js' const fritz = Object.assign( {}, fritzConfig, fritzSystem, fritzLogin, @@ -33,4 +32,4 @@ process.on('unhandledRejection', function (reason, r) { process.exit(1) }) -module.exports = fritz +export default fritz diff --git a/src/dect.js b/src/dect.js index c97ef76..1e948a7 100644 --- a/src/dect.js +++ b/src/dect.js @@ -5,9 +5,9 @@ let fritzDect = {} -const fritzLogin = require('./login.js') -const fritzRequest = require('./request.js') -const fritzSystem = require('./system.js') +import fritzLogin from './login.js' +import fritzRequest from './request.js' +import fritzSystem from './system.js' /** * Get all smart devices and groups. @@ -21,7 +21,9 @@ fritzDect.getSmartDevices = async (options) => { if (response.error) return response - return JSON.parse(response.body).devices + console.log(response) + + return JSON.parse(response).devices } /** @@ -76,4 +78,4 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { // Export fritzDect. -module.exports = fritzDect +export default fritzDect diff --git a/src/fon.js b/src/fon.js index f6104bf..1cefd95 100644 --- a/src/fon.js +++ b/src/fon.js @@ -5,13 +5,13 @@ let fritzFon = {} -const fritzLogin = require('./login.js') -const fritzRequest = require('./request.js') -const fritzFormat = require('./format.js') -const fritzSystem = require('./system.js') +import fritzLogin from './login.js' +import fritzRequest from './request.js' +import fritzFormat from './format.js' +import fritzSystem from './system.js' -const net = require('net') -const events = require('events') +import net from 'net' +import events from 'events' /** * Get the history of telephone calls. @@ -350,4 +350,4 @@ fritzFon.CallMonitor = CallMonitor // Export fritzFon. -module.exports = fritzFon +export default fritzFon diff --git a/src/format.js b/src/format.js index f55e616..3c08b45 100644 --- a/src/format.js +++ b/src/format.js @@ -5,8 +5,8 @@ let fritzFormat = {} -const csvjson = require('csvjson') -const convert = require('xml-to-json-promise') +import csvjson from 'csvjson' +import convert from 'xml-to-json-promise' /** * Format a raw calls array to a more readable array. @@ -186,4 +186,4 @@ fritzFormat.phonebook = (phonebook) => { // Export fritzFon. -module.exports = fritzFormat +export default fritzFormat diff --git a/src/system.js b/src/system.js index f8608ab..5df0fc1 100644 --- a/src/system.js +++ b/src/system.js @@ -2,11 +2,10 @@ /** @module fritzSystem */ let fritzSystem = {} -module.exports = fritzSystem - -const fritzRequest = require('./request.js') -const fritzFormat = require('./format.js') +export default fritzSystem +import fritzRequest from './request.js' +import fritzFormat from './format.js' /** * Get the version of a Fritz!Box without authentication. * @param {Object} options - FritzBox.js options object. diff --git a/src/wlan.js b/src/wlan.js index 4706084..1af1ede 100644 --- a/src/wlan.js +++ b/src/wlan.js @@ -1,3 +1,5 @@ /** * @module fritzWlan */ +const fritzWlan = {} +export default fritzWlan \ No newline at end of file diff --git a/test/login.js b/test/login.js index a8d9cab..7e64bb5 100644 --- a/test/login.js +++ b/test/login.js @@ -1,8 +1,8 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert { type: 'json'} async function login () { - const sessionId = await fritz.getSessionId(options) + const sessionId = await fritz.getSessionId(pckJson.options) if (sessionId.error) { console.log('Error:', sessionId.error.message) process.exit(1) diff --git a/test/monitor.js b/test/monitor.js index 4f1a740..be41041 100644 --- a/test/monitor.js +++ b/test/monitor.js @@ -1,7 +1,7 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type: 'json'} -let monitor = new fritz.CallMonitor(options) +let monitor = new fritz.CallMonitor(pckJson.options) function handler (type, call) { console.log(type, '\n', call) diff --git a/test/smartdevices.js b/test/smartdevices.js index 206b13c..5b4a60f 100644 --- a/test/smartdevices.js +++ b/test/smartdevices.js @@ -1,8 +1,8 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} async function devices () { - const devices = await fritz.getSmartDevices(options) + const devices = await fritz.getSmartDevices(pckJson.options) if (devices.error) { console.log('Error:', devices.error.message) process.exit(1) From aa6f0a618af98e9a363b28a5f472cbd7d4458d80 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Mon, 17 Jul 2023 23:41:59 +0200 Subject: [PATCH 04/32] remove: console logs --- src/login.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/login.js b/src/login.js index 0c5cc87..01d3cc8 100644 --- a/src/login.js +++ b/src/login.js @@ -49,8 +49,6 @@ fritzLogin.getSessionId = async (options) => { console.log(error) }) - - console.log("test ", challengeResponse) if (challengeResponse.error) return challengeResponse // Extract the session ID. From 8e430e269cb53a48fbef67179d1f0cc0bc294e1d Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Mon, 17 Jul 2023 23:59:29 +0200 Subject: [PATCH 05/32] refactor: param interface --- src/request.js | 45 +++++++++++++-------------------------------- 1 file changed, 13 insertions(+), 32 deletions(-) diff --git a/src/request.js b/src/request.js index f78e62f..225595f 100644 --- a/src/request.js +++ b/src/request.js @@ -21,15 +21,15 @@ export default fritzRequest * Send a request to the Fritz!Box. * * @private - * @param {string} path Path to request - * @param {string} method Request method - * @param {Object} options Options object - * @param {string} pipe - * @param {Object} formData - * @param {boolean} formUrlEncoded + * @param {string} path Path to request + * @param {string} method Request method + * @param {Object} options Options object + * @param {HeadersInit} headers + * @param {Object} params + * @param {string} pipe * @return {Object} Request response object */ -fritzRequest.request = async (path, method, options, pipe = false, formData = false, formUrlEncoded = false, params, headers) => { +fritzRequest.request = async (path, method, options, headers, params, pipe = false) => { console.log("PATH: " + path) options.protocol = options.protocol || 'https' @@ -61,33 +61,20 @@ fritzRequest.request = async (path, method, options, pipe = false, formData = fa let body; if (method === 'POST') { body = params - } else { + } else if (method === 'GET'){ for (const key in params) { if (!path.endsWith('&') || !path.endsWith('?')) { path += '&' } - path += `${encodeURIComponent(key)}=${encodeURIComponent(params[key])}` + path += `${key}=${params[key]}` }; } // Set the options for the request. - let requestOptions = { - uri: options.protocol + '://' + options.server + path, - method: method || 'GET', - resolveWithFullResponse: true, - rejectUnauthorized: false - } - - if (formData) { - requestOptions.formData = formData - } - - if (formUrlEncoded) { - requestOptions.form = formUrlEncoded - } + const uri = options.protocol + '://' + options.server + path // Pipe a file to disk. - if (false) { + if (pipe) { let stream = requestNoPromise(requestOptions).pipe(fs.createWriteStream(pipe)) stream.on('finish', () => { return { message: 'File has been saved to ' + pipe } @@ -96,20 +83,14 @@ fritzRequest.request = async (path, method, options, pipe = false, formData = fa // Execute HTTP(S) request. try { - // response = await request(requestOptions) - console.log(requestOptions.uri) - console.log(method) - console.log(body) - - const response = await fetch(requestOptions.uri, { + const response = await fetch(uri, { headers: headers, method: method || 'GET', body: body, agent: httpsAgent }) - const xmlResponse = await response.text(); - return xmlResponse + return await response.text() } catch (error) { return fritzRequest.findFailCause(error) } From efaa6053ba9ce459de95ffa0ea05a68d450282f4 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 18 Jul 2023 00:05:59 +0200 Subject: [PATCH 06/32] removed: unused pipe --- src/request.js | 10 +--------- 1 file changed, 1 insertion(+), 9 deletions(-) diff --git a/src/request.js b/src/request.js index 225595f..d9b3bcc 100644 --- a/src/request.js +++ b/src/request.js @@ -29,7 +29,7 @@ export default fritzRequest * @param {string} pipe * @return {Object} Request response object */ -fritzRequest.request = async (path, method, options, headers, params, pipe = false) => { +fritzRequest.request = async (path, method, options, headers, params) => { console.log("PATH: " + path) options.protocol = options.protocol || 'https' @@ -73,14 +73,6 @@ fritzRequest.request = async (path, method, options, headers, params, pipe = fal // Set the options for the request. const uri = options.protocol + '://' + options.server + path - // Pipe a file to disk. - if (pipe) { - let stream = requestNoPromise(requestOptions).pipe(fs.createWriteStream(pipe)) - stream.on('finish', () => { - return { message: 'File has been saved to ' + pipe } - }) - } - // Execute HTTP(S) request. try { const response = await fetch(uri, { From 60d829ec63dba527baabe3390c2ad677b916a13c Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 18 Jul 2023 00:24:00 +0200 Subject: [PATCH 07/32] removed: dupplicate http request call --- src/login.js | 8 +------- 1 file changed, 1 insertion(+), 7 deletions(-) diff --git a/src/login.js b/src/login.js index 01d3cc8..8ed20b6 100644 --- a/src/login.js +++ b/src/login.js @@ -41,13 +41,7 @@ fritzLogin.getSessionId = async (options) => { "Content-Type": "application/x-www-form-urlencoded", } - const challengeResponse = await fritzRequest.request(path, 'POST', options, true, false, false, formBody.join("&"), headers) - - fritzRequest.request(path, 'POST', options).then((response) => { - console.log(response) - }, (error) => { - console.log(error) - }) + const challengeResponse = await fritzRequest.request(path, 'POST', options, formBody.join("&"), headers) if (challengeResponse.error) return challengeResponse From 83fea9f2d0fba3aab6f1fb5de912d65d4dcbbe94 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 18 Jul 2023 00:25:36 +0200 Subject: [PATCH 08/32] refactor: params order --- src/request.js | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/src/request.js b/src/request.js index d9b3bcc..a5d850d 100644 --- a/src/request.js +++ b/src/request.js @@ -4,9 +4,6 @@ */ import fetch from 'node-fetch'; -import request from 'request-promise' -import requestNoPromise from 'request' -import fs from 'fs' import https from'https'; const httpsAgent = new https.Agent({ @@ -29,7 +26,7 @@ export default fritzRequest * @param {string} pipe * @return {Object} Request response object */ -fritzRequest.request = async (path, method, options, headers, params) => { +fritzRequest.request = async (path, method, options, params, headers) => { console.log("PATH: " + path) options.protocol = options.protocol || 'https' From 5c7a90a15ae1db1bc8e1b24b1540485effb2e936 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 25 Jul 2023 16:27:45 +0200 Subject: [PATCH 09/32] refactor activecalls.js --- src/dect.js | 2 +- src/fon.js | 27 +++++++++++++++++++++------ src/login.js | 18 +++++++++--------- src/request.js | 21 +++++++++++++-------- test/activecalls.js | 6 +++--- test/calls.js | 8 ++++---- 6 files changed, 51 insertions(+), 31 deletions(-) diff --git a/src/dect.js b/src/dect.js index 1e948a7..7f7af7e 100644 --- a/src/dect.js +++ b/src/dect.js @@ -55,7 +55,7 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { cmdValue: value, deviceId: deviceId } - response = await fritzRequest.request(path, 'POST', options, false, false, form) + response = await fritzRequest.request(path, 'POST', options, form) } else { // Pre 06.83 used a GET request. const path = '/myfritz/areas/homeauto.lua?ajax_id=' + diff --git a/src/fon.js b/src/fon.js index 1cefd95..f50af88 100644 --- a/src/fon.js +++ b/src/fon.js @@ -20,12 +20,12 @@ import events from 'events' */ fritzFon.getCalls = async (options) => { const path = '/fon_num/foncalls_list.lua?csv=' - const response = await fritzRequest.request(path, 'GET', options) + const response = fritzRequest.request(path, 'GET', options) if (response.error) return response const csvCalls = response.body - const jsonCalls = await fritzFormat.callsCsvToJson(csvCalls) + const jsonCalls = fritzFormat.callsCsvToJson(csvCalls) const formattedCalls = fritzFormat.calls(jsonCalls) @@ -99,7 +99,8 @@ fritzFon.downloadTamMessage = async (messagePath, localPath, options) => { const path = '/myfritz/cgi-bin/luacgi_notimeout' + '?cmd=tam&script=/http_file_download.lua' + '&cmd_files=' + messagePath - const response = await fritzRequest.request(path, 'GET', options, localPath) + + const response = await fritzRequest.request(path, 'GET', options) if (response.error) return response @@ -167,11 +168,25 @@ fritzFon.getActiveCalls = async (options) => { } options.removeSidFromUri = true - const form = { + const params = { page: 'overview', - sid: options.sid + sid: options.sid, + xhr: '1' + } + + const formBody = [] + for (const property in params) { + const encodedKey = encodeURIComponent(property); + const encodedValue = encodeURIComponent(params[property]); + formBody.push(`${encodedKey}=${encodedValue}`); } - const response = await fritzRequest.request('/data.lua', 'POST', options, false, false, form) + + const headers = { + "Content-Type": "application/x-www-form-urlencoded", + } + + const response = await fritzRequest.request('/data.lua', 'POST', options, formBody.join("&"), headers) + if (response.error) return response const activeCalls = JSON.parse(response.body).data.foncalls.activecalls diff --git a/src/login.js b/src/login.js index 8ed20b6..c9fc639 100644 --- a/src/login.js +++ b/src/login.js @@ -16,12 +16,11 @@ fritzLogin.getSessionId = async (options) => { // Request a challenge for us to solve. const response = await fritzRequest.request('/login_sid.lua?version=2', 'GET', options) - console.log("response: ", response) // Return the response error if one has presented itself. if (response.error) return response // Solve the challenge. - const challenge = response.match('(.*?)')[1] + const challenge = response.body.match('(.*?)')[1] const challengeAnswer = await solveChallenge(challenge, options) // Send our answer to the Fritz!Box. @@ -41,12 +40,12 @@ fritzLogin.getSessionId = async (options) => { "Content-Type": "application/x-www-form-urlencoded", } - const challengeResponse = await fritzRequest.request(path, 'POST', options, formBody.join("&"), headers) + const responseChallenge = await fritzRequest.request(path, 'POST', options, formBody.join("&"), headers) - if (challengeResponse.error) return challengeResponse + if (responseChallenge.error) return responseChallenge // Extract the session ID. - const sessionIdMatch = challengeResponse.match('(.*?)') + const sessionIdMatch = responseChallenge.body.match('(.*?)') if (!sessionIdMatch || !sessionIdMatch[1]) { return { error: { message: 'Failed to retrieve session ID from Fritz!Box.' } } } @@ -69,14 +68,14 @@ async function solveChallenge (challenge, options) { const iter2 = parseInt(challengeSplit[3]) const salt2 = challengeSplit[4] - let hash1 = crypto.pbkdf2Sync(options.password, salt1, iter1, 32, 'sha256') - let hash2 = crypto.pbkdf2Sync(hash1, parseHexToIntArray(salt2), iter2, 32, 'sha256') + let hash1 = Crypto.pbkdf2Sync(options.password, salt1, iter1, 32, 'sha256') + let hash2 = Crypto.pbkdf2Sync(hash1, parseHexToIntArray(salt2), iter2, 32, 'sha256') return `${salt2}$${hash2.toString('hex').trim()}` } else { // MD5 encryption const buffer = Buffer.from(challenge + '-' + options.password, 'UTF-16LE') - return challenge + '-' + require('crypto').createHash('md5').update(buffer).digest('hex') + return challenge + '-' + Crypto.createHash('md5').update(buffer).digest('hex') } } @@ -96,4 +95,5 @@ function parseHexToIntArray (hexNumber) { // <3 Circular dependencies... // https://stackoverflow.com/a/32428290/1878974 import fritzRequest from './request.js' -import crypto from 'crypto' +import Crypto from 'crypto' +import Buffer from 'node:buffer' diff --git a/src/request.js b/src/request.js index a5d850d..0c133ab 100644 --- a/src/request.js +++ b/src/request.js @@ -4,7 +4,7 @@ */ import fetch from 'node-fetch'; -import https from'https'; +import https from 'https'; const httpsAgent = new https.Agent({ //Disable SSL verification since the Fritz.box endpoint is not secure enough @@ -13,7 +13,6 @@ const httpsAgent = new https.Agent({ }); let fritzRequest = {} export default fritzRequest - /** * Send a request to the Fritz!Box. * @@ -26,7 +25,7 @@ export default fritzRequest * @param {string} pipe * @return {Object} Request response object */ -fritzRequest.request = async (path, method, options, params, headers) => { +fritzRequest.request = async (path, method, options, params = false, headers = false) => { console.log("PATH: " + path) options.protocol = options.protocol || 'https' @@ -38,7 +37,7 @@ fritzRequest.request = async (path, method, options, params, headers) => { // Obtain a session id if none was given to us. if (!options.sid && !path.includes('/login_sid.lua') && options.noAuth !== true) { - const sessionId = await fritzLogin.getSessionId(options) + const sessionId = fritzLogin.getSessionId(options) if (sessionId.error) return sessionId options.sid = sessionId } @@ -49,6 +48,9 @@ fritzRequest.request = async (path, method, options, params, headers) => { if (!params) { params = {}; } + if (!headers) { + headers = {}; + } // Add SID to path if one has been given to us. if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { @@ -64,7 +66,7 @@ fritzRequest.request = async (path, method, options, params, headers) => { path += '&' } path += `${key}=${params[key]}` - }; + } } // Set the options for the request. @@ -72,14 +74,17 @@ fritzRequest.request = async (path, method, options, params, headers) => { // Execute HTTP(S) request. try { - const response = await fetch(uri, { + const fetchResponse = await fetch(uri, { headers: headers, method: method || 'GET', body: body, agent: httpsAgent }) - - return await response.text() + const response = { + body: await fetchResponse.text() + } + console.log('response fetch', response) + return response } catch (error) { return fritzRequest.findFailCause(error) } diff --git a/test/activecalls.js b/test/activecalls.js index fb8f59d..6115f8e 100644 --- a/test/activecalls.js +++ b/test/activecalls.js @@ -1,8 +1,8 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} async function activecalls () { - const calls = await fritz.getActiveCalls(options) + const calls = await fritz.getActiveCalls(pckJson.options) if (calls.error) { console.log('Error:', calls.error.message) diff --git a/test/calls.js b/test/calls.js index 55e61cf..d17480c 100644 --- a/test/calls.js +++ b/test/calls.js @@ -1,9 +1,9 @@ -const fritz = require('../index.js') -const options = require('../package.json').options -const fs = require('fs') +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} +import fs from 'fs' async function calls () { - const calls = await fritz.getCalls(options) + const calls = await fritz.getCalls(pckJson.options) if (calls.error) { console.log('Error:', calls.error.message) process.exit(1) From fa70f08ab66f3fc68ff0f74c493940ca196c5ab7 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 25 Jul 2023 17:35:57 +0200 Subject: [PATCH 10/32] fix calls.js --- src/fon.js | 11 ++--------- src/login.js | 8 +------- src/request.js | 45 ++++++++++++++++++++++++--------------------- 3 files changed, 27 insertions(+), 37 deletions(-) diff --git a/src/fon.js b/src/fon.js index f50af88..55fd65e 100644 --- a/src/fon.js +++ b/src/fon.js @@ -20,7 +20,7 @@ import events from 'events' */ fritzFon.getCalls = async (options) => { const path = '/fon_num/foncalls_list.lua?csv=' - const response = fritzRequest.request(path, 'GET', options) + const response = await fritzRequest.request(path, 'GET', options) if (response.error) return response @@ -174,18 +174,11 @@ fritzFon.getActiveCalls = async (options) => { xhr: '1' } - const formBody = [] - for (const property in params) { - const encodedKey = encodeURIComponent(property); - const encodedValue = encodeURIComponent(params[property]); - formBody.push(`${encodedKey}=${encodedValue}`); - } - const headers = { "Content-Type": "application/x-www-form-urlencoded", } - const response = await fritzRequest.request('/data.lua', 'POST', options, formBody.join("&"), headers) + const response = await fritzRequest.request('/data.lua', 'POST', options, params, headers) if (response.error) return response diff --git a/src/login.js b/src/login.js index c9fc639..2914fcb 100644 --- a/src/login.js +++ b/src/login.js @@ -29,18 +29,12 @@ fritzLogin.getSessionId = async (options) => { username: options.username, response: challengeAnswer } - const formBody = [] - for (const property in params) { - const encodedKey = encodeURIComponent(property); - const encodedValue = encodeURIComponent(params[property]); - formBody.push(encodedKey + "=" + encodedValue); - } const headers = { "Content-Type": "application/x-www-form-urlencoded", } - const responseChallenge = await fritzRequest.request(path, 'POST', options, formBody.join("&"), headers) + const responseChallenge = await fritzRequest.request(path, 'POST', options, params, headers) if (responseChallenge.error) return responseChallenge diff --git a/src/request.js b/src/request.js index 0c133ab..5fa8bff 100644 --- a/src/request.js +++ b/src/request.js @@ -21,11 +21,11 @@ export default fritzRequest * @param {string} method Request method * @param {Object} options Options object * @param {HeadersInit} headers - * @param {Object} params + * @param {Object} body * @param {string} pipe * @return {Object} Request response object */ -fritzRequest.request = async (path, method, options, params = false, headers = false) => { +fritzRequest.request = async (path, method, options, body = {}, headers = {}) => { console.log("PATH: " + path) options.protocol = options.protocol || 'https' @@ -37,7 +37,7 @@ fritzRequest.request = async (path, method, options, params = false, headers = f // Obtain a session id if none was given to us. if (!options.sid && !path.includes('/login_sid.lua') && options.noAuth !== true) { - const sessionId = fritzLogin.getSessionId(options) + const sessionId = await fritzLogin.getSessionId(options) if (sessionId.error) return sessionId options.sid = sessionId } @@ -45,33 +45,36 @@ fritzRequest.request = async (path, method, options, params = false, headers = f if (typeof options.removeSidFromUri === 'undefined') { options.removeSidFromUri = false } - if (!params) { - params = {}; - } - if (!headers) { - headers = {}; - } - - // Add SID to path if one has been given to us. - if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { - params['sid'] = options.sid + if (headers['Content-Type'] === 'application/x-www-form-urlencoded') { + const formBody = [] + for (const property in body) { + const encodedKey = encodeURIComponent(property); + const encodedValue = encodeURIComponent(body[property]); + formBody.push(encodedKey + "=" + encodedValue); + } + body = formBody.join("&") + } else if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { + // Add SID to path if one has been given to us. + body['sid'] = options.sid } - let body; if (method === 'POST') { - body = params + } else if (method === 'GET'){ - for (const key in params) { - if (!path.endsWith('&') || !path.endsWith('?')) { - path += '&' - } - path += `${key}=${params[key]}` + for (const key in body) { + if (!path.endsWith('&') || !path.endsWith('?')) { + path += '&' } + path += `${key}=${body[key]}` + } + body = undefined; } // Set the options for the request. const uri = options.protocol + '://' + options.server + path - + console.log('headers: ', headers) + console.log('method: ', method) + console.log('body: ', body) // Execute HTTP(S) request. try { const fetchResponse = await fetch(uri, { From 9d1ef743888a304ed78e087a3c5b15ddfbcdadab Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 25 Jul 2023 17:45:43 +0200 Subject: [PATCH 11/32] refactor dial.js --- src/fon.js | 9 ++++++--- test/dial.js | 6 +++--- 2 files changed, 9 insertions(+), 6 deletions(-) diff --git a/src/fon.js b/src/fon.js index 55fd65e..7e85802 100644 --- a/src/fon.js +++ b/src/fon.js @@ -143,9 +143,12 @@ fritzFon.dialNumber = async (phoneNumber, options) => { return { error: { message: 'Invalid phone number given.' } } } - const path = '/fon_num/foncalls_list.lua?xhr=1' + - '&dial=' + phoneNumber - const response = await fritzRequest.request(path, 'GET', options) + const path = '/fon_num/foncalls_list.lua?' + const param = { + xhr: 1, + dial: phoneNumber + } + const response = await fritzRequest.request(path, 'GET', options, param) if (response.error) return response diff --git a/test/dial.js b/test/dial.js index 9b5d6f8..9fc7752 100644 --- a/test/dial.js +++ b/test/dial.js @@ -1,10 +1,10 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} const number = process.argv[2] async function dial () { - const status = await fritz.dialNumber(number, options) + const status = await fritz.dialNumber(number, pckJson.options) if (status.error) { console.log('Error: ' + status.error.message) process.exit(1) From 4fa23d89f57c410623e6aef21e74289663d5bf07 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 25 Jul 2023 17:52:48 +0200 Subject: [PATCH 12/32] refactor markread.js --- src/fon.js | 11 +++++++---- test/markread.js | 6 +++--- 2 files changed, 10 insertions(+), 7 deletions(-) diff --git a/src/fon.js b/src/fon.js index 7e85802..f577548 100644 --- a/src/fon.js +++ b/src/fon.js @@ -119,10 +119,13 @@ fritzFon.downloadTamMessage = async (messagePath, localPath, options) => { * @return {boolean} Returns true when the message was marked as read. */ fritzFon.markTamMessageAsRead = async (messageId, options, tamId = 0) => { - const path = '/fon_devices/tam_list.lua?useajax=1' + - '&TamNr=' + tamId + - '&idx=' + messageId - const response = await fritzRequest.request(path, 'GET', options) + const path = '/fon_devices/tam_list.lua?' + const param = { + useajax: 1, + TamNr: tamId, + idx: messageId + } + const response = await fritzRequest.request(path, 'GET', options, param) if (response.error) return response diff --git a/test/markread.js b/test/markread.js index 1ff772f..03d4a9e 100644 --- a/test/markread.js +++ b/test/markread.js @@ -1,9 +1,9 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} const messageId = 20 async function markasread () { - const status = await fritz.markTamMessageAsRead(messageId, options) + const status = await fritz.markTamMessageAsRead(messageId, pckJson.options) if (status.error) { console.log('Error:', status.error.message) process.exit(1) From e0f8bcd650ec90f08883117a87705bb8418ee48e Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 25 Jul 2023 18:41:38 +0200 Subject: [PATCH 13/32] added support for Fritzbox!7.50 tam.js --- src/fon.js | 35 +++++++++++++++++++++++++++++++++-- test/tam.js | 8 ++++---- 2 files changed, 37 insertions(+), 6 deletions(-) diff --git a/src/fon.js b/src/fon.js index f577548..306e7f7 100644 --- a/src/fon.js +++ b/src/fon.js @@ -43,8 +43,36 @@ fritzFon.getTamMessages = async (options) => { if (version.error) return version let tamMessages + if (version >= 750) { + /* The following works with Fritz!Box OS 7.50 and newer. */ + if (!options.sid) { + options.sid = await fritzLogin.getSessionId(options) + if (options.sid.error) return options.sid + } + + // Prepare request options + options.removeSidFromUri = true + const path = '/data.lua?' + const param = { + sid: options.sid, + page: 'tam', + xhr: 1 + } - if (version >= 683) { + const response = await fritzRequest.request(path, 'GET', options, param) + if (response.error) return response + + let calls = JSON.parse(response.body).data.foncalls.calls + console.log(calls) + // Filter only TAM messages. + tamMessages = [] + for (var call in calls) { + if (calls[call].type === 'tam') { + tamMessages.push(calls[call]) + } + } + + } else if (version >= 683) { /* The following works with Fritz!Box OS 6.83 and newer. */ // Get a session ID for the POST request first. @@ -60,8 +88,11 @@ fritzFon.getTamMessages = async (options) => { sid: options.sid, ajax_id: 1234 } + const headers = { + "Content-Type": "application/x-www-form-urlencoded", + } - const response = await fritzRequest.request(path, 'POST', options, false, false, form) + const response = await fritzRequest.request(path, 'POST', options, form, headers) if (response.error) return response let calls = JSON.parse(response.body).calls diff --git a/test/tam.js b/test/tam.js index 757b573..161303e 100644 --- a/test/tam.js +++ b/test/tam.js @@ -1,9 +1,9 @@ -const fritz = require('../index.js') -const options = require('../package.json').options -const fs = require('fs') +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} +import fs from 'fs' async function tam () { - const messages = await fritz.getTamMessages(options) + const messages = await fritz.getTamMessages(pckJson.options) if (messages.error) { console.log('Error:', messages.error.message) From 9152d4cf079fc3a068e8cd375d80e4034cf7e506 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 28 Jul 2023 16:49:56 +0200 Subject: [PATCH 14/32] fix phonebook.js & tamdownload.js --- src/fon.js | 53 +++++++++++++++++++++++++++++++++++---------- src/request.js | 20 +++++++++++------ test/phonebook.js | 6 ++--- test/tamdownload.js | 8 +++---- 4 files changed, 61 insertions(+), 26 deletions(-) diff --git a/src/fon.js b/src/fon.js index 306e7f7..2b31f49 100644 --- a/src/fon.js +++ b/src/fon.js @@ -62,13 +62,13 @@ fritzFon.getTamMessages = async (options) => { const response = await fritzRequest.request(path, 'GET', options, param) if (response.error) return response - let calls = JSON.parse(response.body).data.foncalls.calls + let calls = JSON.parse(response.body).data.foncalls console.log(calls) // Filter only TAM messages. tamMessages = [] for (var call in calls) { if (calls[call].type === 'tam') { - tamMessages.push(calls[call]) + tamMessages.push(calls[call].tam_data) } } @@ -127,20 +127,44 @@ fritzFon.getTamMessages = async (options) => { * @return {Object} Returns an object with a message. */ fritzFon.downloadTamMessage = async (messagePath, localPath, options) => { - const path = '/myfritz/cgi-bin/luacgi_notimeout' + - '?cmd=tam&script=/http_file_download.lua' + - '&cmd_files=' + messagePath - - const response = await fritzRequest.request(path, 'GET', options) + const version = await fritzSystem.getVersionNumber(options) - if (response.error) return response + if (version.error) return version + + if (version >= 750) { + const path = '/cgi-bin/luacgi_notimeout?' + const param = { + script: '/lua/photo.lua', + myabfile: messagePath, + } + options.noAuth = false + const response = await fritzRequest.request(path, 'GET', options, param) + + if (response.error) return response + + if (response.headers.get('Content-Type') !== 'audio/x-wav') { + return { error: { message: 'Did not receive wav audio file', raw: response } } + } - if (response.headers['content-type'] !== 'audio/x-wav') { + return { message: 'Saved tam message to ' + localPath } + } else { + const path = '/myfritz/cgi-bin/luacgi_notimeout' + + '?cmd=tam&script=/http_file_download.lua' + + '&cmd_files=' + messagePath + + const response = await fritzRequest.request(path, 'GET', options) + + if (response.error) return response + + if (response.headers['content-type'] !== 'audio/x-wav') { return { error: { message: 'Did not receive wav audio file', raw: response } } + } + + return { message: 'Saved tam message to ' + localPath } + } } - return { message: 'Saved tam message to ' + localPath } -} + /** * Mark a message as read. @@ -244,7 +268,12 @@ fritzFon.getPhonebook = async (phonebookId = 0, options) => { PhonebookExport: '' } const path = '/cgi-bin/firmwarecfg' - const response = await fritzRequest.request(path, 'POST', options, false, formData) + + const headers = { + "Content-Type": "multipart/form-data", + } + + const response = await fritzRequest.request(path, 'POST', options, formData, headers) if (response.error) return response diff --git a/src/request.js b/src/request.js index 5fa8bff..bec2444 100644 --- a/src/request.js +++ b/src/request.js @@ -5,6 +5,7 @@ import fetch from 'node-fetch'; import https from 'https'; +import FormData from 'form-data'; const httpsAgent = new https.Agent({ //Disable SSL verification since the Fritz.box endpoint is not secure enough @@ -46,13 +47,19 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => options.removeSidFromUri = false } if (headers['Content-Type'] === 'application/x-www-form-urlencoded') { - const formBody = [] + const encodedForm = [] for (const property in body) { const encodedKey = encodeURIComponent(property); const encodedValue = encodeURIComponent(body[property]); - formBody.push(encodedKey + "=" + encodedValue); + encodedForm.push(encodedKey + "=" + encodedValue); } - body = formBody.join("&") + body = encodedForm.join("&") + } else if (headers['Content-Type'] === 'multipart/form-data') { + const form = new FormData(); + for (const property in body) { + form.append(property, body[property]) + } + body = form } else if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { // Add SID to path if one has been given to us. body['sid'] = options.sid @@ -72,9 +79,7 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => // Set the options for the request. const uri = options.protocol + '://' + options.server + path - console.log('headers: ', headers) - console.log('method: ', method) - console.log('body: ', body) + console.log('uri: ', uri, 'headers: ', headers, 'method: ', method, 'body: ', body) // Execute HTTP(S) request. try { const fetchResponse = await fetch(uri, { @@ -84,7 +89,8 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => agent: httpsAgent }) const response = { - body: await fetchResponse.text() + body: await fetchResponse.text(), + headers: fetchResponse.headers } console.log('response fetch', response) return response diff --git a/test/phonebook.js b/test/phonebook.js index af0cb1e..9019235 100644 --- a/test/phonebook.js +++ b/test/phonebook.js @@ -1,9 +1,9 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} const phonebookId = 0 async function getPhonebook () { - const phonebook = await fritz.getPhonebook(phonebookId, options) + const phonebook = await fritz.getPhonebook(phonebookId, pckJson.options) if (phonebook.error) { console.log('Error: ', phonebook.error.message) diff --git a/test/tamdownload.js b/test/tamdownload.js index 21ca304..66b7229 100644 --- a/test/tamdownload.js +++ b/test/tamdownload.js @@ -1,10 +1,10 @@ -const fritz = require('../index.js') -const options = require('../package.json').options -const messagePath = '/data/tam/rec/rec.0.020' +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} +const messagePath = '/data/tam/rec/rec.0.000' const localPath = './test/rec20.wav' async function download () { - const status = await fritz.downloadTamMessage(messagePath, localPath, options) + const status = await fritz.downloadTamMessage(messagePath, localPath, pckJson.options) if (status.error) { console.log('Error:', status.error.message) From 7fd4d8f7cff3771128a612b474624ffc61125835 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 28 Jul 2023 21:46:21 +0200 Subject: [PATCH 15/32] added TODO implementation for toggleswitch.js --- src/dect.js | 12 ++++++++++-- src/request.js | 34 +++++++++++++++++----------------- test/toggleswitch.js | 7 +++---- 3 files changed, 30 insertions(+), 23 deletions(-) diff --git a/src/dect.js b/src/dect.js index 7f7af7e..8f2199d 100644 --- a/src/dect.js +++ b/src/dect.js @@ -39,7 +39,10 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { let response - if (version >= 683) { + if (version >= 750) { + //TODO when someone gets a smartswitch + return { error: 'The implementation for FritzOS!7.50 is not available '} + } else if (version >= 683) { // Post 06.83 uses a POST request. if (!options.sid) { @@ -55,7 +58,12 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { cmdValue: value, deviceId: deviceId } - response = await fritzRequest.request(path, 'POST', options, form) + + + const headers = { + "Content-Type": "multipart/form-data", + } + response = await fritzRequest.request(path, 'POST', options, form, headers) } else { // Pre 06.83 used a GET request. const path = '/myfritz/areas/homeauto.lua?ajax_id=' + diff --git a/src/request.js b/src/request.js index bec2444..123922d 100644 --- a/src/request.js +++ b/src/request.js @@ -46,16 +46,17 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => if (typeof options.removeSidFromUri === 'undefined') { options.removeSidFromUri = false } + if (headers['Content-Type'] === 'application/x-www-form-urlencoded') { const encodedForm = [] for (const property in body) { const encodedKey = encodeURIComponent(property); const encodedValue = encodeURIComponent(body[property]); - encodedForm.push(encodedKey + "=" + encodedValue); + encodedForm.push(`${encodedKey}=${encodedValue}`); } body = encodedForm.join("&") } else if (headers['Content-Type'] === 'multipart/form-data') { - const form = new FormData(); + const form = new FormData() for (const property in body) { form.append(property, body[property]) } @@ -65,9 +66,8 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => body['sid'] = options.sid } - if (method === 'POST') { - } else if (method === 'GET'){ + if (method === 'GET'){ for (const key in body) { if (!path.endsWith('&') || !path.endsWith('?')) { path += '&' @@ -78,25 +78,25 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => } // Set the options for the request. - const uri = options.protocol + '://' + options.server + path + const uri = `${options.protocol}://${options.server}${path}` console.log('uri: ', uri, 'headers: ', headers, 'method: ', method, 'body: ', body) // Execute HTTP(S) request. - try { const fetchResponse = await fetch(uri, { headers: headers, method: method || 'GET', body: body, agent: httpsAgent }) - const response = { - body: await fetchResponse.text(), - headers: fetchResponse.headers + if (fetchResponse.ok) { + const response = { + body: await fetchResponse.text(), + headers: fetchResponse.headers, + } + console.log('response fetch', response) + return response + } else { + return fritzRequest.findFailCause(fetchResponse) } - console.log('response fetch', response) - return response - } catch (error) { - return fritzRequest.findFailCause(error) - } } @@ -104,11 +104,11 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => * Find the cause of a failed request. * * @private - * @param {Object} response HTTP request response + * @param {Response} response HTTP request response * @return {string} Detailed error message */ fritzRequest.findFailCause = (response) => { - switch (response.statusCode) { + switch (response.status) { case 403: return { error: { message: 'Not authenticated correctly for communication with Fritz!Box.' } } case 404: @@ -118,7 +118,7 @@ fritzRequest.findFailCause = (response) => { default: if (response.message) { - return { error: { message: response.message } } + return { error: { message: response.statusText } } } return { error: { message: 'Encountered an unexpected error.', raw: response } } diff --git a/test/toggleswitch.js b/test/toggleswitch.js index 40e5bdb..6ce36ed 100644 --- a/test/toggleswitch.js +++ b/test/toggleswitch.js @@ -1,11 +1,10 @@ -const fritz = require('../index.js') -const options = require('../package.json').options - +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} const id = 17 const state = 1 async function toggle () { - const result = await fritz.toggleSwitch(id, state, options) + const result = await fritz.toggleSwitch(id, state, pckJson.options) if (result.error) { console.log('Error:', result.error.message) process.exit(1) From da68ecedd7c4c6d0ed318faa397ff158db1343c3 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 28 Jul 2023 21:47:04 +0200 Subject: [PATCH 16/32] refactor version.js --- test/version.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/test/version.js b/test/version.js index 55f75c4..3447a40 100644 --- a/test/version.js +++ b/test/version.js @@ -1,8 +1,8 @@ -const fritz = require('../index.js') -const options = require('../package.json').options +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} async function version () { - const version = await fritz.getVersion(options) + const version = await fritz.getVersion(pckJson.options) if (version.error) { console.log('Error:', version.error.message) process.exit(1) From cf8089ad1cfc112de844963131ea2d99e67276b0 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 28 Jul 2023 21:56:33 +0200 Subject: [PATCH 17/32] eslint & package.json --- .eslintrc.json | 13 +++++++++++++ package.json | 9 +++++---- 2 files changed, 18 insertions(+), 4 deletions(-) create mode 100644 .eslintrc.json diff --git a/.eslintrc.json b/.eslintrc.json new file mode 100644 index 0000000..8f6a415 --- /dev/null +++ b/.eslintrc.json @@ -0,0 +1,13 @@ +{ + "env": { + "browser": true, + "es2021": true + }, + "extends": "eslint:recommended", + "parserOptions": { + "ecmaVersion": "latest", + "sourceType": "module" + }, + "rules": { + } +} diff --git a/package.json b/package.json index 54589bc..36767ec 100644 --- a/package.json +++ b/package.json @@ -37,12 +37,12 @@ "dependencies": { "crypto": "^0.0.3", "csvjson": "^3.0.0", - "request": "^2.79.0", - "request-promise": "^4.1.1", + "FormData": "^0.10.1", + "node-fetch": "^3.3.1", "xml-to-json-promise": "^0.0.3", - "xml2js": "^0.4.17" }, "devDependencies": { + "eslint": "^8.45.0", "standard": "^8.6.0" }, "standard": { @@ -57,5 +57,6 @@ "password": "xxx", "callmonitorport": "1012", "debug": false - } + }, + "type": "module" } From 87b4d5f8b8d0404b31a78622109a9de2d9b137f0 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 8 Aug 2023 08:53:55 +0200 Subject: [PATCH 18/32] added new GetDevices/GetDeviceByName functionality --- package.json | 17 ++++++------ src/fon.js | 4 +-- src/request.js | 11 +++++--- src/wlan.js | 75 ++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 92 insertions(+), 15 deletions(-) diff --git a/package.json b/package.json index 36767ec..a30c318 100644 --- a/package.json +++ b/package.json @@ -1,9 +1,8 @@ { - "name": "fritzbox.js", + "name": "neofritzbox.js", "version": "2.0.1", - "author": "@lesander", - "description": "The most powerful Fritz!Box API for NodeJS.", - "homepage": "https://git.io/fritzbox", + "author": "@fiwel00", + "description": "The second most powerful Fritz!Box API for NodeJS.", "license": "MIT", "contributors": [], "keywords": [ @@ -25,7 +24,7 @@ ], "repository": { "type": "git", - "url": "git+https://github.com/lesander/fritzbox.js.git" + "url": "git+https://github.com/fiwel00/fritzbox.js.git" }, "main": "index.js", "scripts": { @@ -39,7 +38,7 @@ "csvjson": "^3.0.0", "FormData": "^0.10.1", "node-fetch": "^3.3.1", - "xml-to-json-promise": "^0.0.3", + "xml-to-json-promise": "^0.0.3" }, "devDependencies": { "eslint": "^8.45.0", @@ -52,9 +51,9 @@ }, "options": { "protocol": "https", - "server": "fritz.box", - "username": "xxx", - "password": "xxx", + "server": "192.168.178.1", + "username": "fritz0275", + "password": "acht1714", "callmonitorport": "1012", "debug": false }, diff --git a/src/fon.js b/src/fon.js index 2b31f49..92e209e 100644 --- a/src/fon.js +++ b/src/fon.js @@ -66,7 +66,7 @@ fritzFon.getTamMessages = async (options) => { console.log(calls) // Filter only TAM messages. tamMessages = [] - for (var call in calls) { + for (const call in calls) { if (calls[call].type === 'tam') { tamMessages.push(calls[call].tam_data) } @@ -98,7 +98,7 @@ fritzFon.getTamMessages = async (options) => { // Filter only TAM messages. tamMessages = [] - for (var call in calls) { + for (let call in calls) { if (calls[call].tam_data) { tamMessages.push(calls[call].tam_data) } diff --git a/src/request.js b/src/request.js index 123922d..dfa3a50 100644 --- a/src/request.js +++ b/src/request.js @@ -47,6 +47,11 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => options.removeSidFromUri = false } + if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { + // Add SID to path if one has been given to us. + body['sid'] = options.sid + } + if (headers['Content-Type'] === 'application/x-www-form-urlencoded') { const encodedForm = [] for (const property in body) { @@ -61,10 +66,8 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => form.append(property, body[property]) } body = form - } else if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { - // Add SID to path if one has been given to us. - body['sid'] = options.sid - } + } + if (method === 'GET'){ diff --git a/src/wlan.js b/src/wlan.js index 1af1ede..9e4e8d8 100644 --- a/src/wlan.js +++ b/src/wlan.js @@ -1,5 +1,80 @@ /** * @module fritzWlan */ + +import fritzRequest from './request.js' + const fritzWlan = {} +/** + * Get the name of the connected devices connected to the LAN + * @return {Array} The connected devices to the LAN as an array. + */ +fritzWlan.getDevices = async (options) => { + + const path = `/data.lua` + const params = { + xhr:1, + page: 'netDev', + xhrId: 'cleanup', + useajax: '1', + } + + const headers = { + 'Content-Type': 'application/x-www-form-urlencoded' + } + + const response = await fritzRequest.request(path, 'POST', options, params, headers) + + if (response.error) return response + + return JSON.parse(response.body).data +} + +fritzWlan.getDeviceByName = async (deviceName, options) => { + const deviceData = await fritzWlan.getDevices(options); + const activeDevices = deviceData.active + const passiveDevices = deviceData.passiv + + for (let activeDevice in activeDevices) { + if (activeDevices[activeDevice].name === deviceName) { + return activeDevices[activeDevice] + } + } + for (let passivDevice in passiveDevices) { + if (activeDevices[passivDevice].name === deviceName){ + return activeDevices[passivDevice] + } + } + + return { + error : { message: 'Device could not be found', raw: deviceData} + } +} + +fritzWlan.resetDevice = async (deviceName, options) => { + const device = await fritzWlan.getDeviceByName(deviceName, options) + + const path = `/data.lua` + const params = { + xhr: "1", + dev_name: device.name, + internetdetail: "unlimited", + allow_pcp_and_upnp: "off", + static_dhcp: "off", + dev: device.UID, + btn_reset_dev: "", + page: "edit_device" + } + + const headers = { + 'Content-Type': 'application/x-www-form-urlencoded' + } + + const response = await fritzRequest.request(path, 'POST', options, params, headers) + + if (response.error) return response + + return "" +} + export default fritzWlan \ No newline at end of file From b2d4f036cde8f540a919c43741229b333003b5f0 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 8 Aug 2023 08:57:41 +0200 Subject: [PATCH 19/32] updated readme --- README.md | 54 ++++++++---------------------------------------------- 1 file changed, 8 insertions(+), 46 deletions(-) diff --git a/README.md b/README.md index c0351ac..d5ccd4d 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -# FritzBox.js +# NeoFritzBox.js [![GitHub release](https://img.shields.io/github/release/lesander/fritzbox.js.svg?maxAge=1)]() [![Completion Status](https://img.shields.io/badge/completion-70%25-green.svg)]() [![Build Status](https://travis-ci.org/lesander/fritzbox.js.svg?branch=master&cache=pls)](https://travis-ci.org/lesander/fritzbox.js) @@ -8,19 +8,19 @@ The most powerful, simple and complete [AVM](https://avm.de) Fritz!Box [API](https://avm.de/Schnittstellen). -**This project is still a work in progress. [See issue #1 for the current status.](https://github.com/lesander/fritzbox.js/issues/1)** +**This project is a fork of [fritzbox.js](https://github.com/lesander/fritzbox.js) is mostly a refactoring excerise. But it already welcomes some new changes** ## Getting Started This module is future-proof and uses async/await promises. This means that you need to run NodeJS version `7.6.0` or newer. If your NodeJS version is between `7.0.0` and `7.5.0` you can use the harmony flag `--harmony-async-await` to make use of the async/await promises. -This package was tested on Fritz!Box 7390 and 7490, with firmware versions `6.53`, `6.51` and `6.83`. +This package was tested on Fritz!Box 7520, with firmware versions `7.50`. ## Installing ``` -npm install fritzbox.js +npm install neofritzbox.js ``` ## Usage @@ -47,52 +47,14 @@ const options = { To minimize overhead and limit login requests made to the Fritz!Box it is recommended to store the SID once one has been obtained using [`fritz.getSessionId`](https://fritzbox.js.org/api/#fritzLogin.getSessionId). ## Documentation -Want to get started with FritzBox.js? Cool! The API is -[documented and available here](https://fritzbox.js.org/api), and you can -[see some examples](/test) in the `test/` folder. +The only available documentation is from the base repository and does not include any additional feature that have been implemented see [fritzbox.js documentation](https://fritzbox.js.org/api). + +Tests for new features have been written and you can [see some examples](/test) in the `test/` folder. ## Contributing -If you'd like to contribute to FritzBox.js, or file a bug or feature request, +If you'd like to contribute to NeoFritzBox.js, or file a bug or feature request, please head over to [the issue tracker](/issues) or [open a pull request](/pulls). -## Migrating from 1.x.x to 2.x.x -FritzBox.js v2.x is not backwards compatible with v1.x. -One of the mayor changes includes the switch to `async/await` Promises. -In v1.x, Promises were implemented with a `then`, `catch`: - -```js -fritz.getCalls(options) -.then((callHistory) => { - console.log(callHistory) -}) -.catch((error) => { - console.log(error) -}) -``` - -With v2.x, the `catch` will no longer catch any errors, since the module is now -built to provide support for `await`. Any errors will be passed along like this: - -```js -fritz.getCalls(options) -.then((callHistory) => { - if (callHistory.error) return console.log(callHistory.error.message) - console.log(callHistory) -}) -``` - -Of course, this can be simplified using `await`: - -```js -let callHistory = await fritz.getCalls(options) -if (callHistory.error) return console.log(callHistory.error.message) -console.log(callHistory) -``` - -Note that any Promise waiting to be fulfilled using `await` should be put inside an `async` function. - -For more changes, please see the [roadmap](https://github.com/lesander/fritzbox.js/issues/1). - ## License This software is open-sourced under the MIT License ([see the LICENSE file for the full license](/LICENSE)). From 0e9055db95b110092ed038be6bfc36658b87ae22 Mon Sep 17 00:00:00 2001 From: Fiwel Date: Tue, 8 Aug 2023 09:07:08 +0200 Subject: [PATCH 20/32] Create npm-publish.yml --- .github/workflows/npm-publish.yml | 33 +++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) create mode 100644 .github/workflows/npm-publish.yml diff --git a/.github/workflows/npm-publish.yml b/.github/workflows/npm-publish.yml new file mode 100644 index 0000000..f96a9dd --- /dev/null +++ b/.github/workflows/npm-publish.yml @@ -0,0 +1,33 @@ +# This workflow will run tests using node and then publish a package to GitHub Packages when a release is created +# For more information see: https://docs.github.com/en/actions/publishing-packages/publishing-nodejs-packages + +name: Node.js Package + +on: + release: + types: [created] + +jobs: + build: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v3 + - uses: actions/setup-node@v3 + with: + node-version: 18 + - run: npm ci + - run: npm test + + publish-npm: + needs: build + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v3 + - uses: actions/setup-node@v3 + with: + node-version: 16 + registry-url: https://registry.npmjs.org/ + - run: npm ci + - run: npm publish + env: + NODE_AUTH_TOKEN: ${{secrets.npm_token}} From 12fb25ea29a98486fee0739b12aae9dbc265fe96 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 8 Aug 2023 09:21:12 +0200 Subject: [PATCH 21/32] added test for getDeviceByName --- test/devices.js | 14 ++++++++++++++ 1 file changed, 14 insertions(+) create mode 100644 test/devices.js diff --git a/test/devices.js b/test/devices.js new file mode 100644 index 0000000..0233799 --- /dev/null +++ b/test/devices.js @@ -0,0 +1,14 @@ +import fritz from '../index.js' +import pckJson from '../package.json' assert {type:'json'} + +async function device () { + const device = await fritz.getDeviceByName('Pixel-6', pckJson.options) + if (device.error) { + console.log('Error:', device.error.message) + process.exit(1) + } else { + console.log('Device is connected to the LAN', device) + } +} + +device() From 0c6f3510f221f33db8d149c3c618ab39a90ec58c Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 8 Aug 2023 09:45:15 +0200 Subject: [PATCH 22/32] release 2.1.1 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index a30c318..177d88b 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "neofritzbox.js", - "version": "2.0.1", + "version": "2.1.1", "author": "@fiwel00", "description": "The second most powerful Fritz!Box API for NodeJS.", "license": "MIT", From 5fc28274bef0671cd42ed36783b0d5c92f5785a3 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Tue, 8 Aug 2023 11:19:57 +0200 Subject: [PATCH 23/32] added package-lock.js --- package-lock.json | 3498 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 3498 insertions(+) create mode 100644 package-lock.json diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..f5ebe71 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,3498 @@ +{ + "name": "fritzbox.js", + "version": "2.0.1", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "fritzbox.js", + "version": "2.0.1", + "license": "MIT", + "dependencies": { + "axios": "^1.4.0", + "crypto": "^0.0.3", + "csvjson": "^3.0.0", + "fetch": "^1.1.0", + "FormData": "^0.10.1", + "node-fetch": "^3.3.1", + "request": "^2.79.0", + "request-promise": "^4.1.1", + "xml-to-json-promise": "^0.0.3", + "xml2js": "^0.4.17" + }, + "devDependencies": { + "eslint": "^8.45.0", + "standard": "^8.6.0" + } + }, + "node_modules/@aashutoshrathi/word-wrap": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", + "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^3.3.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.5.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.5.1.tgz", + "integrity": "sha512-Z5ba73P98O1KUYCCJTUeVpja9RcGoMdncZ6T49FCUl2lN38JtCJ+3WgIDBv0AuY4WChU5PmtJmOCTlN6FZTFKQ==", + "dev": true, + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.0.tgz", + "integrity": "sha512-Lj7DECXqIVCqnqjjHMPna4vn6GJcMgul/wuS0je9OZ9gsL0zzDpKPVtcG1HaDVc+9y+qgXneTeUMbCqXJNpH1A==", + "dev": true, + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/ignore": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", + "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/@eslint/js": { + "version": "8.44.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.44.0.tgz", + "integrity": "sha512-Ag+9YM4ocKQx9AarydN0KY2j0ErMHNIocPDrVo8zAE44xLTjEtz81OdR68/cydGtk6m6jDb5Za3r2useMzYmSw==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.10.tgz", + "integrity": "sha512-KVVjQmNUepDVGXNuoRRdmmEjruj0KfiGSbS8LVc12LMsWDQzRXJ0qdhN8L8uUigKpfEHRhlaQFY0ib1tnUbNeQ==", + "dev": true, + "dependencies": { + "@humanwhocodes/object-schema": "^1.2.1", + "debug": "^4.1.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz", + "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==", + "dev": true + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/acorn": { + "version": "8.10.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.10.0.tgz", + "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ajv-keywords": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-1.5.1.tgz", + "integrity": "sha512-vuBv+fm2s6cqUyey2A7qYcvsik+GMDJsw8BARP2sDE76cqmaZVarsvHf7Vx6VJ0Xk8gLl+u3MoAPf6gKzJefeA==", + "dev": true, + "peerDependencies": { + "ajv": ">=4.10.0" + } + }, + "node_modules/ansi-escapes": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-1.4.0.tgz", + "integrity": "sha512-wiXutNjDUlNEDWHcYH3jtZUhd3c4/VojassD8zHdHCY13xbZy2XbW+NKQwA0tWGBVzDA9qEzYwfoSsWmviidhw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "node_modules/asn1": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz", + "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==", + "dependencies": { + "safer-buffer": "~2.1.0" + } + }, + "node_modules/assert-plus": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", + "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==", + "engines": { + "node": ">=0.8" + } + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, + "node_modules/aws-sign2": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", + "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==", + "engines": { + "node": "*" + } + }, + "node_modules/aws4": { + "version": "1.12.0", + "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.12.0.tgz", + "integrity": "sha512-NmWvPnx0F1SfrQbYwOi7OeaNGokp9XhzNioJ/CSBs8Qa4vxug81mhJEAVZwxXuBmYB5KDRfMq/F3RR0BIU7sWg==" + }, + "node_modules/axios": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.4.0.tgz", + "integrity": "sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==", + "dependencies": { + "follow-redirects": "^1.15.0", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/axios/node_modules/form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha512-XqYMR2dfdGMW+hd0IUZ2PwK+fGeFkOxZJ0wY+JaQAHzt1Zx8LcvpiZD2NiGkEG8qx0CfkAOr5xt76d1e8vG90g==", + "dev": true, + "dependencies": { + "chalk": "^1.1.3", + "esutils": "^2.0.2", + "js-tokens": "^3.0.2" + } + }, + "node_modules/babel-code-frame/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/babel-code-frame/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/bcrypt-pbkdf": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", + "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==", + "dependencies": { + "tweetnacl": "^0.14.3" + } + }, + "node_modules/biskviit": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/biskviit/-/biskviit-1.0.1.tgz", + "integrity": "sha512-VGCXdHbdbpEkFgtjkeoBN8vRlbj1ZRX2/mxhE8asCCRalUx2nBzOomLJv8Aw/nRt5+ccDb+tPKidg4XxcfGW4w==", + "dependencies": { + "psl": "^1.1.7" + }, + "engines": { + "node": ">=1.0.0" + } + }, + "node_modules/bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==" + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "node_modules/bufferjs": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/bufferjs/-/bufferjs-3.0.1.tgz", + "integrity": "sha512-qrCIGPcd9ODawCNyqR2o55zgaC/r7XHZ7oUh2s99uk+NVBS3SjIHigxS1S2KXpt8wsoQxAN55iPi8GIH8TGMRg==", + "engines": { + "node": ">=0.2.0" + } + }, + "node_modules/caller-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", + "integrity": "sha512-UJiE1otjXPF5/x+T3zTnSFiTOEmJoGTD9HmBoxnCUwho61a2eSNn/VwtwuIBDAo2SEOv1AJ7ARI5gCmohFLu/g==", + "dev": true, + "dependencies": { + "callsites": "^0.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/caller-path/node_modules/callsites": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", + "integrity": "sha512-Zv4Dns9IbXXmPkgRRUjAaJQgfN4xX5p6+RQFhWUqscdvvK2xK/ZL8b3IXIJsj+4sD+f24NwnWy2BY8AJ82JB0A==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/caseless": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", + "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==" + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/circular-json": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", + "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", + "deprecated": "CircularJSON is in maintenance only, flatted is its successor.", + "dev": true + }, + "node_modules/cli-cursor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-1.0.2.tgz", + "integrity": "sha512-25tABq090YNKkF6JH7lcwO0zFJTRke4Jcq9iX2nr/Sz0Cjjv4gckmwlW6Ty/aoyFd6z3ysR2hMGC2GFugmBo6A==", + "dev": true, + "dependencies": { + "restore-cursor": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", + "dev": true + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "node_modules/concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "dev": true, + "engines": [ + "node >= 0.8" + ], + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", + "dev": true + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/crypto": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/crypto/-/crypto-0.0.3.tgz", + "integrity": "sha512-Q6Ka98WcvWXXg+9cnqd3jHpTSIOaH6/q0m/bESMfQo/0uFxy6e/7EqVS4JdaWx9qLdqV56tDufy2b12dj7BHJg==", + "deprecated": "This package is no longer supported. It's now a built-in Node module. If you've depended on crypto, you should switch to the one that's built-in.", + "engines": { + "node": "*" + } + }, + "node_modules/csvjson": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/csvjson/-/csvjson-3.0.0.tgz", + "integrity": "sha512-nCig2BkiCDN5g9UyXOWP0il7KfVKNjHZorI+qlRBckxV66h/ww1Fl9Or9P3/C1L9XEygLpdszu/uxFIJmgl4DA==" + }, + "node_modules/d": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz", + "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==", + "dev": true, + "dependencies": { + "es5-ext": "^0.10.50", + "type": "^1.0.1" + } + }, + "node_modules/dashdash": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", + "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==", + "dependencies": { + "assert-plus": "^1.0.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/data-uri-to-buffer": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz", + "integrity": "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==", + "engines": { + "node": ">= 12" + } + }, + "node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/debug-log": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/debug-log/-/debug-log-1.0.1.tgz", + "integrity": "sha512-gV/pe1YIaKNgLYnd1g9VNW80tcb7oV5qvNUxG7NM8rbDpnl6RGunzlAtlGSb0wEs3nesu2vHNiX9TSsZ+Y+RjA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "node_modules/deglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/deglob/-/deglob-2.1.1.tgz", + "integrity": "sha512-2kjwuGGonL7gWE1XU4Fv79+vVzpoQCl0V+boMwWtOQJV2AGDabCwez++nB1Nli/8BabAfZQ/UuHPlp6AymKdWw==", + "dev": true, + "dependencies": { + "find-root": "^1.0.0", + "glob": "^7.0.5", + "ignore": "^3.0.9", + "pkg-config": "^1.1.0", + "run-parallel": "^1.1.2", + "uniq": "^1.0.1" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/ecc-jsbn": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", + "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==", + "dependencies": { + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" + } + }, + "node_modules/encoding": { + "version": "0.1.12", + "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.12.tgz", + "integrity": "sha512-bl1LAgiQc4ZWr++pNYUdRe/alecaHFeHxIJ/pNciqGdKXghaTCOwKkbKp6ye7pKZGu/GcaSXFk8PBVhgs+dJdA==", + "dependencies": { + "iconv-lite": "~0.4.13" + } + }, + "node_modules/es5-ext": { + "version": "0.10.62", + "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.62.tgz", + "integrity": "sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==", + "dev": true, + "hasInstallScript": true, + "dependencies": { + "es6-iterator": "^2.0.3", + "es6-symbol": "^3.1.3", + "next-tick": "^1.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/es6-iterator": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz", + "integrity": "sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==", + "dev": true, + "dependencies": { + "d": "1", + "es5-ext": "^0.10.35", + "es6-symbol": "^3.1.1" + } + }, + "node_modules/es6-map": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/es6-map/-/es6-map-0.1.5.tgz", + "integrity": "sha512-mz3UqCh0uPCIqsw1SSAkB/p0rOzF/M0V++vyN7JqlPtSW/VsYgQBvVvqMLmfBuyMzTpLnNqi6JmcSizs4jy19A==", + "dev": true, + "dependencies": { + "d": "1", + "es5-ext": "~0.10.14", + "es6-iterator": "~2.0.1", + "es6-set": "~0.1.5", + "es6-symbol": "~3.1.1", + "event-emitter": "~0.3.5" + } + }, + "node_modules/es6-set": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/es6-set/-/es6-set-0.1.6.tgz", + "integrity": "sha512-TE3LgGLDIBX332jq3ypv6bcOpkLO0AslAQo7p2VqX/1N46YNsvIWgvjojjSEnWEGWMhr1qUbYeTSir5J6mFHOw==", + "dev": true, + "dependencies": { + "d": "^1.0.1", + "es5-ext": "^0.10.62", + "es6-iterator": "~2.0.3", + "es6-symbol": "^3.1.3", + "event-emitter": "^0.3.5", + "type": "^2.7.2" + }, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/es6-set/node_modules/type": { + "version": "2.7.2", + "resolved": "https://registry.npmjs.org/type/-/type-2.7.2.tgz", + "integrity": "sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==", + "dev": true + }, + "node_modules/es6-symbol": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz", + "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==", + "dev": true, + "dependencies": { + "d": "^1.0.1", + "ext": "^1.1.2" + } + }, + "node_modules/es6-weak-map": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.3.tgz", + "integrity": "sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==", + "dev": true, + "dependencies": { + "d": "1", + "es5-ext": "^0.10.46", + "es6-iterator": "^2.0.3", + "es6-symbol": "^3.1.1" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/escope": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/escope/-/escope-3.6.0.tgz", + "integrity": "sha512-75IUQsusDdalQEW/G/2esa87J7raqdJF+Ca0/Xm5C3Q58Nr4yVYjZGp/P1+2xiEVgXRrA39dpRb8LcshajbqDQ==", + "dev": true, + "dependencies": { + "es6-map": "^0.1.3", + "es6-weak-map": "^2.0.1", + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/escope/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/eslint": { + "version": "8.45.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.45.0.tgz", + "integrity": "sha512-pd8KSxiQpdYRfYa9Wufvdoct3ZPQQuVuU5O6scNgMuOMYuxvH0IGaYK0wUFjo4UYYQQCUndlXiMbnxopwvvTiw==", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.4.0", + "@eslint/eslintrc": "^2.1.0", + "@eslint/js": "8.44.0", + "@humanwhocodes/config-array": "^0.11.10", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "ajv": "^6.10.0", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.0", + "eslint-visitor-keys": "^3.4.1", + "espree": "^9.6.0", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-config-standard": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-6.2.1.tgz", + "integrity": "sha512-CIPBOuTRFzXaLTKtbGJi1pmgJrweQ4K9GW5Zu5ndEHWPCfNCQSIndLbuqtdpf5RW2F2ERqEEvwK9sT1HIL2wUg==", + "dev": true, + "peerDependencies": { + "eslint": ">=3.8.1", + "eslint-plugin-promise": ">=3.3.0", + "eslint-plugin-standard": ">=2.0.0" + } + }, + "node_modules/eslint-plugin-promise": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-3.4.2.tgz", + "integrity": "sha512-wyiQUH4buAhRZIz6ipgxVwvbFrGL+6v6z/TEt7Dw/O12l/BPnbEL3NNY9j4i0kKrIqyT7Ebow9kZYpSQ3HTeag==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-plugin-standard": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-standard/-/eslint-plugin-standard-2.0.1.tgz", + "integrity": "sha512-IZyq3HcnnLh7zK/hUzqniIiioHPUo42sSuezC6OY9PLQvUU/w4McmZJAvUbY6VRx6nFjYCfH/9wgBEf+aKjfnw==", + "dev": true, + "peerDependencies": { + "eslint": ">=3.0.0" + } + }, + "node_modules/eslint-scope": { + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.1.tgz", + "integrity": "sha512-CvefSOsDdaYYvxChovdrPo/ZGt8d5lrJWleAc1diXRKhHGiTYEI26cvo8Kle/wGnsizoCJjK73FMg1/IkIwiNA==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.1.tgz", + "integrity": "sha512-pZnmmLwYzf+kWaM/Qgrvpen51upAktaaiI01nsJD/Yr3lMOdNtq0cxkrrg16w64VtisN6okbs7Q8AfGqj4c9fA==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/ignore": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", + "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/esquery": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", + "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "dev": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/event-emitter": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz", + "integrity": "sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==", + "dev": true, + "dependencies": { + "d": "1", + "es5-ext": "~0.10.14" + } + }, + "node_modules/exit-hook": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/exit-hook/-/exit-hook-1.1.1.tgz", + "integrity": "sha512-MsG3prOVw1WtLXAZbM3KiYtooKR1LvxHh3VHsVtIy0uiUu8usxgB/94DP2HxtD/661lLdB6yzQ09lGJSQr6nkg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ext": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz", + "integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==", + "dev": true, + "dependencies": { + "type": "^2.7.2" + } + }, + "node_modules/ext/node_modules/type": { + "version": "2.7.2", + "resolved": "https://registry.npmjs.org/type/-/type-2.7.2.tgz", + "integrity": "sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==", + "dev": true + }, + "node_modules/extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" + }, + "node_modules/extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==", + "engines": [ + "node >=0.6.0" + ] + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "node_modules/fastq": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz", + "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/fetch": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fetch/-/fetch-1.1.0.tgz", + "integrity": "sha512-5O8TwrGzoNblBG/jtK4NFuZwNCkZX6s5GfRNOaGtm+QGJEuNakSC/i2RW0R93KX6E0jVjNXm6O3CRN4Ql3K+yA==", + "dependencies": { + "biskviit": "1.0.1", + "encoding": "0.1.12" + } + }, + "node_modules/fetch-blob": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.2.0.tgz", + "integrity": "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/jimmywarting" + }, + { + "type": "paypal", + "url": "https://paypal.me/jimmywarting" + } + ], + "dependencies": { + "node-domexception": "^1.0.0", + "web-streams-polyfill": "^3.0.3" + }, + "engines": { + "node": "^12.20 || >= 14.13" + } + }, + "node_modules/figures": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-1.7.0.tgz", + "integrity": "sha512-UxKlfCRuCBxSXU4C6t9scbDyWZ4VlaFFdojKtzJuSkuOBQ5CNFum+zZXFwHjo+CxBC1t6zlYPgHIgFjL8ggoEQ==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5", + "object-assign": "^4.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/figures/node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/File": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/File/-/File-0.10.2.tgz", + "integrity": "sha512-gomQVTq/10wIR399uhGTWtYcYneTXbfe3p2RO/NR0MPrLkIyOaE9DCEPXihAm+72epLtXaplitwfJ/wkmj88dg==", + "dependencies": { + "mime": ">= 0.0.0" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/FileReader": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/FileReader/-/FileReader-0.10.2.tgz", + "integrity": "sha512-O0K+5KRnDBFNogBPtV5hWjyJSQOoB2j3ZLbre1pBjCdE9CNpeIUPI8IX3G7eafIZ/HsllbGaIucEbt2qL+EX0g==", + "engines": { + "node": "*" + } + }, + "node_modules/find-root": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/find-root/-/find-root-1.1.0.tgz", + "integrity": "sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==", + "dev": true + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", + "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "dev": true, + "dependencies": { + "flatted": "^3.1.0", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz", + "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==", + "dev": true + }, + "node_modules/follow-redirects": { + "version": "1.15.2", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz", + "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/forEachAsync": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/forEachAsync/-/forEachAsync-3.0.0.tgz", + "integrity": "sha512-a6qFIJll2VuqpWNRvJfgT7pGgobR89qnARafcCkGg6q1BCWrowc3WvRoYidxtFWRwaeUpk+5kiKZtdlRLwSufQ==", + "deprecated": "Please use 'foreachasync' instead. See https://www.npmjs.com/package/foreachasync" + }, + "node_modules/forever-agent": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", + "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==", + "engines": { + "node": "*" + } + }, + "node_modules/form-data": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", + "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 0.12" + } + }, + "node_modules/FormData": { + "version": "0.10.1", + "resolved": "https://registry.npmjs.org/FormData/-/FormData-0.10.1.tgz", + "integrity": "sha512-Win+aXiahTJE/uwQWcq7BaLRdqOjNlDiWq2RkGvywkUluz8uejQrUPfmuXJJ6+4kaoH0r4S7Y+J6R/PGcNyp5w==", + "dependencies": { + "bufferjs": ">= 0.0.0", + "File": ">= 0.0.0", + "FileReader": ">= 0.0.0", + "forEachAsync": ">= 2.0.0", + "remedial": ">= 0.0.0" + } + }, + "node_modules/formdata-polyfill": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz", + "integrity": "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==", + "dependencies": { + "fetch-blob": "^3.1.2" + }, + "engines": { + "node": ">=12.20.0" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true + }, + "node_modules/generate-function": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.3.1.tgz", + "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==", + "dev": true, + "dependencies": { + "is-property": "^1.0.2" + } + }, + "node_modules/generate-object-property": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/generate-object-property/-/generate-object-property-1.2.0.tgz", + "integrity": "sha512-TuOwZWgJ2VAMEGJvAyPWvpqxSANF0LDpmyHauMjFYzaACvn+QTT/AZomvPCzVBV7yDN3OmwHQ5OvHaeLKre3JQ==", + "dev": true, + "dependencies": { + "is-property": "^1.0.0" + } + }, + "node_modules/get-stdin": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-5.0.1.tgz", + "integrity": "sha512-jZV7n6jGE3Gt7fgSTJoz91Ak5MuTLwMwkoYdjxuJ/AmjIsE1UC03y/IWkZCQGEvVNS9qoRNwy5BCqxImv0FVeA==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/getpass": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", + "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==", + "dependencies": { + "assert-plus": "^1.0.0" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "13.20.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.20.0.tgz", + "integrity": "sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "node_modules/har-schema": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", + "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==", + "engines": { + "node": ">=4" + } + }, + "node_modules/har-validator": { + "version": "5.1.5", + "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", + "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", + "deprecated": "this library is no longer supported", + "dependencies": { + "ajv": "^6.12.3", + "har-schema": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha512-C8vBJ8DwUCx19vhm7urhTuUsr4/IyP6l4VzNQDv+ryHQObW3TTTp9yB68WpYgRe2bbaGuZ/se74IqFeVnMnLZg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-ansi/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/home-or-tmp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/home-or-tmp/-/home-or-tmp-2.0.0.tgz", + "integrity": "sha512-ycURW7oUxE2sNiPVw1HVEFsW+ecOpJ5zaj7eC0RlwhibhRBod20muUN8qu/gzx956YrLolVvs1MTXwKgC2rVEg==", + "dev": true, + "dependencies": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-signature": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", + "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==", + "dependencies": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + }, + "engines": { + "node": ">=0.8", + "npm": ">=1.3.7" + } + }, + "node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "node_modules/inquirer": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-0.12.0.tgz", + "integrity": "sha512-bOetEz5+/WpgaW4D1NYOk1aD+JCqRjqu/FwRFgnIfiP7FC/zinsrfyO1vlS3nyH/R7S0IH3BIHBu4DBIDSqiGQ==", + "dev": true, + "dependencies": { + "ansi-escapes": "^1.1.0", + "ansi-regex": "^2.0.0", + "chalk": "^1.0.0", + "cli-cursor": "^1.0.1", + "cli-width": "^2.0.0", + "figures": "^1.3.5", + "lodash": "^4.3.0", + "readline2": "^1.0.1", + "run-async": "^0.1.0", + "rx-lite": "^3.1.2", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.0", + "through": "^2.3.6" + } + }, + "node_modules/inquirer/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inquirer/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inquirer/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inquirer/node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/inquirer/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inquirer/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/interpret": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz", + "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/is-core-module": { + "version": "2.12.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.12.1.tgz", + "integrity": "sha512-Q4ZuBAe2FUsKtyQJoQHlvP8OvBERxO3jEmy1I7hcRXcJBGGHFh/aJBswbXuS9sgrDH2QUO8ilkwNPHvHMd8clg==", + "dev": true, + "dependencies": { + "has": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==", + "dev": true, + "dependencies": { + "number-is-nan": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-my-ip-valid": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-my-ip-valid/-/is-my-ip-valid-1.0.1.tgz", + "integrity": "sha512-jxc8cBcOWbNK2i2aTkCZP6i7wkHF1bqKFrwEHuN5Jtg5BSaZHUZQ/JTOJwoV41YvHnOaRyWWh72T/KvfNz9DJg==", + "dev": true + }, + "node_modules/is-my-json-valid": { + "version": "2.20.6", + "resolved": "https://registry.npmjs.org/is-my-json-valid/-/is-my-json-valid-2.20.6.tgz", + "integrity": "sha512-1JQwulVNjx8UqkPE/bqDaxtH4PXCe/2VRh/y3p99heOV87HG4Id5/VfDswd+YiAfHcRTfDlWgISycnHuhZq1aw==", + "dev": true, + "dependencies": { + "generate-function": "^2.0.0", + "generate-object-property": "^1.1.0", + "is-my-ip-valid": "^1.0.0", + "jsonpointer": "^5.0.0", + "xtend": "^4.0.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-property": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz", + "integrity": "sha512-Ks/IoX00TtClbGQr4TWXemAnktAQvYB7HzcCxDGqEZU6oCmb2INHuOoKxbtR+HFkmYWBKv/dOZtGRiAjDhj92g==", + "dev": true + }, + "node_modules/is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", + "dev": true + }, + "node_modules/is-typedarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", + "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==" + }, + "node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", + "dev": true + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/isstream": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", + "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==" + }, + "node_modules/js-tokens": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha512-RjTcuD4xjtthQkaWH7dFlH85L+QaVtSoOyGdZ3g6HFhS9dFNDfLyqgm2NFe2X6cQpeFmt0452FJjFG5UameExg==", + "dev": true + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsbn": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", + "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==" + }, + "node_modules/json-schema": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz", + "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==" + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==" + }, + "node_modules/json-stable-stringify": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.0.2.tgz", + "integrity": "sha512-eunSSaEnxV12z+Z73y/j5N37/In40GK4GmsSy+tEHJMxknvqnA7/djeYtAgW0GsWHUfg+847WJjKaEylk2y09g==", + "dev": true, + "dependencies": { + "jsonify": "^0.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "node_modules/json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==" + }, + "node_modules/jsonify": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/jsonify/-/jsonify-0.0.1.tgz", + "integrity": "sha512-2/Ki0GcmuqSrgFyelQq9M05y7PS0mEwuIzrf3f1fPqkVDVRvZrPZtVSMHxdgo8Aq0sxAOb/cr2aqqA3LeWHVPg==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/jsonpointer": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/jsonpointer/-/jsonpointer-5.0.1.tgz", + "integrity": "sha512-p/nXbhSEcu3pZRdkW1OfJhpsVtW1gd4Wa1fnQc9YLiTfAjn0312eMKimbdIQzuZl9aa9xUGaRlP9T/CJE/ditQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/jsprim": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz", + "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==", + "dependencies": { + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.4.0", + "verror": "1.10.0" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/jsx-ast-utils": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-1.4.1.tgz", + "integrity": "sha512-0LwSmMlQjjUdXsdlyYhEfBJCn2Chm0zgUBmfmf1++KUULh+JOdlzrZfiwe2zmlVJx44UF+KX/B/odBoeK9hxmw==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true + }, + "node_modules/mime": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-3.0.0.tgz", + "integrity": "sha512-jSCU7/VB1loIWBZe14aEYHU/+1UMEHoaO7qxCOVJOw9GgH72VAWppxNcjU+x9a2k3GSIBXNKxXQFqRvvZ7vr3A==", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dev": true, + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/mute-stream": { + "version": "0.0.5", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.5.tgz", + "integrity": "sha512-EbrziT4s8cWPmzr47eYVW3wimS4HsvlnV5ri1xw1aR6JQo/OrJX5rkl32K/QQHdxeabJETtfeaROGhd8W7uBgg==", + "dev": true + }, + "node_modules/native-promise-only": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/native-promise-only/-/native-promise-only-0.8.1.tgz", + "integrity": "sha512-zkVhZUA3y8mbz652WrL5x0fB0ehrBkulWT3TomAQ9iDtyXZvzKeEA6GPxAItBYeNYl5yngKRX612qHOhvMkDeg==" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/next-tick": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz", + "integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==", + "dev": true + }, + "node_modules/node-domexception": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", + "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/jimmywarting" + }, + { + "type": "github", + "url": "https://paypal.me/jimmywarting" + } + ], + "engines": { + "node": ">=10.5.0" + } + }, + "node_modules/node-fetch": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.3.1.tgz", + "integrity": "sha512-cRVc/kyto/7E5shrWca1Wsea4y6tL9iYJE5FBCius3JQfb/4P4I295PfhgbJQBLTx6lATE4z+wK0rPM4VS2uow==", + "dependencies": { + "data-uri-to-buffer": "^4.0.0", + "fetch-blob": "^3.1.4", + "formdata-polyfill": "^4.0.10" + }, + "engines": { + "node": "^12.20.0 || ^14.13.1 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/node-fetch" + } + }, + "node_modules/number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/oauth-sign": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", + "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", + "engines": { + "node": "*" + } + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-1.1.0.tgz", + "integrity": "sha512-GZ+g4jayMqzCRMgB2sol7GiCLjKfS1PINkjmx8spcKce1LiVqcbQreXwqs2YAFXC6R03VIG28ZS31t8M866v6A==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/optionator": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", + "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", + "dev": true, + "dependencies": { + "@aashutoshrathi/word-wrap": "^1.2.3", + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/os-homedir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", + "integrity": "sha512-B5JU3cabzk8c67mRRd3ECmROafjYMXbuzlwtqdM8IbS8ktlTix8aFGb2bAGKrSRIlnfKwovGUUr72JUPyOb6kQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha512-DUWJr3+ULp4zXmol/SZkFf3JGsS9/SIv+Y3Rt93/UjPpDpklB5f1er4O3POIbUuUJ3FXgqte2Q7SrU6zAqwk8w==", + "dev": true + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/performance-now": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", + "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==" + }, + "node_modules/pkg-config": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/pkg-config/-/pkg-config-1.1.1.tgz", + "integrity": "sha512-ft/WI9YK6FuTuw4Ql+QUaNXtm/ASQNqDUUsZEgFZKyFpW6amyP8Gx01xrRs8KdiNbbqXfYxkOXplpq1euWbOjw==", + "dev": true, + "dependencies": { + "debug-log": "^1.0.0", + "find-root": "^1.0.0", + "xtend": "^4.0.1" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/pluralize": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-1.2.1.tgz", + "integrity": "sha512-TH+BeeL6Ct98C7as35JbZLf8lgsRzlNJb5gklRIGHKaPkGl1esOKBc5ALUMd+q08Sr6tiEKM+Icbsxg5vuhMKQ==", + "dev": true + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", + "dev": true + }, + "node_modules/progress": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/progress/-/progress-1.1.8.tgz", + "integrity": "sha512-UdA8mJ4weIkUBO224tIarHzuHs4HuYiJvsuGT7j/SPQiUJVjYvNDBIPa0hAorduOfjGohB/qHWRa/lrrWX/mXw==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, + "node_modules/psl": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz", + "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==" + }, + "node_modules/punycode": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz", + "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/qs": { + "version": "6.5.3", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.3.tgz", + "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", + "dev": true, + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/readable-stream/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + }, + "node_modules/readline2": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/readline2/-/readline2-1.0.1.tgz", + "integrity": "sha512-8/td4MmwUB6PkZUbV25uKz7dfrmjYWxsW8DVfibWdlHRk/l/DfHKn4pU+dfcoGLFgWOdyGCzINRQD7jn+Bv+/g==", + "dev": true, + "dependencies": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "mute-stream": "0.0.5" + } + }, + "node_modules/rechoir": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz", + "integrity": "sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==", + "dev": true, + "dependencies": { + "resolve": "^1.1.6" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/remedial": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/remedial/-/remedial-1.0.8.tgz", + "integrity": "sha512-/62tYiOe6DzS5BqVsNpH/nkGlX45C/Sp6V+NtiN6JQNS1Viay7cWkazmRkrQrdFj2eshDe96SIQNIoMxqhzBOg==", + "engines": { + "node": "*" + } + }, + "node_modules/request": { + "version": "2.88.2", + "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", + "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", + "deprecated": "request has been deprecated, see https://github.com/request/request/issues/3142", + "dependencies": { + "aws-sign2": "~0.7.0", + "aws4": "^1.8.0", + "caseless": "~0.12.0", + "combined-stream": "~1.0.6", + "extend": "~3.0.2", + "forever-agent": "~0.6.1", + "form-data": "~2.3.2", + "har-validator": "~5.1.3", + "http-signature": "~1.2.0", + "is-typedarray": "~1.0.0", + "isstream": "~0.1.2", + "json-stringify-safe": "~5.0.1", + "mime-types": "~2.1.19", + "oauth-sign": "~0.9.0", + "performance-now": "^2.1.0", + "qs": "~6.5.2", + "safe-buffer": "^5.1.2", + "tough-cookie": "~2.5.0", + "tunnel-agent": "^0.6.0", + "uuid": "^3.3.2" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/request-promise": { + "version": "4.2.6", + "resolved": "https://registry.npmjs.org/request-promise/-/request-promise-4.2.6.tgz", + "integrity": "sha512-HCHI3DJJUakkOr8fNoCc73E5nU5bqITjOYFMDrKHYOXWXrgD/SBaC7LjwuPymUprRyuF06UK7hd/lMHkmUXglQ==", + "deprecated": "request-promise has been deprecated because it extends the now deprecated request package, see https://github.com/request/request/issues/3142", + "dependencies": { + "bluebird": "^3.5.0", + "request-promise-core": "1.1.4", + "stealthy-require": "^1.1.1", + "tough-cookie": "^2.3.3" + }, + "engines": { + "node": ">=0.10.0" + }, + "peerDependencies": { + "request": "^2.34" + } + }, + "node_modules/request-promise-core": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/request-promise-core/-/request-promise-core-1.1.4.tgz", + "integrity": "sha512-TTbAfBBRdWD7aNNOoVOBH4pN/KigV6LyapYNNlAPA8JwbovRti1E88m3sYAwsLi5ryhPKsE9APwnjFTgdUjTpw==", + "dependencies": { + "lodash": "^4.17.19" + }, + "engines": { + "node": ">=0.10.0" + }, + "peerDependencies": { + "request": "^2.34" + } + }, + "node_modules/require-uncached": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", + "integrity": "sha512-Xct+41K3twrbBHdxAgMoOS+cNcoqIjfM2/VxBF4LL2hVph7YsF8VSKyQ3BDFZwEVbok9yeDl2le/qo0S77WG2w==", + "dev": true, + "dependencies": { + "caller-path": "^0.1.0", + "resolve-from": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/require-uncached/node_modules/resolve-from": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", + "integrity": "sha512-kT10v4dhrlLNcnO084hEjvXCI1wUG9qZLoz2RogxqDQQYy7IxjI/iMUkOtQTNEh6rzHxvdQWHsJyel1pKOVCxg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.2", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.2.tgz", + "integrity": "sha512-Sb+mjNHOULsBv818T40qSPeRiuWLyaGMa5ewydRLFimneixmVy2zdivRl+AF6jaYPC8ERxGDmFSiqui6SfPd+g==", + "dev": true, + "dependencies": { + "is-core-module": "^2.11.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/restore-cursor": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-1.0.1.tgz", + "integrity": "sha512-reSjH4HuiFlxlaBaFCiS6O76ZGG2ygKoSlCsipKdaZuKSPx/+bt9mULkn4l0asVzbEfQQmXRg6Wp6gv6m0wElw==", + "dev": true, + "dependencies": { + "exit-hook": "^1.0.0", + "onetime": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/run-async": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-0.1.0.tgz", + "integrity": "sha512-qOX+w+IxFgpUpJfkv2oGN0+ExPs68F4sZHfaRRx4dDexAQkG83atugKVEylyT5ARees3HBbfmuvnjbrd8j9Wjw==", + "dev": true, + "dependencies": { + "once": "^1.3.0" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/rx-lite": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-3.1.2.tgz", + "integrity": "sha512-1I1+G2gteLB8Tkt8YI1sJvSIfa0lWuRtC8GjvtyPBcLSF5jBCCJJqKrpER5JU5r6Bhe+i9/pK3VMuUcXu0kdwQ==", + "dev": true + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/sax": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", + "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/shelljs": { + "version": "0.7.8", + "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.7.8.tgz", + "integrity": "sha512-/YF5Uk8hcwi7ima04ppkbA4RaRMdPMBfwAvAf8sufYOxsJRtbdoBsT8vGvlb+799BrlGdYrd+oczIA2eN2JdWA==", + "dev": true, + "dependencies": { + "glob": "^7.0.0", + "interpret": "^1.0.0", + "rechoir": "^0.6.2" + }, + "bin": { + "shjs": "bin/shjs" + }, + "engines": { + "iojs": "*", + "node": ">=0.11.0" + } + }, + "node_modules/slice-ansi": { + "version": "0.0.4", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-0.0.4.tgz", + "integrity": "sha512-up04hB2hR92PgjpyU3y/eg91yIBILyjVY26NvvciY3EVVPjybkMszMpXQ9QAkcS3I5rtJBDLoTxxg+qvW8c7rw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "dev": true + }, + "node_modules/sshpk": { + "version": "1.17.0", + "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.17.0.tgz", + "integrity": "sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ==", + "dependencies": { + "asn1": "~0.2.3", + "assert-plus": "^1.0.0", + "bcrypt-pbkdf": "^1.0.0", + "dashdash": "^1.12.0", + "ecc-jsbn": "~0.1.1", + "getpass": "^0.1.1", + "jsbn": "~0.1.0", + "safer-buffer": "^2.0.2", + "tweetnacl": "~0.14.0" + }, + "bin": { + "sshpk-conv": "bin/sshpk-conv", + "sshpk-sign": "bin/sshpk-sign", + "sshpk-verify": "bin/sshpk-verify" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard": { + "version": "8.6.0", + "resolved": "https://registry.npmjs.org/standard/-/standard-8.6.0.tgz", + "integrity": "sha512-kcNkHtA0pNqg2+GZmUoKpvQyXl5/3ADX/cE6n7sTa9qdRhgnbQxqIuMWq+NW9R54VQTYWU7Woytei6PV0s4bQQ==", + "dev": true, + "dependencies": { + "eslint": "~3.10.2", + "eslint-config-standard": "6.2.1", + "eslint-config-standard-jsx": "3.2.0", + "eslint-plugin-promise": "~3.4.0", + "eslint-plugin-react": "~6.7.1", + "eslint-plugin-standard": "~2.0.1", + "standard-engine": "~5.2.0" + }, + "bin": { + "standard": "bin/cmd.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/standard-engine": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/standard-engine/-/standard-engine-5.2.0.tgz", + "integrity": "sha512-0uep1ucc2XrTB84GdzY1yFbFiHehjJrkrMa56YYxWLnlpfCkhvXkhGd++jc2mSzcdUn8OYJ+eDWgd7yBGvDNTg==", + "dev": true, + "dependencies": { + "deglob": "^2.0.0", + "find-root": "^1.0.0", + "get-stdin": "^5.0.1", + "home-or-tmp": "^2.0.0", + "minimist": "^1.1.0", + "pkg-config": "^1.0.1" + } + }, + "node_modules/standard/node_modules/acorn": { + "version": "5.7.4", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.7.4.tgz", + "integrity": "sha512-1D++VG7BhrtvQpNbBzovKNc1FLGGEE/oGe7b9xJm/RFHMBeUaUGpluV9RLjZa47YFdPcDAenEYuq9pQPcMdLJg==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/standard/node_modules/acorn-jsx": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz", + "integrity": "sha512-AU7pnZkguthwBjKgCg6998ByQNIMjbuDQZ8bb78QAFZwPfmKia8AIzgY/gWgqCjnht8JLdXmB4YxA0KaV60ncQ==", + "dev": true, + "dependencies": { + "acorn": "^3.0.4" + } + }, + "node_modules/standard/node_modules/acorn-jsx/node_modules/acorn": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz", + "integrity": "sha512-OLUyIIZ7mF5oaAUT1w0TFqQS81q3saT46x8t7ukpPjMNk+nbs4ZHhs7ToV8EWnLYLepjETXd4XaCE4uxkMeqUw==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/standard/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/standard/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/standard/node_modules/doctrine": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", + "integrity": "sha512-lsGyRuYr4/PIB0txi+Fy2xOMI2dGaTguCaotzFGkVZuKR5usKfcRWIFKNM3QNrU7hh/+w2bwTW+ZeXPK5l8uVg==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2", + "isarray": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/standard/node_modules/eslint": { + "version": "3.10.2", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-3.10.2.tgz", + "integrity": "sha512-GUy8nH40PQxReArM6e1SKbzq6IZ4xxR1AQDc3qOIbOSbUHmgI8BYJ5Sqw5vQwsmAbQvU/gcJ3+cP2j0rrotNxA==", + "dev": true, + "dependencies": { + "babel-code-frame": "^6.16.0", + "chalk": "^1.1.3", + "concat-stream": "^1.4.6", + "debug": "^2.1.1", + "doctrine": "^1.2.2", + "escope": "^3.6.0", + "espree": "^3.3.1", + "estraverse": "^4.2.0", + "esutils": "^2.0.2", + "file-entry-cache": "^2.0.0", + "glob": "^7.0.3", + "globals": "^9.2.0", + "ignore": "^3.2.0", + "imurmurhash": "^0.1.4", + "inquirer": "^0.12.0", + "is-my-json-valid": "^2.10.0", + "is-resolvable": "^1.0.0", + "js-yaml": "^3.5.1", + "json-stable-stringify": "^1.0.0", + "levn": "^0.3.0", + "lodash": "^4.0.0", + "mkdirp": "^0.5.0", + "natural-compare": "^1.4.0", + "optionator": "^0.8.2", + "path-is-inside": "^1.0.1", + "pluralize": "^1.2.1", + "progress": "^1.1.8", + "require-uncached": "^1.0.2", + "shelljs": "^0.7.5", + "strip-bom": "^3.0.0", + "strip-json-comments": "~1.0.1", + "table": "^3.7.8", + "text-table": "~0.2.0", + "user-home": "^2.0.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/standard/node_modules/eslint-config-standard-jsx": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/eslint-config-standard-jsx/-/eslint-config-standard-jsx-3.2.0.tgz", + "integrity": "sha512-K8q9AFljE/JIwhYP0iklqU9J+oUB3Kcd7XwUKCBI4GgMUOP51FVqJhlenf51UPGAv+Q66L3LaqhE4BZDBhWDPQ==", + "dev": true, + "peerDependencies": { + "eslint": ">=3.0.0", + "eslint-plugin-react": ">=2.7.1" + } + }, + "node_modules/standard/node_modules/eslint-plugin-react": { + "version": "6.7.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-6.7.1.tgz", + "integrity": "sha512-XePiF5td1CLgvnIrE+O/xH4fjuqYu1SgSRxyX/Q8KhDQslOiUGaB8++q+i+5ytiLJpxgt+VApXZ11BnkGMtNqA==", + "dev": true, + "dependencies": { + "doctrine": "^1.2.2", + "jsx-ast-utils": "^1.3.3" + }, + "engines": { + "node": ">=0.10" + }, + "peerDependencies": { + "eslint": "^2.0.0 || ^3.0.0" + } + }, + "node_modules/standard/node_modules/espree": { + "version": "3.5.4", + "resolved": "https://registry.npmjs.org/espree/-/espree-3.5.4.tgz", + "integrity": "sha512-yAcIQxtmMiB/jL32dzEp2enBeidsB7xWPLNiw3IIkpVds1P+h7qF9YwJq1yUNzp2OKXgAprs4F61ih66UsoD1A==", + "dev": true, + "dependencies": { + "acorn": "^5.5.0", + "acorn-jsx": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/standard/node_modules/file-entry-cache": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", + "integrity": "sha512-uXP/zGzxxFvFfcZGgBIwotm+Tdc55ddPAzF7iHshP4YGaXMww7rSF9peD9D1sui5ebONg5UobsZv+FfgEpGv/w==", + "dev": true, + "dependencies": { + "flat-cache": "^1.2.1", + "object-assign": "^4.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/flat-cache": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.4.tgz", + "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", + "dev": true, + "dependencies": { + "circular-json": "^0.3.1", + "graceful-fs": "^4.1.2", + "rimraf": "~2.6.2", + "write": "^0.2.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/globals": { + "version": "9.18.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-9.18.0.tgz", + "integrity": "sha512-S0nG3CLEQiY/ILxqtztTWH/3iRRdyBLw6KMDxnKMchrtbj2OFmehVh0WUCfW3DUrIgx/qFrJPICrq4Z4sTR9UQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/standard/node_modules/levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/standard/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", + "dev": true + }, + "node_modules/standard/node_modules/optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "dependencies": { + "deep-is": "~0.1.3", + "fast-levenshtein": "~2.0.6", + "levn": "~0.3.0", + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2", + "word-wrap": "~1.2.3" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/standard/node_modules/prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/standard/node_modules/rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/standard/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/standard/node_modules/strip-json-comments": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-1.0.4.tgz", + "integrity": "sha512-AOPG8EBc5wAikaG1/7uFCNFJwnKOuQwFTpYBdTW6OvWHeZBQBrAA/amefHGrEiOnCPcLFZK6FUPtWVKpQVIRgg==", + "dev": true, + "bin": { + "strip-json-comments": "cli.js" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/standard/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/standard/node_modules/type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/stealthy-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/stealthy-require/-/stealthy-require-1.1.1.tgz", + "integrity": "sha512-ZnWpYnYugiOVEY5GkcuJK1io5V8QmNYChG62gSit9pQVGErXtrKuPC55ITaVSukmMta5qpMU7vqLt2Lnni4f/g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dev": true, + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/string_decoder/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + }, + "node_modules/string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==", + "dev": true, + "dependencies": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string-width/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string-width/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/table": { + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/table/-/table-3.8.3.tgz", + "integrity": "sha512-RZuzIOtzFbprLCE0AXhkI0Xi42ZJLZhCC+qkwuMLf/Vjz3maWpA8gz1qMdbmNoI9cOROT2Am/DxeRyXenrL11g==", + "dev": true, + "dependencies": { + "ajv": "^4.7.0", + "ajv-keywords": "^1.0.0", + "chalk": "^1.1.1", + "lodash": "^4.0.0", + "slice-ansi": "0.0.4", + "string-width": "^2.0.0" + } + }, + "node_modules/table/node_modules/ajv": { + "version": "4.11.8", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-4.11.8.tgz", + "integrity": "sha512-I/bSHSNEcFFqXLf91nchoNB9D1Kie3QKcWdchYUaoIg1+1bdWDkdfdlvdIOJbi9U8xR0y+MWc5D+won9v95WlQ==", + "dev": true, + "dependencies": { + "co": "^4.6.0", + "json-stable-stringify": "^1.0.1" + } + }, + "node_modules/table/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/table/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/table/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/table/node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/table/node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/table/node_modules/string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "dependencies": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/table/node_modules/string-width/node_modules/ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/table/node_modules/string-width/node_modules/strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "dev": true, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/table/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/table/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", + "dev": true + }, + "node_modules/tough-cookie": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", + "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", + "dependencies": { + "psl": "^1.1.28", + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/tunnel-agent": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", + "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", + "dependencies": { + "safe-buffer": "^5.0.1" + }, + "engines": { + "node": "*" + } + }, + "node_modules/tweetnacl": { + "version": "0.14.5", + "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", + "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==" + }, + "node_modules/type": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz", + "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==", + "dev": true + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==", + "dev": true + }, + "node_modules/uniq": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", + "integrity": "sha512-Gw+zz50YNKPDKXs+9d+aKAjVwpjNwqzvNpLigIruT4HA9lMZNdMqs9x07kKHB/L9WRzqp4+DlTU5s4wG2esdoA==", + "dev": true + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/user-home": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/user-home/-/user-home-2.0.0.tgz", + "integrity": "sha512-KMWqdlOcjCYdtIJpicDSFBQ8nFwS2i9sslAd6f4+CBGcU4gist2REnr2fxj2YocvJFxSF3ZOHLYLVZnUxv4BZQ==", + "dev": true, + "dependencies": { + "os-homedir": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "dev": true + }, + "node_modules/uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "deprecated": "Please upgrade to version 7 or higher. Older versions may use Math.random() in certain circumstances, which is known to be problematic. See https://v8.dev/blog/math-random for details.", + "bin": { + "uuid": "bin/uuid" + } + }, + "node_modules/verror": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", + "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==", + "engines": [ + "node >=0.6.0" + ], + "dependencies": { + "assert-plus": "^1.0.0", + "core-util-is": "1.0.2", + "extsprintf": "^1.2.0" + } + }, + "node_modules/verror/node_modules/core-util-is": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", + "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==" + }, + "node_modules/web-streams-polyfill": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.2.1.tgz", + "integrity": "sha512-e0MO3wdXWKrLbL0DgGnUV7WHVuw9OUvL4hjgnPkIeEvESk74gAITi5G606JtZPp39cd8HA9VQzCIvA49LpPN5Q==", + "engines": { + "node": ">= 8" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", + "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "node_modules/write": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", + "integrity": "sha512-CJ17OoULEKXpA5pef3qLj5AxTJ6mSt7g84he2WIskKwqFO4T97d5V7Tadl0DYDk7qyUOQD5WlUlOMChaYrhxeA==", + "dev": true, + "dependencies": { + "mkdirp": "^0.5.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/xml-to-json-promise": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/xml-to-json-promise/-/xml-to-json-promise-0.0.3.tgz", + "integrity": "sha512-632cfTj1PZPmFicwq3y7uFOf563DkcFIJpS3jz4c/LTDxjRiHQP1qGIvYUAQ9tcKUEyw05NXmxj/nbBq+DWQXw==", + "dependencies": { + "native-promise-only": "^0.8.1", + "object-assign": "^4.0.1", + "xml2js": "~0.4.16" + } + }, + "node_modules/xml2js": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", + "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", + "dependencies": { + "sax": ">=0.6.0", + "xmlbuilder": "~11.0.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/xmlbuilder": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", + "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "dev": true, + "engines": { + "node": ">=0.4" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} From e7f78ff797f48ebc0d1415ad6cacac5b88337b72 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Thu, 10 Aug 2023 22:44:31 +0200 Subject: [PATCH 24/32] fix linting issues --- index.js | 4 +- src/dect.js | 10 ++--- src/fon.js | 25 ++++++------ src/login.js | 4 +- src/request.js | 69 ++++++++++++++++----------------- src/wlan.js | 102 ++++++++++++++++++++++++------------------------- 6 files changed, 103 insertions(+), 111 deletions(-) diff --git a/index.js b/index.js index 227f3ea..24ff2fb 100644 --- a/index.js +++ b/index.js @@ -24,11 +24,11 @@ const fritz = Object.assign( ) process.on('unhandledRejection', function (reason, r) { - console.log('\x1b[31m[FritzBox.js] Encountered unhandled Promise rejection.\n' + + console.error('\x1b[31m[FritzBox.js] Encountered unhandled Promise rejection.\n' + 'The script has been stopped. See the full stack trace below for more information.\n' + 'If you think this has to do with FritzBox.js directly,\n' + 'please do no hesitate to open an issue on GitHub.\n') - console.log(reason, r) + console.error(reason, r) process.exit(1) }) diff --git a/src/dect.js b/src/dect.js index 8f2199d..f651523 100644 --- a/src/dect.js +++ b/src/dect.js @@ -40,8 +40,9 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { let response if (version >= 750) { - //TODO when someone gets a smartswitch - return { error: 'The implementation for FritzOS!7.50 is not available '} + // TODO when someone gets a smartswitch + + return {error: 'The implementation for FritzOS!7.50 is not available '} } else if (version >= 683) { // Post 06.83 uses a POST request. @@ -58,10 +59,9 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { cmdValue: value, deviceId: deviceId } - - + const headers = { - "Content-Type": "multipart/form-data", + 'Content-Type': 'multipart/form-data' } response = await fritzRequest.request(path, 'POST', options, form, headers) } else { diff --git a/src/fon.js b/src/fon.js index 92e209e..8f1e605 100644 --- a/src/fon.js +++ b/src/fon.js @@ -71,7 +71,6 @@ fritzFon.getTamMessages = async (options) => { tamMessages.push(calls[call].tam_data) } } - } else if (version >= 683) { /* The following works with Fritz!Box OS 6.83 and newer. */ @@ -89,7 +88,7 @@ fritzFon.getTamMessages = async (options) => { ajax_id: 1234 } const headers = { - "Content-Type": "application/x-www-form-urlencoded", + 'Content-Type': 'application/x-www-form-urlencoded' } const response = await fritzRequest.request(path, 'POST', options, form, headers) @@ -131,11 +130,11 @@ fritzFon.downloadTamMessage = async (messagePath, localPath, options) => { if (version.error) return version - if (version >= 750) { + if (version >= 750) { const path = '/cgi-bin/luacgi_notimeout?' const param = { script: '/lua/photo.lua', - myabfile: messagePath, + myabfile: messagePath } options.noAuth = false const response = await fritzRequest.request(path, 'GET', options, param) @@ -151,20 +150,18 @@ fritzFon.downloadTamMessage = async (messagePath, localPath, options) => { const path = '/myfritz/cgi-bin/luacgi_notimeout' + '?cmd=tam&script=/http_file_download.lua' + '&cmd_files=' + messagePath - + const response = await fritzRequest.request(path, 'GET', options) if (response.error) return response if (response.headers['content-type'] !== 'audio/x-wav') { - return { error: { message: 'Did not receive wav audio file', raw: response } } + return { error: { message: 'Did not receive wav audio file', raw: response } } } return { message: 'Saved tam message to ' + localPath } - } } - - +} /** * Mark a message as read. @@ -184,7 +181,7 @@ fritzFon.markTamMessageAsRead = async (messageId, options, tamId = 0) => { if (response.error) return response - if (response.body !== '{"state":1,"cur_idx":1}') { + if (response.body !== "{'state':1,'cur_idx':1}") { return { error: { message: 'Message not marked as read.', raw: response.body } } } @@ -236,7 +233,7 @@ fritzFon.getActiveCalls = async (options) => { } const headers = { - "Content-Type": "application/x-www-form-urlencoded", + 'Content-Type': 'application/x-www-form-urlencoded' } const response = await fritzRequest.request('/data.lua', 'POST', options, params, headers) @@ -268,9 +265,9 @@ fritzFon.getPhonebook = async (phonebookId = 0, options) => { PhonebookExport: '' } const path = '/cgi-bin/firmwarecfg' - + const headers = { - "Content-Type": "multipart/form-data", + 'Content-Type': 'multipart/form-data' } const response = await fritzRequest.request(path, 'POST', options, formData, headers) @@ -424,4 +421,4 @@ fritzFon.CallMonitor = CallMonitor // Export fritzFon. -export default fritzFon +export default fritzFon diff --git a/src/login.js b/src/login.js index 2914fcb..a78ca38 100644 --- a/src/login.js +++ b/src/login.js @@ -31,13 +31,13 @@ fritzLogin.getSessionId = async (options) => { } const headers = { - "Content-Type": "application/x-www-form-urlencoded", + 'Content-Type': 'application/x-www-form-urlencoded' } const responseChallenge = await fritzRequest.request(path, 'POST', options, params, headers) if (responseChallenge.error) return responseChallenge - + // Extract the session ID. const sessionIdMatch = responseChallenge.body.match('(.*?)') if (!sessionIdMatch || !sessionIdMatch[1]) { diff --git a/src/request.js b/src/request.js index dfa3a50..091aafe 100644 --- a/src/request.js +++ b/src/request.js @@ -3,17 +3,17 @@ * @ignore */ -import fetch from 'node-fetch'; -import https from 'https'; -import FormData from 'form-data'; +import fetch from 'node-fetch' +import https from 'https' +import FormData from 'form-data' const httpsAgent = new https.Agent({ - //Disable SSL verification since the Fritz.box endpoint is not secure enough - //The certificate uses a weak Diffie-Hellman key and is vulnerable to a Logjam Attack - rejectUnauthorized: false, - }); + // Disable SSL verification since the Fritz.box endpoint is not secure enough + // The certificate uses a weak Diffie-Hellman key and is vulnerable to a Logjam Attack + rejectUnauthorized: false +}) let fritzRequest = {} -export default fritzRequest +export default fritzRequest /** * Send a request to the Fritz!Box. * @@ -27,7 +27,7 @@ export default fritzRequest * @return {Object} Request response object */ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => { - console.log("PATH: " + path) + console.debug('PATH: ' + path) options.protocol = options.protocol || 'https' // Make sure we have the required options. @@ -46,7 +46,7 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => if (typeof options.removeSidFromUri === 'undefined') { options.removeSidFromUri = false } - + if (options.sid && options.removeSidFromUri !== true && options.noAuth !== true) { // Add SID to path if one has been given to us. body['sid'] = options.sid @@ -55,52 +55,49 @@ fritzRequest.request = async (path, method, options, body = {}, headers = {}) => if (headers['Content-Type'] === 'application/x-www-form-urlencoded') { const encodedForm = [] for (const property in body) { - const encodedKey = encodeURIComponent(property); - const encodedValue = encodeURIComponent(body[property]); - encodedForm.push(`${encodedKey}=${encodedValue}`); + const encodedKey = encodeURIComponent(property) + const encodedValue = encodeURIComponent(body[property]) + encodedForm.push(`${encodedKey}=${encodedValue}`) } - body = encodedForm.join("&") + body = encodedForm.join('&') } else if (headers['Content-Type'] === 'multipart/form-data') { const form = new FormData() for (const property in body) { form.append(property, body[property]) } body = form - } - - + } - if (method === 'GET'){ + if (method === 'GET') { for (const key in body) { if (!path.endsWith('&') || !path.endsWith('?')) { path += '&' } path += `${key}=${body[key]}` } - body = undefined; + body = undefined } // Set the options for the request. const uri = `${options.protocol}://${options.server}${path}` - console.log('uri: ', uri, 'headers: ', headers, 'method: ', method, 'body: ', body) + console.debug('uri: ', uri, 'headers: ', headers, 'method: ', method, 'body: ', body) // Execute HTTP(S) request. - const fetchResponse = await fetch(uri, { - headers: headers, - method: method || 'GET', - body: body, - agent: httpsAgent - }) - if (fetchResponse.ok) { - const response = { - body: await fetchResponse.text(), - headers: fetchResponse.headers, - } - console.log('response fetch', response) - return response - } else { - return fritzRequest.findFailCause(fetchResponse) + const fetchResponse = await fetch(uri, { + headers: headers, + method: method || 'GET', + body: body, + agent: httpsAgent + }) + if (fetchResponse.ok) { + const response = { + body: await fetchResponse.text(), + headers: fetchResponse.headers } - + console.debug('response fetch', response) + return response + } else { + return fritzRequest.findFailCause(fetchResponse) + } } /** diff --git a/src/wlan.js b/src/wlan.js index 9e4e8d8..1500c9e 100644 --- a/src/wlan.js +++ b/src/wlan.js @@ -10,71 +10,69 @@ const fritzWlan = {} * @return {Array} The connected devices to the LAN as an array. */ fritzWlan.getDevices = async (options) => { + const path = `/data.lua` + const params = { + xhr: 1, + page: 'netDev', + xhrId: 'cleanup', + useajax: '1' + } - const path = `/data.lua` - const params = { - xhr:1, - page: 'netDev', - xhrId: 'cleanup', - useajax: '1', - } - - const headers = { - 'Content-Type': 'application/x-www-form-urlencoded' - } + const headers = { + 'Content-Type': 'application/x-www-form-urlencoded' + } - const response = await fritzRequest.request(path, 'POST', options, params, headers) + const response = await fritzRequest.request(path, 'POST', options, params, headers) - if (response.error) return response + if (response.error) return response - return JSON.parse(response.body).data + return JSON.parse(response.body).data } fritzWlan.getDeviceByName = async (deviceName, options) => { - const deviceData = await fritzWlan.getDevices(options); - const activeDevices = deviceData.active - const passiveDevices = deviceData.passiv + const deviceData = await fritzWlan.getDevices(options) + const activeDevices = deviceData.active + const passiveDevices = deviceData.passiv - for (let activeDevice in activeDevices) { - if (activeDevices[activeDevice].name === deviceName) { - return activeDevices[activeDevice] - } + for (let activeDevice in activeDevices) { + if (activeDevices[activeDevice].name === deviceName) { + return activeDevices[activeDevice] } - for (let passivDevice in passiveDevices) { - if (activeDevices[passivDevice].name === deviceName){ - return activeDevices[passivDevice] - } + } + for (let passivDevice in passiveDevices) { + if (activeDevices[passivDevice].name === deviceName) { + return activeDevices[passivDevice] } + } - return { - error : { message: 'Device could not be found', raw: deviceData} - } + return { + error: {message: 'Device could not be found', raw: deviceData} + } } fritzWlan.resetDevice = async (deviceName, options) => { - const device = await fritzWlan.getDeviceByName(deviceName, options) - - const path = `/data.lua` - const params = { - xhr: "1", - dev_name: device.name, - internetdetail: "unlimited", - allow_pcp_and_upnp: "off", - static_dhcp: "off", - dev: device.UID, - btn_reset_dev: "", - page: "edit_device" - } - - const headers = { - 'Content-Type': 'application/x-www-form-urlencoded' - } - - const response = await fritzRequest.request(path, 'POST', options, params, headers) - - if (response.error) return response - - return "" + const device = await fritzWlan.getDeviceByName(deviceName, options) + const path = `/data.lua` + const params = { + xhr: '1', + dev_name: device.name, + internetdetail: 'unlimited', + allow_pcp_and_upnp: 'off', + static_dhcp: 'off', + dev: device.UID, + btn_reset_dev: '', + page: 'edit_device' + } + + const headers = { + 'Content-Type': 'application/x-www-form-urlencoded' + } + + const response = await fritzRequest.request(path, 'POST', options, params, headers) + + if (response.error) return response + + return '' } -export default fritzWlan \ No newline at end of file +export default fritzWlan From 715dc2e2134a40c2a4c3172a977f65ddf1f181ed Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Thu, 10 Aug 2023 22:46:09 +0200 Subject: [PATCH 25/32] updated test script --- package.json | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/package.json b/package.json index 177d88b..b8adca1 100644 --- a/package.json +++ b/package.json @@ -28,8 +28,7 @@ }, "main": "index.js", "scripts": { - "test": "standard", - "local-test": "test/ci.sh", + "test": "test/ci.sh", "build-docs": "documentation build --github --sort-order=alpha --output=docs/api --format=html src/**", "watch-docs": "documentation serve --watch --sort-order=alpha src/**" }, From 50834d87592afccdecd257c288d97d11859f7c9b Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Thu, 10 Aug 2023 22:53:28 +0200 Subject: [PATCH 26/32] remove tests for github actions --- .github/workflows/npm-publish.yml | 1 - 1 file changed, 1 deletion(-) diff --git a/.github/workflows/npm-publish.yml b/.github/workflows/npm-publish.yml index f96a9dd..1ea392c 100644 --- a/.github/workflows/npm-publish.yml +++ b/.github/workflows/npm-publish.yml @@ -16,7 +16,6 @@ jobs: with: node-version: 18 - run: npm ci - - run: npm test publish-npm: needs: build From 1b36e9c3a0834d5fd80aa1a84bdc45b9060b751a Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Thu, 28 Sep 2023 11:32:35 +0200 Subject: [PATCH 27/32] fix getSmartDevice --- src/dect.js | 20 +++++++++++++------- test/smartdevices.js | 8 ++++---- 2 files changed, 17 insertions(+), 11 deletions(-) diff --git a/src/dect.js b/src/dect.js index f651523..351fcaf 100644 --- a/src/dect.js +++ b/src/dect.js @@ -16,14 +16,20 @@ import fritzSystem from './system.js' * @return {Array} An array of all found smart devices. */ fritzDect.getSmartDevices = async (options) => { - const path = '/myfritz/areas/homeauto.lua?ajax_id=1&cmd=getData' - const response = await fritzRequest.request(path, 'GET', options) + const path = '/data.lua' + const body = { + xhr: '1', + page: 'sh_dev', + xhrid: 'all' + } + const headers = { + 'Content-Type': 'application/x-www-form-urlencoded' + } + const response = await fritzRequest.request(path, 'POST', options, body, headers) if (response.error) return response - - console.log(response) - - return JSON.parse(response).devices + + return JSON.parse(response.body).data } /** @@ -38,7 +44,7 @@ fritzDect.toggleSwitch = async (deviceId, value, options) => { if (version.error) return version let response - + console.log("Fritzbox version is: " + version) if (version >= 750) { // TODO when someone gets a smartswitch diff --git a/test/smartdevices.js b/test/smartdevices.js index 5b4a60f..7a864e6 100644 --- a/test/smartdevices.js +++ b/test/smartdevices.js @@ -2,12 +2,12 @@ import fritz from '../index.js' import pckJson from '../package.json' assert {type:'json'} async function devices () { - const devices = await fritz.getSmartDevices(pckJson.options) - if (devices.error) { - console.log('Error:', devices.error.message) + const data = await fritz.getSmartDevices(pckJson.options) + if (data.error) { + console.log('Error:', data.error.message) process.exit(1) } - console.log('Got ' + devices.length + ' DECT Smart Home devices.') + console.log('Got ' + data.devices.length + ' DECT Smart Home devices.') } devices() From d681208b982fa08c31062710f530efd36af9a80a Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 29 Sep 2023 08:58:16 +0200 Subject: [PATCH 28/32] fix error handling for resetDevice --- src/wlan.js | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/wlan.js b/src/wlan.js index 1500c9e..15cc67c 100644 --- a/src/wlan.js +++ b/src/wlan.js @@ -52,6 +52,11 @@ fritzWlan.getDeviceByName = async (deviceName, options) => { fritzWlan.resetDevice = async (deviceName, options) => { const device = await fritzWlan.getDeviceByName(deviceName, options) + + if (device.error) { + return device; + } + const path = `/data.lua` const params = { xhr: '1', From 9592cca5b261e8d03e6bc8d9b3e4e903c6b3d614 Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 29 Sep 2023 09:12:13 +0200 Subject: [PATCH 29/32] fix resetDevice response --- src/wlan.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/wlan.js b/src/wlan.js index 15cc67c..02e572a 100644 --- a/src/wlan.js +++ b/src/wlan.js @@ -77,7 +77,7 @@ fritzWlan.resetDevice = async (deviceName, options) => { if (response.error) return response - return '' + return JSON.parse(response.body) } export default fritzWlan From 1b6d6ef66ddc6db4e26db1576c2988522e22818d Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Fri, 29 Sep 2023 09:18:49 +0200 Subject: [PATCH 30/32] fix github action release to npm --- .github/workflows/npm-publish.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/npm-publish.yml b/.github/workflows/npm-publish.yml index 1ea392c..232c629 100644 --- a/.github/workflows/npm-publish.yml +++ b/.github/workflows/npm-publish.yml @@ -24,9 +24,9 @@ jobs: - uses: actions/checkout@v3 - uses: actions/setup-node@v3 with: - node-version: 16 + node-version: 18 registry-url: https://registry.npmjs.org/ - run: npm ci - run: npm publish env: - NODE_AUTH_TOKEN: ${{secrets.npm_token}} + NODE_AUTH_TOKEN: ${{secrets.NPM_SECRET_NEO_FRITZBOX}} From dd10476f599ad81729eaa5c2fe4f3cb76fb32bfd Mon Sep 17 00:00:00 2001 From: Fiwel Date: Fri, 29 Sep 2023 09:53:29 +0200 Subject: [PATCH 31/32] Update package.json --- package.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index b8adca1..138a0aa 100644 --- a/package.json +++ b/package.json @@ -50,9 +50,9 @@ }, "options": { "protocol": "https", - "server": "192.168.178.1", - "username": "fritz0275", - "password": "acht1714", + "server": "fritz.box", + "username": "", + "password": "", "callmonitorport": "1012", "debug": false }, From e69d243b3e45be7525fc8ee99cea25677165761a Mon Sep 17 00:00:00 2001 From: MehdiAyadi Date: Wed, 4 Oct 2023 13:07:49 +0200 Subject: [PATCH 32/32] update version package.json --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 138a0aa..d230d35 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "neofritzbox.js", - "version": "2.1.1", + "version": "2.1.3", "author": "@fiwel00", "description": "The second most powerful Fritz!Box API for NodeJS.", "license": "MIT",