diff --git a/.c8rc.json b/.c8rc.json new file mode 100644 index 0000000..52f22b5 --- /dev/null +++ b/.c8rc.json @@ -0,0 +1,8 @@ +{ + "src": "./src", + "exclude": ["**/bot/lib", "__mocks__", "**/mock"], + "reporter": ["html"], + "report-dir": "./coverage", + "check-coverage": true, + "lines": 95 +} diff --git a/.env.example b/.env.example deleted file mode 100644 index 92b91de..0000000 --- a/.env.example +++ /dev/null @@ -1,12 +0,0 @@ -######DATABASE: none, mysql, dialogflow - -DEFAULT_MESSAGE=true -SAVE_MEDIA=true -PORT=3000 -DATABASE=none -LANGUAGE=es -SQL_HOST= -SQL_USER= -SQL_PASS= -SQL_DATABASE= -MULTI_DEVICE=false \ No newline at end of file diff --git a/.eslintrc.js b/.eslintrc.js new file mode 100644 index 0000000..ec92cad --- /dev/null +++ b/.eslintrc.js @@ -0,0 +1,14 @@ +module.exports = { + env: { + browser: true, + commonjs: true, + es2021: true, + node: true, + }, + extends: 'eslint:recommended', + overrides: [], + parserOptions: { + ecmaVersion: 'latest', + }, + rules: {}, +} diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..fa3a7b0 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,26 @@ +name: Test / Coverage + +on: + push: + branches: [feature/monorepo] + pull_request: + branches: [main] + +jobs: + build: + runs-on: ubuntu-latest + + strategy: + matrix: + node-version: [16.x] + + steps: + - uses: actions/checkout@v3 + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v3 + with: + node-version: ${{ matrix.node-version }} + - run: npm install + - run: npm run build --if-present + - run: npm run test.unit + - run: npm run test.coverage diff --git a/.gitignore b/.gitignore index b38ba23..ce3a630 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,7 @@ /node_modules -/node_modules/* +/packages/*/node_modules +/packages/*/dist +/packages/*/docs/dist session.json chats/* !chats/.gitkeep @@ -9,4 +11,16 @@ mediaSend/* !mediaSend/.gitkeep !mediaSend/nota-de-voz.mp3 .env -.wwebjs_auth \ No newline at end of file +.wwebjs_auth +packages/cli/config.json +config.json +coverage/ +*.lcov +log +lib +tmp/ +.yarn/* +.fleet/ +example-app/ +qr.svg +package-lock.json \ No newline at end of file diff --git a/.husky/commit-msg b/.husky/commit-msg new file mode 100644 index 0000000..80416c7 --- /dev/null +++ b/.husky/commit-msg @@ -0,0 +1,4 @@ +#!/usr/bin/env sh +. "$(dirname -- "$0")/_/husky.sh" + +npx --no-install commitlint --edit "$1" diff --git a/.husky/pre-commit b/.husky/pre-commit new file mode 100644 index 0000000..bc41446 --- /dev/null +++ b/.husky/pre-commit @@ -0,0 +1,4 @@ +#!/usr/bin/env sh +. "$(dirname -- "$0")/_/husky.sh" + +yarn run format:check && yarn run format:write && git add . diff --git a/.husky/pre-push b/.husky/pre-push new file mode 100644 index 0000000..879e935 --- /dev/null +++ b/.husky/pre-push @@ -0,0 +1,4 @@ +#!/usr/bin/env sh +. "$(dirname -- "$0")/_/husky.sh" + +npm run test diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000..4f7ce3d --- /dev/null +++ b/.prettierignore @@ -0,0 +1,9 @@ +packages/**/lib +packages/docs +**/.git +**/.svn +**/.hg +**/node_modules +*.mjs +*.cjs +*.md \ No newline at end of file diff --git a/.prettierrc.json b/.prettierrc.json index 46b134b..e74ed9f 100644 --- a/.prettierrc.json +++ b/.prettierrc.json @@ -1 +1,6 @@ - \ No newline at end of file +{ + "trailingComma": "es5", + "tabWidth": 4, + "semi": false, + "singleQuote": true +} diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..553081c --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,15 @@ +{ + // Use IntelliSense para saber los atributos posibles. + // Mantenga el puntero para ver las descripciones de los existentes atributos. + // Para más información, visite: https://go.microsoft.com/fwlink/?linkid=830387 + "version": "0.2.0", + "configurations": [ + { + "type": "node", + "request": "launch", + "name": "Iniciar el programa", + "skipFiles": ["/**"], + "program": "${workspaceFolder}\\example-app\\app.js" + } + ] +} diff --git a/.yarnrc.yml b/.yarnrc.yml new file mode 100644 index 0000000..d0d772e --- /dev/null +++ b/.yarnrc.yml @@ -0,0 +1 @@ +yarnPath: .yarn/releases/yarn-3.3.0.cjs diff --git a/CHANGELOG.md b/CHANGELOG.md index cdf499c..e69de29 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,14 +0,0 @@ -#### Actualización 14 Ene 2022 -- npm update -- remove ora and chalk -- add env -- add mysql -- add dialogflow -- add scan qr from webpage -- update route with middleware -- fix send message to story -- external download -- easy deploy heroku -- add support for ubuntu/linux - -https://stackoverflow.com/questions/51855169/dialogflow-403-iam-permission-dialogflow-sessions-detectintent \ No newline at end of file diff --git a/EXAMPLE.md b/EXAMPLE.md new file mode 100644 index 0000000..17e38ca --- /dev/null +++ b/EXAMPLE.md @@ -0,0 +1,88 @@ + +```js + +const { + createBot, + createProvider, + createFlow, + addKeyword, + toSerialize, +} = require('@bot-whatsapp/bot') + +const WebWhatsappProvider = require('@bot-whatsapp/provider/web-whatsapp') +const MongoAdapter = require('@bot-whatsapp/database/mongo') + +const flowArepa1 = toSerialize( + addKeyword(['1', 'AREPA14']) + .addAnswer('Esta es una arepa calificada ⭐⭐⭐⭐⭐') + .addAnswer(['Ingredientes:', '10g Aguacate', '20g Huevo'].join('\n')) + .toJson() +) + +const flowArepa2_2 = toSerialize( + addKeyword('SI').addAnswer('te pongo huevo de mentira!').toJson() +) + +const flowArepa2 = toSerialize( + addKeyword(['arepa2']) + .addAnswer('Esta es una arepa calificada ⭐⭐⭐⭐') + .addAnswer( + ['Ingredientes:', '10g perico', '20g huevo', '10g queso'].join('\n') + ) + .addAnswer( + 'Eres Vegano SI o NO', + { + capture: true, + }, + null, + [...flowArepa2_2] + ) + .toJson() +) + +const flowArepa3 = toSerialize( + addKeyword(['arepa3']) + .addAnswer('Esta es una arepa calificada LAMEJOR ⭐⭐⭐⭐⭐') + .toJson() +) + +//////////////--MENU--PRINCIPAL--////////////////// + +const flujoMenuArepa = addKeyword(['hola', 'ola', 'buenos']) + .addAnswer('Bienvenido "Arepera Aji Picante 🤯🚀😅"') + .addAnswer( + [ + 'El menú de hoy es el siguiente:', + '👉 [1 -AREPA14] - Arepa tradicional con Aguacate y Huevo', + '👉 [arepa2] - Arepa rellena de perico y huevo con un toque de queso', + '👉 [arepa3] - Rellena de Jamon y Queso', + ].join('\n') + ) + .addAnswer( + 'Esperando respuesta...', + { + capture: true, + }, + () => { + console.log('Enviar un mail!') + }, + [...flowArepa1, ...flowArepa2, ...flowArepa3] + ) + .addAnswer('Gracias!') + +const main = async () => { + const adapterDB = new MongoAdapter() + const adapterFlow = createFlow([flujoMenuArepa]) + const adapterProvider = createProvider(WebWhatsappProvider) + + createBot({ + flow: adapterFlow, + provider: adapterProvider, + database: adapterDB, + }) +} + +main() + + +``` diff --git a/README.md b/README.md index 82ad962..c3a9dac 100644 --- a/README.md +++ b/README.md @@ -1,154 +1,20 @@ -## Chatbot Whatsapp (OpenSource) -#### Actualizado Enero 2022 - -El siguiente proyecto se realizó con fines educativos para el canal de [Youtube (Leifer Mendez)](https://www.youtube.com/channel/UCgrIGp5QAnC0J8LfNJxDRDw?sub_confirmation=1) donde aprendemos a crear y implementar un chatbot increíble usando [node.js](https://codigoencasa.com/tag/nodejs/) además le agregamos inteligencia artificial gracias al servicio de __dialogflow__. - -[![Video](https://i.giphy.com/media/OBDi3CXC83WkNeLEZP/giphy.webp)](https://youtu.be/5lEMCeWEJ8o) - -### ATENCION 1 🔴 -> 💥💥 Si te aparece el Error Multi-device es porque tienes la cuenta de whatsapp afiliada al modo "BETA de Multi dispositivo" por el momento no se tiene soporte para esas personas si tu quieres hacer uso de este __BOT__ debes de salir del modo BETA y intentarlo de la manera tradicional - -### ATENCION 2 🔴 -> El core de whatsapp esta en constante actualizaciones por lo cual siempre revisa la ultima fecha de la actualizacion -> [VER](https://github.com/leifermendez/bot-whatsapp/commits/main) - - -#### Acceso rápido -> Si tienes una cuenta en __heroku__ puedes desplegar este proyecto con (1 click) - -[![Deploy](https://www.herokucdn.com/deploy/button.svg)](https://heroku.com/deploy?template=https://github.com/leifermendez/bot-whatsapp) - -> Comprarme un cafe! - -[![Comprar](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/leifermendez) - -#### Actualización - -| Feature | Status | -| ------------- | ------------- | -| Dialogflow | ✅ | -| MySQL | ✅ | -| JSON File | ✅ | -| QR Scan (route) | ✅ | -| Easy deploy heroku | ✅ | -| Buttons | ✅ | -| Send Voice Note | ✅ | -| Add support ubuntu/linux | ✅ | - -## Requisitos -- node v14 o superior -- VSCode (Editor de codigo) [Descargar](https://code.visualstudio.com/download) -- MySql (opcional) solo aplica si vas a usar el modo 'mysql' [sql-bot.sql migración](https://github.com/leifermendez/bot-whatsapp/blob/main/sql-bot.sql) -- Dialogflow (opcional) solo aplica si vas a usar el modo 'dialogflow' - -### (Nuevo) Botones - -[![btn](https://i.imgur.com/W7oYlSu.png)](https://youtu.be/5lEMCeWEJ8o) - -> Implementar los botones solo necesitas hacer uso del metodo __sendMessageButton__ que se encuentra dentro `./controllers/send` dejo un ejemplo de como usarlo. -[Ver implementación](https://github.com/leifermendez/bot-whatsapp/blob/main/app.js#L123) - -``` javascript -const { sendMessageButton } = require('./controllers/send') - -await sendMessageButton( - { - "title":"¿Que te interesa ver?", - "message":"Recuerda todo este contenido es gratis y estaria genial que me siguas!", - "footer":"Gracias", - "buttons":[ - {"body":"😎 Cursos"}, - {"body":"👉 Youtube"}, - {"body":"😁 Telegram"} - ] - } -) - -``` - -## Notas de Voz -[![voice note](https://i.imgur.com/zq6xYDp.png)](https://i.imgur.com/zq6xYDp.png) - -> Se pueden enviar notas de voz con formato nativo para que no se vea como reenviado. En este ejemplo enviare el archivo __PTT-20220223-WA0000.opus__ que se encuentra dentro de la carpeta de __/mediaSend__ - -``` javascript -const { sendMediaVoiceNote } = require('./controllers/send') - -await sendMediaVoiceNote(client, from, 'PTT-20220223-WA0000.opus') - -``` - -## Instruciones -__Descargar o Clonar repositorio__ -![](https://i.imgur.com/dSpUbFz.png) - -__Usas ¿Ubuntu / Linux?__ -> Asegurate de instalar los siguientes paquetes -``` -sudo apt-get install -y libgbm-dev -sudo apt install -y gconf-service libasound2 libatk1.0-0 libc6 libcairo2 libcups2 libdbus-1-3 libexpat1 libfontconfig1 libgcc1 libgconf-2-4 libgdk-pixbuf2.0-0 libglib2.0-0 libgtk-3-0 libnspr4 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 libxcomposite1 libxcursor1 libxdamage1 libxext6 libxfixes3 libxi6 libxrandr2 libxrender1 libxss1 libxtst6 ca-certificates fonts-liberation libappindicator1 libnss3 lsb-release xdg-utils wget -``` - -__Instalar dependencias (npm install)__ -> Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando - -`npm install` - -![](https://i.imgur.com/BJuMjGR.png) - -__Configurar .env__ -> Con el editor de texto crea un archivo `.env` el cual debes de guiarte del archivo `.env.example` -[Ver video explicando](https://youtu.be/5lEMCeWEJ8o?t=381) -``` -######DATABASE: none, mysql, dialogflow - -DEFAULT_MESSAGE=true -SAVE_MEDIA=true -PORT=3000 -DATABASE=none -LANGUAGE=es -SQL_HOST= -SQL_USER= -SQL_PASS= -SQL_DATABASE= -``` - -![](https://i.imgur.com/9poNnW0.png) - -__Ejecutar el script__ -> Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando -`npm run start` - -![](https://i.imgur.com/eMkBkuJ.png) - -__Whatsapp en tu celular__ -> Ahora abre la aplicación de Whatsapp en tu dispositivo y escanea el código QR - -Visitar la pagina -`http://localhost:3000/qr` - -![](https://i.imgur.com/Q3JEDlP.png) - -__Listo 😎__ -> Cuando sale este mensaje tu BOT está __listo__ para trabajar! -![](https://i.imgur.com/eoJ4Ruk.png) - -# ¿Quieres ver como se creó? 🤖 -- [Ver Video 1](https://www.youtube.com/watch?v=A_Xu0OR_HkE) -- [¿Como instalarlo? (Actulización)](https://youtu.be/5lEMCeWEJ8o) - -## ¿Como usarlo el chatbot de whatsapp? -> Escribe un mensaje al whatsapp que vinculaste con tu BOT - -![](https://i.imgur.com/OSUgljQ.png) - -> Ahora deberías obtener un arespuesta por parte del BOT como la siguiente, ademas de esto tambien se crea un archivo excel -con el historial de conversación con el número de tu cliente - -![](https://i.imgur.com/lrMLgR8.png) -![](https://i.imgur.com/UYcoUSV.png) - -## Preguntar al BOT -> Puedes interactuar con el bot ejemplo escribele __hola__ y el bot debe responderte! - -![](https://i.imgur.com/cNAS51I.png) +[![Test / Coverage](https://github.com/leifermendez/bot-whatsapp/actions/workflows/ci.yml/badge.svg)](https://github.com/leifermendez/bot-whatsapp/actions/workflows/ci.yml) +[![Commitizen friendly](https://img.shields.io/badge/commitizen-friendly-brightgreen.svg)](http://commitizen.github.io/cz-cli/) +-------- +🦊 Documentación: [https://bot-whatsapp.pages.dev/](https://bot-whatsapp.pages.dev/) +Video como hacer PR: https://youtu.be/Lxt8Acob6aU + +- [ ] Evitar dependencias + + +**Comunidad** + +> Forma parte de este proyecto. + +- [Discord](https://link.codigoencasa.com/DISCORD) +- [Twitter](https://twitter.com/leifermendez) +- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR) +- [Telegram](https://t.me/leifermendez) + + + \ No newline at end of file diff --git a/TODO.md b/TODO.md new file mode 100644 index 0000000..b2ca878 --- /dev/null +++ b/TODO.md @@ -0,0 +1,31 @@ +### Genral +- [X] __(doc)__ Video de como colaborar PR +- [ ] __(doc)__ Video implementación de test y cobertura +- [ ] __(doc)__ Video explicacion de github action + +### @bot-whatsapp/bot +- [ ] agregar export package +- [X] Posibilidad de en el capture meter todo un nuevo CTX de FLOW .addAnswer('Marca la opcion',{capture:true, join:CTX}) +- [X] .addKeyword('1') no funciona con 1 caracter +- [X] sensitivy viene activado por defecto +- [ ] fallback respuesta en hijo: Se puede colocar en option el ref de la answer fallback +- [ ] colocar mensaje esperando conectando whatsapp (provider) +- [ ] Cuando Envian Sticket devuelve mensaje raro +- [ ] createDatabase validar implementacion de funciones + +### @bot-whatsapp/database +- [X] agregar export package +- [X] __(doc):__ Video para explicar como implementar nuevos database +- [X] Mongo adapter +- [ ] MySQL adapter +- [ ] JsonFile adapter + +### @bot-whatsapp/provider +- [X] agregar export package +- [ ] __(doc):__ Video para explicar como implementar nuevos providers +- [ ] WhatsappWeb provider enviar imagenes +- [ ] WhatsappWeb provider enviar audio +- [ ] Twilio adapter +- [ ] Meta adapter + +### @bot-whatsapp/cli diff --git a/__mocks__/mobile.mock.js b/__mocks__/mobile.mock.js new file mode 100644 index 0000000..7f19f74 --- /dev/null +++ b/__mocks__/mobile.mock.js @@ -0,0 +1,6 @@ +const MOCK_MOBILE_WS = { + from: 'XXXXXX', + hasMedia: false, +} + +module.exports = { MOCK_MOBILE_WS } diff --git a/__mocks__/mock.provider.js b/__mocks__/mock.provider.js new file mode 100644 index 0000000..67b793e --- /dev/null +++ b/__mocks__/mock.provider.js @@ -0,0 +1,21 @@ +const ProviderClass = require('../packages/bot/provider/provider.class') +class MockProvider extends ProviderClass { + constructor() { + super() + } + + delaySendMessage = (miliseconds, eventName, payload) => + new Promise((res) => + setTimeout(() => { + this.emit(eventName, payload) + res + }, miliseconds) + ) + + sendMessage = async (userId, message) => { + console.log(`Enviando... ${userId}, ${message}`) + return Promise.resolve({ userId, message }) + } +} + +module.exports = MockProvider diff --git a/adapter/diaglogflow.js b/adapter/diaglogflow.js deleted file mode 100644 index c00146b..0000000 --- a/adapter/diaglogflow.js +++ /dev/null @@ -1,75 +0,0 @@ -const dialogflow = require('@google-cloud/dialogflow'); -const fs = require('fs') -const nanoid = require('nanoid') -/** - * Debes de tener tu archivo con el nombre "chatbot-account.json" en la raíz del proyecto - */ -let PROJECID; -let CONFIGURATION; -let sessionClient; - -const checkFileCredentials = () => { - if(!fs.existsSync(`${__dirname}/../chatbot-account.json`)){ - return false - } - - const parseCredentials = JSON.parse(fs.readFileSync(`${__dirname}/../chatbot-account.json`)); - PROJECID = parseCredentials.project_id; - CONFIGURATION = { - credentials: { - private_key: parseCredentials['private_key'], - client_email: parseCredentials['client_email'] - } - } - sessionClient = new dialogflow.SessionsClient(CONFIGURATION); -} - -// Create a new session - - -// Detect intent method -const detectIntent = async (queryText) => { - let media = null; - const sessionId = nanoid.nanoid() - const sessionPath = sessionClient.projectAgentSessionPath(PROJECID, sessionId); - const languageCode = process.env.LANGUAGE - const request = { - session: sessionPath, - queryInput: { - text: { - text: queryText, - languageCode: languageCode, - }, - }, - }; - - const responses = await sessionClient.detectIntent(request); - const [singleResponse] = responses; - const { queryResult } = singleResponse - const { intent } = queryResult || { intent: {} } - const parseIntent = intent['displayName'] || null - const parsePayload = queryResult['fulfillmentMessages'].find((a) => a.message === 'payload'); - // console.log(singleResponse) - if (parsePayload && parsePayload.payload) { - const { fields } = parsePayload.payload - media = fields.media.stringValue || null - } - // const customPayload = parsePayload['payload'] - - const parseData = { - replyMessage: queryResult.fulfillmentText, - media, - trigger: null - } - return parseData -} - -const getDataIa = (message = '', cb = () => { }) => { - detectIntent(message).then((res) => { - cb(res) - }) -} - -checkFileCredentials(); - -module.exports = { getDataIa } \ No newline at end of file diff --git a/adapter/index.js b/adapter/index.js deleted file mode 100644 index 3b8dcd3..0000000 --- a/adapter/index.js +++ /dev/null @@ -1,89 +0,0 @@ -const { getData, getReply, saveMessageMysql } = require('./mysql') -const { saveMessageJson } = require('./jsonDb') -const { getDataIa } = require('./diaglogflow') -const stepsInitial = require('../flow/initial.json') -const stepsReponse = require('../flow/response.json') - -const get = (message) => new Promise((resolve, reject) => { - /** - * Si no estas usando un gesto de base de datos - */ - - if (process.env.DATABASE === 'none') { - const { key } = stepsInitial.find(k => k.keywords.includes(message)) || { key: null } - const response = key || null - resolve(response) - } - /** - * Si usas MYSQL - */ - if (process.env.DATABASE === 'mysql') { - getData(message, (dt) => { - resolve(dt) - }); - } - -}) - - -const reply = (step) => new Promise((resolve, reject) => { - /** - * Si no estas usando un gesto de base de datos - */ - if (process.env.DATABASE === 'none') { - let resData = { replyMessage: '', media: null, trigger: null } - const responseFind = stepsReponse[step] || {}; - resData = { - ...resData, - ...responseFind, - replyMessage:responseFind.replyMessage.join('')} - resolve(resData); - return - } - /** - * Si usas MYSQL - */ - if (process.env.DATABASE === 'mysql') { - let resData = { replyMessage: '', media: null, trigger: null } - getReply(step, (dt) => { - resData = { ...resData, ...dt } - resolve(resData) - }); - } -}) - -const getIA = (message) => new Promise((resolve, reject) => { - /** - * Si usas dialogflow - */ - if (process.env.DATABASE === 'dialogflow') { - let resData = { replyMessage: '', media: null, trigger: null } - getDataIa(message,(dt) => { - resData = { ...resData, ...dt } - resolve(resData) - }) - } -}) - -/** - * - * @param {*} message - * @param {*} date - * @param {*} trigger - * @param {*} number - * @returns - */ -const saveMessage = ( message, trigger, number ) => new Promise( async (resolve, reject) => { - switch ( process.env.DATABASE ) { - case 'mysql': - resolve( await saveMessageMysql( message, trigger, number ) ) - break; - case 'none': - resolve( await saveMessageJson( message, trigger, number ) ) - break; - default: - break; - } -}) - -module.exports = { get, reply, getIA, saveMessage } \ No newline at end of file diff --git a/adapter/jsonDb.js b/adapter/jsonDb.js deleted file mode 100644 index dbc2626..0000000 --- a/adapter/jsonDb.js +++ /dev/null @@ -1,20 +0,0 @@ -const Path = require('path') -const StormDB = require("stormdb"); -const date = new Date().toISOString(); -const saveMessageJson = (message, trigger, number) => new Promise( async(resolve,reject) =>{ - try { - const engine = new StormDB.localFileEngine( Path.join(__dirname, `/../chats/${number}.json`) ); - const db = new StormDB(engine); - // set default db value if db is empty - db.default({ messages: [] }); - // add new users entry - db.get("messages").push({ message, date, trigger }); - db.save(); - resolve('Saved') - } catch (error) { - console.log(error) - reject(error) - } -}) - -module.exports = { saveMessageJson } \ No newline at end of file diff --git a/adapter/mysql.js b/adapter/mysql.js deleted file mode 100644 index 6db2f15..0000000 --- a/adapter/mysql.js +++ /dev/null @@ -1,71 +0,0 @@ -const {connection} = require('../config/mysql') -const DATABASE_NAME = process.env.SQL_DATABASE || 'db_test' - -getData = (message = '', callback) => connection.query( - `SELECT * FROM ${DATABASE_NAME}.initial WHERE keywords LIKE '%${message}%' LIMIT 1`, - (error, results - ) => { - const [response] = results - const key = response?.option_key || null - callback(key) -}); - - -getReply = (option_key = '', callback) => connection.query( - `SELECT * FROM ${DATABASE_NAME}.response WHERE option_key = '${option_key}' LIMIT 1`, - (error, results - ) => { - const [response] = results; - console.log(response) - const value = { - replyMessage:response?.replyMessage || '', - trigger:response?.trigger || '', - media:response?.media || '' - } - callback(value) -}); - -getMessages = ( number ) => new Promise((resolve,reejct) => { - try { - connection.query( - `SELECT * FROM ${DATABASE_NAME}.response WHERE number = '${number}'`, (error, results) => { - if(error) { - console.log(error) - } - const [response] = results; - console.log(response) - const value = { - replyMessage:response?.replyMessage || '', - trigger:response?.trigger || '', - media:response?.media || '' - } - resolve(value) - }) - } catch (error) { - - } -}) - -saveMessageMysql = ( message, date, trigger, number ) => new Promise((resolve,reejct) => { - try { - connection.query( - `INSERT INTO ${DATABASE_NAME}.messages `+"( `message`, `date`, `trigger`, `number`)"+` VALUES ('${message}','${date}','${trigger}', '${number}')` , (error, results) => { - if(error) { - if( error.code === 'ER_NO_SUCH_TABLE' ){ - connection.query( `CREATE TABLE ${DATABASE_NAME}.messages `+"( `date` DATE NOT NULL , `message` VARCHAR(450) NOT NULL , `trigger` VARCHAR(450) NOT NULL , `number` VARCHAR(50) NOT NULL ) ENGINE = InnoDB", async (error, results) => { - setTimeout( async () => { - return resolve( await this.saveMessageMysql( message, date, trigger, number ) ) - }, 150) - }) - } - } - console.log('Saved') - console.log( results ) - resolve(results) - }) - } catch (error) { - - } -}) - -module.exports = {getData, getReply, saveMessageMysql} \ No newline at end of file diff --git a/app.js b/app.js deleted file mode 100644 index 246dd69..0000000 --- a/app.js +++ /dev/null @@ -1,229 +0,0 @@ -/** - * ⚡⚡⚡ DECLARAMOS LAS LIBRERIAS y CONSTANTES A USAR! ⚡⚡⚡ - */ -require('dotenv').config() -const fs = require('fs'); -const express = require('express'); -const cors = require('cors') -const qrcode = require('qrcode-terminal'); -const { Client, LegacySessionAuth, LocalAuth } = require('whatsapp-web.js'); -const mysqlConnection = require('./config/mysql') -const { middlewareClient } = require('./middleware/client') -const { generateImage, cleanNumber, checkEnvFile, createClient } = require('./controllers/handle') -const { connectionReady, connectionLost } = require('./controllers/connection') -const { saveMedia } = require('./controllers/save') -const { getMessages, responseMessages, bothResponse } = require('./controllers/flows') -const { sendMedia, sendMessage, lastTrigger, sendMessageButton, readChat } = require('./controllers/send') -const app = express(); -app.use(cors()) -app.use(express.json()) -const MULTI_DEVICE = process.env.MULTI_DEVICE || 'false'; -const server = require('http').Server(app) -const io = require('socket.io')(server, { - cors: { - origins: ['http://localhost:4200'] - } -}) - -let socketEvents = {sendQR:() => {} ,sendStatus:() => {}}; - -io.on('connection', (socket) => { - const CHANNEL = 'main-channel'; - socket.join(CHANNEL); - socketEvents = require('./controllers/socket')(socket) - console.log('Se conecto') -}) - -app.use('/', require('./routes/web')) - -const port = process.env.PORT || 3000 -const SESSION_FILE_PATH = './session.json'; -var client; -var sessionData; - -/** - * Escuchamos cuando entre un mensaje - */ -const listenMessage = () => client.on('message', async msg => { - const { from, body, hasMedia } = msg; - // Este bug lo reporto Lucas Aldeco Brescia para evitar que se publiquen estados - if (from === 'status@broadcast') { - return - } - message = body.toLowerCase(); - console.log('BODY',message) - const number = cleanNumber(from) - await readChat(number, message) - - /** - * Guardamos el archivo multimedia que envia - */ - if (process.env.SAVE_MEDIA && hasMedia) { - const media = await msg.downloadMedia(); - saveMedia(media); - } - - /** - * Si estas usando dialogflow solo manejamos una funcion todo es IA - */ - - if (process.env.DATABASE === 'dialogflow') { - const response = await bothResponse(message); - await sendMessage(client, from, response.replyMessage); - if (response.media) { - sendMedia(client, from, response.media); - } - return - } - - /** - * Ver si viene de un paso anterior - * Aqui podemos ir agregando más pasos - * a tu gusto! - */ - - const lastStep = await lastTrigger(from) || null; - if (lastStep) { - const response = await responseMessages(lastStep) - await sendMessage(client, from, response.replyMessage); - } - - /** - * Respondemos al primero paso si encuentra palabras clave - */ - const step = await getMessages(message); - - if (step) { - const response = await responseMessages(step); - - /** - * Si quieres enviar botones - */ - - await sendMessage(client, from, response.replyMessage, response.trigger); - if(response.hasOwnProperty('actions')){ - const { actions } = response; - await sendMessageButton(client, from, null, actions); - return - } - - if (!response.delay && response.media) { - sendMedia(client, from, response.media); - } - if (response.delay && response.media) { - setTimeout(() => { - sendMedia(client, from, response.media); - }, response.delay) - } - return - } - - //Si quieres tener un mensaje por defecto - if (process.env.DEFAULT_MESSAGE === 'true') { - const response = await responseMessages('DEFAULT') - await sendMessage(client, from, response.replyMessage, response.trigger); - - /** - * Si quieres enviar botones - */ - if(response.hasOwnProperty('actions')){ - const { actions } = response; - await sendMessageButton(client, from, null, actions); - } - return - } -}); - -/** - * Revisamos si tenemos credenciales guardadas para inciar sessio - * este paso evita volver a escanear el QRCODE - */ -const withSession = () => { - console.log(`Validando session con Whatsapp...`) - sessionData = require(SESSION_FILE_PATH); - client = new Client(createClient(sessionData,true)); - - client.on('ready', () => { - connectionReady() - listenMessage() - loadRoutes(client); - socketEvents.sendStatus() - }); - - client.on('auth_failure', () => connectionLost()) - - client.initialize(); -} - -/** - * Generamos un QRCODE para iniciar sesion - */ -const withOutSession = () => { - console.log('No tenemos session guardada'); - console.log([ - '🙌 El core de whatsapp se esta actualizando', - '🙌 para proximamente dar paso al multi-device', - '🙌 falta poco si quieres estar al pendiente unete', - '🙌 http://t.me/leifermendez', - '________________________', - ].join('\n')); - - client = new Client(createClient()); - - client.on('qr', qr => generateImage(qr, () => { - qrcode.generate(qr, { small: true }); - console.log(`Ver QR http://localhost:${port}/qr`) - socketEvents.sendQR(qr) - })) - - client.on('ready', (a) => { - connectionReady() - listenMessage() - loadRoutes(client); - // socketEvents.sendStatus(client) - }); - - client.on('auth_failure', (e) => { - // console.log(e) - // connectionLost() - }); - - client.on('authenticated', (session) => { - sessionData = session; - if(sessionData){ - fs.writeFile(SESSION_FILE_PATH, JSON.stringify(session), function (err) { - if (err) { - console.log(`Ocurrio un error con el archivo: `, err); - } - }); - } - }); - - client.initialize(); -} - -/** - * Cargamos rutas de express - */ - -const loadRoutes = (client) => { - app.use('/api/', middlewareClient(client), require('./routes/api')) -} -/** - * Revisamos si existe archivo con credenciales! - */ -(fs.existsSync(SESSION_FILE_PATH) && MULTI_DEVICE === 'false') ? withSession() : withOutSession(); - -/** - * Verificamos si tienes un gesto de db - */ - -if (process.env.DATABASE === 'mysql') { - mysqlConnection.connect() -} - -server.listen(port, () => { - console.log(`El server esta listo por el puerto ${port}`); -}) -checkEnvFile(); - diff --git a/app.json b/app.json deleted file mode 100644 index a602ccf..0000000 --- a/app.json +++ /dev/null @@ -1,35 +0,0 @@ -{ - "name": "Chatbot Whatsapp (Leifer Mendez)", - "description": "El siguiente proyecto se realizó con fines educativos para el canal de Youtube (Leifer Mendez) donde aprendemos como usando node.js podemos crear un chatbot increíble que además le agregamos inteligencia artificial gracias al servicio de dialogflow", - "repository": "https://github.com/leifermendez/bot-whatsapp", - "logo": "https://avatars0.githubusercontent.com/u/15802366?s=460&u=77ec7ef359e8ed842aef769693f1675c0ed460fd&v=4", - "keywords": [ - "nodejs", - "whatsapp", - "bot", - "chatbot", - "dialogflow" - ], - "addons": [ - ], - "buildpacks": [ - { - "url": "heroku/nodejs" - }, - { - "url": "https://github.com/jontewks/puppeteer-heroku-buildpack" - } - ], - "env": { - "SAVE_MEDIA": "false", - "DATABASE": { - "description": "'none', 'mysql', 'dialogflow' por defecto 'none' Puedes usar alguna de los siguientes opciones. Pero antes debes de saber como funciona y eso lo explico en el video. Puedes obtener más información https://github.com/leifermendez/bot-whatsapp/blob/main/README.md", - "value": "none" - }, - "LANGUAGE": "es", - "SQL_HOST":"your_host", - "SQL_USER":"your_user", - "SQL_PASS":"your_password", - "SQL_DATABASE":"your_database" - } - } \ No newline at end of file diff --git a/changelog.config.js b/changelog.config.js new file mode 100644 index 0000000..70a500d --- /dev/null +++ b/changelog.config.js @@ -0,0 +1,91 @@ +module.exports = { + disableEmoji: false, + format: '{type}{scope}: {emoji}{subject}', + list: [ + 'test', + 'feat', + 'fix', + 'chore', + 'docs', + 'refactor', + 'style', + 'ci', + 'perf', + ], + maxMessageLength: 64, + minMessageLength: 3, + questions: [ + 'type', + 'scope', + 'subject', + 'body', + 'breaking', + 'issues', + 'lerna', + ], + scopes: [], + types: { + chore: { + description: 'Build process or auxiliary tool changes', + emoji: '(🤖)', + value: 'chore', + }, + ci: { + description: 'CI related changes', + emoji: '(🎡)', + value: 'ci', + }, + docs: { + description: 'Documentation only changes', + emoji: '(✏️)', + value: 'docs', + }, + feat: { + description: 'A new feature', + emoji: '(🎸)', + value: 'feat', + }, + fix: { + description: 'A bug fix', + emoji: '(🐛)', + value: 'fix', + }, + perf: { + description: 'A code change that improves performance', + emoji: '(⚡️)', + value: 'perf', + }, + refactor: { + description: + 'A code change that neither fixes a bug or adds a feature', + emoji: '(💡)', + value: 'refactor', + }, + release: { + description: 'Create a release commit', + emoji: '(🏹)', + value: 'release', + }, + style: { + description: + 'Markup, white-space, formatting, missing semi-colons...', + emoji: '(💄)', + value: 'style', + }, + test: { + description: 'Adding missing tests', + emoji: '(💍)', + value: 'test', + }, + messages: { + type: "Select the type of change that you're committing:", + customScope: 'Select the scope this component affects:', + subject: + 'Write a short, imperative mood description of the change:\n', + body: 'Provide a longer description of the change:\n ', + breaking: 'List any breaking changes:\n', + footer: 'Issues this commit closes, e.g #123:', + confirmCommit: 'The packages that this commit has affected\n', + }, + }, +} diff --git a/chatbot-account.json b/chatbot-account.json deleted file mode 100644 index cb8956d..0000000 --- a/chatbot-account.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "type": "", - "project_id": "", - "private_key_id": "", - "private_key":"", - "client_email": "", - "client_id": "", - "auth_uri": "", - "token_uri": "", - "auth_provider_x509_cert_url": "", - "client_x509_cert_url":"" - } - \ No newline at end of file diff --git a/commitlint.config.js b/commitlint.config.js new file mode 100644 index 0000000..4fedde6 --- /dev/null +++ b/commitlint.config.js @@ -0,0 +1 @@ +module.exports = { extends: ['@commitlint/config-conventional'] } diff --git a/config/mysql.js b/config/mysql.js deleted file mode 100644 index a5ab990..0000000 --- a/config/mysql.js +++ /dev/null @@ -1,18 +0,0 @@ -const mysql = require('mysql'); -const connection = mysql.createConnection({ - host : process.env.SQL_HOST || 'localhost', - user : process.env.SQL_USER || 'me', - password : process.env.SQL_PASS || '', - database : process.env.SQL_DATABASE || 'my_db' -}); - -const connect = () => connection.connect(function(err) { - if (err) { - console.error('error connecting: ' + err.stack); - return; - } - - console.log('Conexion correcta con tu base de datos MySQL') -}); - -module.exports = {connect, connection} \ No newline at end of file diff --git a/controllers/connection.js b/controllers/connection.js deleted file mode 100644 index 4fbbb10..0000000 --- a/controllers/connection.js +++ /dev/null @@ -1,13 +0,0 @@ -const connectionReady = (cb = () =>{}) => { - console.log('Listo para escuchas mensajes') - console.log('Client is ready!'); - cb() -} - -const connectionLost = (cb = () =>{}) => { - console.log('** Error de autentificacion vuelve a generar el QRCODE (Borrar el archivo session.json) **'); - cb() -} - - -module.exports = {connectionReady, connectionLost} \ No newline at end of file diff --git a/controllers/flows.js b/controllers/flows.js deleted file mode 100644 index 0e769cd..0000000 --- a/controllers/flows.js +++ /dev/null @@ -1,28 +0,0 @@ -const {get, reply, getIA} = require('../adapter') -const {saveExternalFile, checkIsUrl} = require('./handle') - -const getMessages = async (message) => { - const data = await get(message) - return data -} - -const responseMessages = async (step) => { - const data = await reply(step) - if(data && data.media){ - const file = checkIsUrl(data.media) ? await saveExternalFile(data.media) : data.media; - return {...data,...{media:file}} - } - return data -} - -const bothResponse = async (message) => { - const data = await getIA(message) - if(data && data.media){ - const file = await saveExternalFile(data.media) - return {...data,...{media:file}} - } - return data -} - - -module.exports = { getMessages, responseMessages, bothResponse } \ No newline at end of file diff --git a/controllers/handle.js b/controllers/handle.js deleted file mode 100644 index e9584d3..0000000 --- a/controllers/handle.js +++ /dev/null @@ -1,95 +0,0 @@ -const { Client, LegacySessionAuth, LocalAuth } = require('whatsapp-web.js'); -const http = require('http'); // or 'https' for https:// URLs -const https = require('https'); // or 'https' for https:// URLs -const fs = require('fs'); -const qr = require('qr-image') - -const MULTI_DEVICE = process.env.MULTI_DEVICE || 'false'; - -const cleanNumber = (number) => { - number = number.replace('@c.us', ''); - number = `${number}@c.us`; - return number -} - -const saveExternalFile = (url) => new Promise((resolve, reject) => { - const ext = url.split('.').pop() - const checkProtocol = url.split('/').includes('https:'); - const handleHttp = checkProtocol ? https : http; - const name = `${Date.now()}.${ext}`; - const file = fs.createWriteStream(`${__dirname}/../mediaSend/${name}`); - console.log(url) - handleHttp.get(url, function(response) { - response.pipe(file); - file.on('finish', function() { - file.close(); // close() is async, call cb after close completes. - resolve(name) - }); - file.on('error', function() { - console.log('errro') - file.close(); // close() is async, call cb after close completes. - resolve(null) - }); - }); -}) - -const checkIsUrl = (path) => { - try{ - regex = /^(http(s)?:\/\/)[\w.-]+(?:\.[\w\.-]+)+[\w\-\._~:/?#[\]@!\$&'\(\)\*\+,;=.]+$/i; - match = path.match(regex); - return match[0] - }catch(e){ - return null - } -} - -const generateImage = (base64, cb = () => {}) => { - let qr_svg = qr.image(base64, { type: 'svg', margin: 4 }); - qr_svg.pipe(require('fs').createWriteStream('./mediaSend/qr-code.svg')); - console.log(`⚡ Recuerda que el QR se actualiza cada minuto ⚡'`); - console.log(`⚡ Actualiza F5 el navegador para mantener el mejor QR⚡`); - cb() -} - -const checkEnvFile = () => { - const pathEnv = `${__dirname}/../.env`; - const isExist = fs.existsSync(pathEnv); - if(!isExist){ - console.log(`🆗 ATENCION! 🆗 te falta crear tu archivo .env de lo contrario no funcionara`) - } -} - -/** - * - * @param {*} session - * @param {*} cb - */ -const createClient = (session = {}, login = false) => { - console.log(`Mode: ${(MULTI_DEVICE === 'false') ? 'No Multi-device' : 'Si Multi-device'} `) - const objectLegacy = (login) ? { - authStrategy: new LegacySessionAuth({ - session - }) - } : {session}; - - if(MULTI_DEVICE == 'false') { - return {...objectLegacy, - restartOnAuthFail: true, - puppeteer: { - args: [ - '--no-sandbox' - ], - } - } - }else{ - return { - puppeteer: { - headless: true, - args: ['--no-sandbox'] - }, - clientId: 'client-one' - } - } -} - -module.exports = {cleanNumber, saveExternalFile, generateImage, checkIsUrl, checkEnvFile, createClient} \ No newline at end of file diff --git a/controllers/save.js b/controllers/save.js deleted file mode 100644 index 92555e0..0000000 --- a/controllers/save.js +++ /dev/null @@ -1,18 +0,0 @@ -const mimeDb = require('mime-db') -const fs = require('fs') - -/** - * Guardamos archivos multimedia que nuestro cliente nos envie! - * @param {*} media - */ - - -const saveMedia = (media) => { - const extensionProcess = mimeDb[media.mimetype] - const ext = extensionProcess.extensions[0] - fs.writeFile(`./media/${Date.now()}.${ext}`, media.data, { encoding: 'base64' }, function (err) { - console.log('** Archivo Media Guardado **'); - }); -} - -module.exports = {saveMedia} \ No newline at end of file diff --git a/controllers/send.js b/controllers/send.js deleted file mode 100644 index ccbe134..0000000 --- a/controllers/send.js +++ /dev/null @@ -1,102 +0,0 @@ - -const ExcelJS = require('exceljs'); -const moment = require('moment'); -const fs = require('fs'); -const { MessageMedia, Buttons } = require('whatsapp-web.js'); -const { cleanNumber } = require('./handle') -const DELAY_TIME = 170; //ms -const DIR_MEDIA = `${__dirname}/../mediaSend`; -// import { Low, JSONFile } from 'lowdb' -// import { join } from 'path' -const { saveMessage } = require('../adapter') -/** - * Enviamos archivos multimedia a nuestro cliente - * @param {*} number - * @param {*} fileName - */ - -const sendMedia = (client, number, fileName) => { - number = cleanNumber(number) - const file = `${DIR_MEDIA}/${fileName}`; - if (fs.existsSync(file)) { - const media = MessageMedia.fromFilePath(file); - client.sendMessage(number, media, { sendAudioAsVoice: true }); - } -} - -/** - * Enviamos archivos como notas de voz - * @param {*} number - * @param {*} fileName - */ - - const sendMediaVoiceNote = (client, number, fileName) => { - number = cleanNumber(number) - const file = `${DIR_MEDIA}/${fileName}`; - if (fs.existsSync(file)) { - const media = MessageMedia.fromFilePath(file); - client.sendMessage(number, media ,{ sendAudioAsVoice: true }); - } -} - -/** - * Enviamos un mensaje simple (texto) a nuestro cliente - * @param {*} number - */ -const sendMessage = async (client, number = null, text = null, trigger = null) => { - setTimeout(async () => { - number = cleanNumber(number) - const message = text - client.sendMessage(number, message); - await readChat(number, message, trigger) - console.log(`⚡⚡⚡ Enviando mensajes....`); - },DELAY_TIME) -} - -/** - * Enviamos un mensaje con buttons a nuestro cliente - * @param {*} number - */ -const sendMessageButton = async (client, number = null, text = null, actionButtons) => { - number = cleanNumber(number) - const { title = null, message = null, footer = null, buttons = [] } = actionButtons; - let button = new Buttons(message,[...buttons], title, footer); - client.sendMessage(number, button); - - console.log(`⚡⚡⚡ Enviando mensajes....`); -} - - -/** - * Opte - */ -const lastTrigger = (number) => new Promise((resolve, reject) => { - number = cleanNumber(number) - const pathExcel = `${__dirname}/../chats/${number}.xlsx`; - const workbook = new ExcelJS.Workbook(); - if (fs.existsSync(pathExcel)) { - workbook.xlsx.readFile(pathExcel) - .then(() => { - const worksheet = workbook.getWorksheet(1); - const lastRow = worksheet.lastRow; - const getRowPrevStep = worksheet.getRow(lastRow.number); - const lastStep = getRowPrevStep.getCell('C').value; - resolve(lastStep) - }); - } else { - resolve(null) - } -}) - -/** - * Guardar historial de conversacion - * @param {*} number - * @param {*} message - */ -const readChat = async (number, message, trigger = null) => { - number = cleanNumber(number) - await saveMessage( message, trigger, number ) - console.log('Saved') -} - -module.exports = { sendMessage, sendMedia, lastTrigger, sendMessageButton, readChat, sendMediaVoiceNote } \ No newline at end of file diff --git a/controllers/socket.js b/controllers/socket.js deleted file mode 100644 index 05ff8d3..0000000 --- a/controllers/socket.js +++ /dev/null @@ -1,16 +0,0 @@ -module.exports = (socket) => { - return { - sendQR:(qr) => { - socket.emit('connection_qr',{ - qr - }) - }, - sendStatus:() => { - socket.emit('connection_status',{ - a:1 - }) - } - } - -} - diff --git a/controllers/web.js b/controllers/web.js deleted file mode 100644 index 8718049..0000000 --- a/controllers/web.js +++ /dev/null @@ -1,16 +0,0 @@ -const fs = require('fs') -const { sendMessage } = require('../controllers/send') - -const sendMessagePost = (req, res) => { - const { message, number } = req.body - const client = req.clientWs || null; - sendMessage(client, number, message) - res.send({ status: 'Enviado!' }) -} - -const getQr = (req, res) => { - res.writeHead(200, { 'content-type': 'image/svg+xml' }); - fs.createReadStream(`${__dirname}/../mediaSend/qr-code.svg`).pipe(res); -} - -module.exports = { sendMessagePost, getQr } \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 0000000..1e49d41 --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,25 @@ +version: '3.3' + +services: + mongo: + image: mongo + container_name: app_enviroment + restart: always + ports: + - '27019:27017' + environment: + MONGO_INITDB_DATABASE: bot + expose: + - 27019 + mysql: + image: mysql + command: --default-authentication-plugin=mysql_native_password + restart: always + environment: + MYSQL_ROOT_PASSWORD: example + MYSQL_DATABASE: bot + container_name: app_mysql + ports: + - '3306:3306' + expose: + - 3306 diff --git a/flow/initial.json b/flow/initial.json deleted file mode 100644 index 6a91c75..0000000 --- a/flow/initial.json +++ /dev/null @@ -1,93 +0,0 @@ -[ - { - "keywords": [ - "hola", - "hola!", - "ola", - "ole", - "inicio", - "welcome", - "buenos días", - "buenas tardes", - "buenas noches", - "me dieron este número", - "venden a crédito", - "quisiera saber si venden", - "necesito saber" - ], - "key": "STEP_1" - }, - { - "keywords": [ - "cursos", - "info", - "curso" ], - "key": "STEP_2" - }, - { - "keywords": [ - "angular" - ], - "key": "STEP_2_1" - }, - { - "keywords": [ - "node" - ], - "key": "STEP_2_2" - }, - { - "keywords": [ - "ngrx" - ], - "key": "STEP_2_3" - }, - { - "keywords": [ - "aws" - ], - "key": "STEP_2_4" - }, - { - "keywords": [ - "asesor", - "asesores", - "Vendedor", - "cobrador" - ], - "key": "STEP_3" - }, - { - "keywords": [ - "muchas gracias", - "ok", - "gracias", - "vale gracias" - ], - "key": "STEP_4" - }, - { - "keywords": [ - "youtube" - ], - "key": "STEP_5" - }, - { - "keywords": [ - "VER_CURSOS" - ], - "key": "STEP_6" - }, - { - "keywords": [ - "telegram" - ], - "key": "STEP_7" - }, - { - "keywords": [ - "audio" - ], - "key": "STEP_8" - } -] \ No newline at end of file diff --git a/flow/response.json b/flow/response.json deleted file mode 100644 index f8a57f7..0000000 --- a/flow/response.json +++ /dev/null @@ -1,148 +0,0 @@ -{ - "DEFAULT":{ - "replyMessage":[ - "*Esta respuesta es un respuesta default* cuando no se consigue una palabra clave \n", - "la puedes desactivar en tu archivo .env DEFAULT_MESSAGE=false \n", - "tambien te quiero recordar que si presentas algun error pasarte por el repositorio \n", - "https://github.com/leifermendez/bot-whatsapp#chatbot-whatsapp-opensource \n", - "y recuerda tener la ultima versión del proyecto \n\n", - "Prueba escribiendo *hola* \n" - ], - "media":null, - "trigger":null - }, - "STEP_0":{ - "replyMessage":[ - "El flujo ha finalizado \n", - "pero puedes ver todo el codigo de este \n", - "repositorio en https://github.com/leifermendez/bot-whatsapp.git" - ], - "media":null, - "trigger":null - }, - "STEP_1":{ - "replyMessage":[ - "Hola! y✌️ Bienvenido a este 🤖 CHATBOT de Whatsapp, lo primero \n", - "decirte que mi nombre es *Leifer Mendez*😎 \n", - "\n Si necesitas ver más info sobre las capacitacion tecnicas ", - "escribe *cursos* o *info* o escribe *audio*" - ], - "media":null, - "trigger":null, - "actions":{ - "title":"¿Que te interesa ver?", - "message":"Recuerda todo este contenido es gratis y estaria genial que me siguas!", - "footer":"Gracias", - "buttons":[ - {"body":"Cursos"}, - {"body":"Youtube"}, - {"body":"Telegram"} - ] - } - }, - "STEP_2":{ - "replyMessage":[ - "Perfecto, te voy a pasar la lista ", - "de los temas que tengo y un breve video 🙂🤖 \n\n", - "*Angular* Basico (Pago) \n", - "*Angular* Basico (Gratis) \n", - "*Node* Basico (Gratis) \n", - "*NGRX* Basico (Gratis) \n", - "*AWS* Basico (Pago) \n\n", - "Escribe la palabra del tema que te interese \n" - ], - "media":"https://i.giphy.com/media/5J5gN0WUk0VToHaK2p/giphy-downsized.gif", - "trigger":null - }, - "STEP_2_1":{ - "replyMessage":[ - "Si te interesa Angular tienes disponible \n", - "*(Gratis)* https://bit.ly/367tJ32 \n\n", - "*(Pago)* https://link.codigoencasa.com/PROMO-INICIAL \n\n", - "*(Pago)* https://link.codigoencasa.com/ANGULAR-BASICO-EDTEAM \n\n", - "😎😎😎" - ], - "media":"https://i.imgur.com/Q0a5UQI.jpg", - "trigger":null - }, - "STEP_2_2":{ - "replyMessage":[ - "Si te interesa NODE tienes disponible \n", - "*(Gratis)* https://bit.ly/3od1Bl6 \n\n", - "Espero pronto tener más material disponible", - "🤖" - ], - "media":null, - "trigger":null - }, - "STEP_2_3":{ - "replyMessage":[ - "NGRX para manejar estados en Angular \n", - "*(Gratis)* https://bit.ly/ngrx-desde-cero \n", - "A darle! 😮" - ], - "media":null, - "trigger":null - }, - "STEP_2_4":{ - "replyMessage":[ - "Muy bien AWS esta pronto a salir pre-registrate aquí \n", - "*(Pre-registro)* https://link.codigoencasa.com/AWS-BASICO-INVITACION \n", - "😮😮" - ], - "media":null, - "trigger":null - }, - "STEP_3":{ - "replyMessage":[ - "¿Ok cual curso de intereso? \n", - "*angular* , *node*, *ngrx*, *aws*" - ], - "media":null, - "trigger":null - }, - "STEP_4":{ - "replyMessage":[ - "Gracias a ti! \n" - ], - "media":"https://media4.giphy.com/media/hur0SFIU5SH4mxNBWa/giphy.gif", - "trigger":null - }, - "STEP_5":{ - "replyMessage":[ - "Muy bien te comparto el canal de Youtube \n", - "https://youtube.com/leifermendez \n" - ], - "media":null, - "trigger":null - }, - "STEP_6":{ - "replyMessage":[ - "Perfecto, te voy a pasar la lista ", - "de los temas que tengo y un breve video 🙂🤖 \n\n", - "*Angular* Basico (Pago) \n", - "*Angular* Basico (Gratis) \n", - "*Node* Basico (Gratis) \n", - "*NGRX* Basico (Gratis) \n", - "*AWS* Basico (Pago) \n\n", - "Escribe la palabra del tema que te interese \n" - ], - "media":"https://i.giphy.com/media/5J5gN0WUk0VToHaK2p/giphy-downsized.gif", - "trigger":null - }, - "STEP_7":{ - "replyMessage":[ - "Vente al telegram \n", - "https://t.me/leifermendez \n" - ], - "media":null, - "trigger":null - }, - "STEP_8":{ - "replyMessage":[ - "Esto es una nota de voz \n" - ], - "media":"nota-de-voz.mp3", - "trigger":null - } -} \ No newline at end of file diff --git a/mediaSend/PTT-20220223-WA0000.opus b/mediaSend/PTT-20220223-WA0000.opus deleted file mode 100644 index 6e6e995..0000000 Binary files a/mediaSend/PTT-20220223-WA0000.opus and /dev/null differ diff --git a/mediaSend/nota-de-voz.mp3 b/mediaSend/nota-de-voz.mp3 deleted file mode 100644 index a10f4f8..0000000 Binary files a/mediaSend/nota-de-voz.mp3 and /dev/null differ diff --git a/middleware/client.js b/middleware/client.js deleted file mode 100644 index 67c0d30..0000000 --- a/middleware/client.js +++ /dev/null @@ -1,21 +0,0 @@ -const middlewareClient = (client = null) => async (req, res, next) => { - try { - - if(!client){ - res.status(409) - console.log(client) - res.send({ error: 'Error de client.' }) - }else{ - req.clientWs = client; - next() - } - - - } catch (e) { - console.log(e) - res.status(409) - res.send({ error: 'Error de client' }) - } - -} -module.exports = { middlewareClient } \ No newline at end of file diff --git a/middleware/db.js b/middleware/db.js deleted file mode 100644 index e69de29..0000000 diff --git a/package-lock.json b/package-lock.json deleted file mode 100644 index bc7dbe0..0000000 --- a/package-lock.json +++ /dev/null @@ -1,3460 +0,0 @@ -{ - "name": "test-ws-bot", - "version": "1.0.0", - "lockfileVersion": 1, - "requires": true, - "dependencies": { - "@fast-csv/format": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/@fast-csv/format/-/format-4.3.5.tgz", - "integrity": "sha512-8iRn6QF3I8Ak78lNAa+Gdl5MJJBM5vRHivFtMRUWINdevNo00K7OXxS2PshawLKTejVwieIlPmK5YlLu6w4u8A==", - "requires": { - "@types/node": "^14.0.1", - "lodash.escaperegexp": "^4.1.2", - "lodash.isboolean": "^3.0.3", - "lodash.isequal": "^4.5.0", - "lodash.isfunction": "^3.0.9", - "lodash.isnil": "^4.0.0" - }, - "dependencies": { - "@types/node": { - "version": "14.18.12", - "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.12.tgz", - "integrity": "sha512-q4jlIR71hUpWTnGhXWcakgkZeHa3CCjcQcnuzU8M891BAWA2jHiziiWEPEkdS5pFsz7H9HJiy8BrK7tBRNrY7A==" - } - } - }, - "@fast-csv/parse": { - "version": "4.3.6", - "resolved": "https://registry.npmjs.org/@fast-csv/parse/-/parse-4.3.6.tgz", - "integrity": "sha512-uRsLYksqpbDmWaSmzvJcuApSEe38+6NQZBUsuAyMZKqHxH0g1wcJgsKUvN3WC8tewaqFjBMMGrkHmC+T7k8LvA==", - "requires": { - "@types/node": "^14.0.1", - "lodash.escaperegexp": "^4.1.2", - "lodash.groupby": "^4.6.0", - "lodash.isfunction": "^3.0.9", - "lodash.isnil": "^4.0.0", - "lodash.isundefined": "^3.0.1", - "lodash.uniq": "^4.5.0" - }, - "dependencies": { - "@types/node": { - "version": "14.18.12", - "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.12.tgz", - "integrity": "sha512-q4jlIR71hUpWTnGhXWcakgkZeHa3CCjcQcnuzU8M891BAWA2jHiziiWEPEkdS5pFsz7H9HJiy8BrK7tBRNrY7A==" - } - } - }, - "@google-cloud/dialogflow": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/@google-cloud/dialogflow/-/dialogflow-4.6.0.tgz", - "integrity": "sha512-+S/nDcUAwFdyMMtQVXn5yEiCNqS9+un3DjHxZV9D+cyCq0e2fuRcW2XeqasxM/hEA+BUHcO8ETDAOAsk0e5IHA==", - "requires": { - "google-gax": "^2.24.1", - "protobufjs": "^6.8.9" - } - }, - "@grpc/grpc-js": { - "version": "1.5.5", - "resolved": "https://registry.npmjs.org/@grpc/grpc-js/-/grpc-js-1.5.5.tgz", - "integrity": "sha512-FTd27ItHlsSG/7hp62xgI9YnqSwRbHRSVmDVR8DwOoC+6t8JhHRXe2JL0U8N9GLc0jS0HrtEbO/KP5+G0ebjLQ==", - "requires": { - "@grpc/proto-loader": "^0.6.4", - "@types/node": ">=12.12.47" - } - }, - "@grpc/proto-loader": { - "version": "0.6.9", - "resolved": "https://registry.npmjs.org/@grpc/proto-loader/-/proto-loader-0.6.9.tgz", - "integrity": "sha512-UlcCS8VbsU9d3XTXGiEVFonN7hXk+oMXZtoHHG2oSA1/GcDP1q6OUgs20PzHDGizzyi8ufGSUDlk3O2NyY7leg==", - "requires": { - "@types/long": "^4.0.1", - "lodash.camelcase": "^4.3.0", - "long": "^4.0.0", - "protobufjs": "^6.10.0", - "yargs": "^16.2.0" - } - }, - "@opencensus/core": { - "version": "0.0.9", - "resolved": "https://registry.npmjs.org/@opencensus/core/-/core-0.0.9.tgz", - "integrity": "sha512-31Q4VWtbzXpVUd2m9JS6HEaPjlKvNMOiF7lWKNmXF84yUcgfAFL5re7/hjDmdyQbOp32oGc+RFV78jXIldVz6Q==", - "dev": true, - "requires": { - "continuation-local-storage": "^3.2.1", - "log-driver": "^1.2.7", - "semver": "^5.5.0", - "shimmer": "^1.2.0", - "uuid": "^3.2.1" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "@opencensus/propagation-b3": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/@opencensus/propagation-b3/-/propagation-b3-0.0.8.tgz", - "integrity": "sha512-PffXX2AL8Sh0VHQ52jJC4u3T0H6wDK6N/4bg7xh4ngMYOIi13aR1kzVvX1sVDBgfGwDOkMbl4c54Xm3tlPx/+A==", - "dev": true, - "requires": { - "@opencensus/core": "^0.0.8", - "uuid": "^3.2.1" - }, - "dependencies": { - "@opencensus/core": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/@opencensus/core/-/core-0.0.8.tgz", - "integrity": "sha512-yUFT59SFhGMYQgX0PhoTR0LBff2BEhPrD9io1jWfF/VDbakRfs6Pq60rjv0Z7iaTav5gQlttJCX2+VPxFWCuoQ==", - "dev": true, - "requires": { - "continuation-local-storage": "^3.2.1", - "log-driver": "^1.2.7", - "semver": "^5.5.0", - "shimmer": "^1.2.0", - "uuid": "^3.2.1" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "@pedroslopez/moduleraid": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/@pedroslopez/moduleraid/-/moduleraid-5.0.2.tgz", - "integrity": "sha512-wtnBAETBVYZ9GvcbgdswRVSLkFkYAGv1KzwBBTeRXvGT9sb9cPllOgFFWXCn9PyARQ0H+Ijz6mmoRrGateUDxQ==" - }, - "@pm2/agent": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/@pm2/agent/-/agent-2.0.1.tgz", - "integrity": "sha512-QKHMm6yexcvdDfcNE7PL9D6uEjoQPGRi+8dh+rc4Hwtbpsbh5IAvZbz3BVGjcd4HaX6pt2xGpOohG7/Y2L4QLw==", - "dev": true, - "requires": { - "async": "~3.2.0", - "chalk": "~3.0.0", - "dayjs": "~1.8.24", - "debug": "~4.3.1", - "eventemitter2": "~5.0.1", - "fast-json-patch": "^3.0.0-1", - "fclone": "~1.0.11", - "nssocket": "0.6.0", - "pm2-axon": "~4.0.1", - "pm2-axon-rpc": "~0.7.0", - "proxy-agent": "~5.0.0", - "semver": "~7.2.0", - "ws": "~7.4.0" - }, - "dependencies": { - "dayjs": { - "version": "1.8.36", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.8.36.tgz", - "integrity": "sha512-3VmRXEtw7RZKAf+4Tv1Ym9AGeo8r8+CjDi26x+7SYQil1UqtqdaokhzoEJohqlzt0m5kacJSDhJQkG/LWhpRBw==", - "dev": true - }, - "semver": { - "version": "7.2.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.2.3.tgz", - "integrity": "sha512-utbW9Z7ZxVvwiIWkdOMLOR9G/NFXh2aRucghkVrEMJWuC++r3lCkBC3LwqBinyHzGMAJxY5tn6VakZGHObq5ig==", - "dev": true - }, - "ws": { - "version": "7.4.6", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.6.tgz", - "integrity": "sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==", - "dev": true - } - } - }, - "@pm2/io": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@pm2/io/-/io-5.0.0.tgz", - "integrity": "sha512-3rToDVJaRoob5Lq8+7Q2TZFruoEkdORxwzFpZaqF4bmH6Bkd7kAbdPrI/z8X6k1Meq5rTtScM7MmDgppH6aLlw==", - "dev": true, - "requires": { - "@opencensus/core": "0.0.9", - "@opencensus/propagation-b3": "0.0.8", - "async": "~2.6.1", - "debug": "~4.3.1", - "eventemitter2": "^6.3.1", - "require-in-the-middle": "^5.0.0", - "semver": "6.3.0", - "shimmer": "^1.2.0", - "signal-exit": "^3.0.3", - "tslib": "1.9.3" - }, - "dependencies": { - "async": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", - "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", - "dev": true, - "requires": { - "lodash": "^4.17.14" - } - }, - "eventemitter2": { - "version": "6.4.5", - "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-6.4.5.tgz", - "integrity": "sha512-bXE7Dyc1i6oQElDG0jMRZJrRAn9QR2xyyFGmBdZleNmyQX0FqGYmhZIrIrpPfm/w//LTo4tVQGOGQcGCb5q9uw==", - "dev": true - }, - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - }, - "tslib": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz", - "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==", - "dev": true - } - } - }, - "@pm2/js-api": { - "version": "0.6.7", - "resolved": "https://registry.npmjs.org/@pm2/js-api/-/js-api-0.6.7.tgz", - "integrity": "sha512-jiJUhbdsK+5C4zhPZNnyA3wRI01dEc6a2GhcQ9qI38DyIk+S+C8iC3fGjcjUbt/viLYKPjlAaE+hcT2/JMQPXw==", - "dev": true, - "requires": { - "async": "^2.6.3", - "axios": "^0.21.0", - "debug": "~4.3.1", - "eventemitter2": "^6.3.1", - "ws": "^7.0.0" - }, - "dependencies": { - "async": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", - "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", - "dev": true, - "requires": { - "lodash": "^4.17.14" - } - }, - "eventemitter2": { - "version": "6.4.5", - "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-6.4.5.tgz", - "integrity": "sha512-bXE7Dyc1i6oQElDG0jMRZJrRAn9QR2xyyFGmBdZleNmyQX0FqGYmhZIrIrpPfm/w//LTo4tVQGOGQcGCb5q9uw==", - "dev": true - }, - "ws": { - "version": "7.5.7", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.7.tgz", - "integrity": "sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A==", - "dev": true - } - } - }, - "@protobufjs/aspromise": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@protobufjs/aspromise/-/aspromise-1.1.2.tgz", - "integrity": "sha1-m4sMxmPWaafY9vXQiToU00jzD78=" - }, - "@protobufjs/base64": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@protobufjs/base64/-/base64-1.1.2.tgz", - "integrity": "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==" - }, - "@protobufjs/codegen": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/@protobufjs/codegen/-/codegen-2.0.4.tgz", - "integrity": "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==" - }, - "@protobufjs/eventemitter": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz", - "integrity": "sha1-NVy8mLr61ZePntCV85diHx0Ga3A=" - }, - "@protobufjs/fetch": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@protobufjs/fetch/-/fetch-1.1.0.tgz", - "integrity": "sha1-upn7WYYUr2VwDBYZ/wbUVLDYTEU=", - "requires": { - "@protobufjs/aspromise": "^1.1.1", - "@protobufjs/inquire": "^1.1.0" - } - }, - "@protobufjs/float": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/@protobufjs/float/-/float-1.0.2.tgz", - "integrity": "sha1-Xp4avctz/Ap8uLKR33jIy9l7h9E=" - }, - "@protobufjs/inquire": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@protobufjs/inquire/-/inquire-1.1.0.tgz", - "integrity": "sha1-/yAOPnzyQp4tyvwRQIKOjMY48Ik=" - }, - "@protobufjs/path": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@protobufjs/path/-/path-1.1.2.tgz", - "integrity": "sha1-bMKyDFya1q0NzP0hynZz2Nf79o0=" - }, - "@protobufjs/pool": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@protobufjs/pool/-/pool-1.1.0.tgz", - "integrity": "sha1-Cf0V8tbTq/qbZbw2ZQbWrXhG/1Q=" - }, - "@protobufjs/utf8": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@protobufjs/utf8/-/utf8-1.1.0.tgz", - "integrity": "sha1-p3c2C1s5oaLlEG+OhY8v0tBgxXA=" - }, - "@socket.io/base64-arraybuffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/@socket.io/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz", - "integrity": "sha512-dOlCBKnDw4iShaIsH/bxujKTM18+2TOAsYz+KSc11Am38H4q5Xw8Bbz97ZYdrVNM+um3p7w86Bvvmcn9q+5+eQ==" - }, - "@tokenizer/token": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/@tokenizer/token/-/token-0.3.0.tgz", - "integrity": "sha512-OvjF+z51L3ov0OyAU0duzsYuvO01PH7x4t6DJx+guahgTnBHkhJdG7soQeTSFLWN3efnHyibZ4Z8l2EuWwJN3A==" - }, - "@tootallnate/once": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", - "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", - "dev": true - }, - "@types/component-emitter": { - "version": "1.2.11", - "resolved": "https://registry.npmjs.org/@types/component-emitter/-/component-emitter-1.2.11.tgz", - "integrity": "sha512-SRXjM+tfsSlA9VuG8hGO2nft2p8zjXCK1VcC6N4NXbBbYbSia9kzCChYQajIjzIqOOOuh5Ock6MmV2oux4jDZQ==" - }, - "@types/cookie": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.1.tgz", - "integrity": "sha512-XW/Aa8APYr6jSVVA1y/DEIZX0/GMKLEVekNG727R8cs56ahETkRAy/3DR7+fJyh7oUgGwNQaRfXCun0+KbWY7Q==" - }, - "@types/cors": { - "version": "2.8.12", - "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.12.tgz", - "integrity": "sha512-vt+kDhq/M2ayberEtJcIN/hxXy1Pk+59g2FV/ZQceeaTyCtCucjL2Q7FXlFjtWn4n15KCr1NE2lNNFhp0lEThw==" - }, - "@types/long": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/@types/long/-/long-4.0.1.tgz", - "integrity": "sha512-5tXH6Bx/kNGd3MgffdmP4dy2Z+G4eaXw0SE81Tq3BNadtnMR5/ySMzX4SLEzHJzSmPNn4HIdpQsBvXMUykr58w==" - }, - "@types/node": { - "version": "17.0.18", - "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.18.tgz", - "integrity": "sha512-eKj4f/BsN/qcculZiRSujogjvp5O/k4lOW5m35NopjZM/QwLOR075a8pJW5hD+Rtdm2DaCVPENS6KtSQnUD6BA==" - }, - "@types/yauzl": { - "version": "2.9.2", - "resolved": "https://registry.npmjs.org/@types/yauzl/-/yauzl-2.9.2.tgz", - "integrity": "sha512-8uALY5LTvSuHgloDVUvWP3pIauILm+8/0pDMokuDYIoNsOkSwd5AiHBTSEJjKTDcZr5z8UpgOWZkxBF4iJftoA==", - "optional": true, - "requires": { - "@types/node": "*" - } - }, - "abort-controller": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", - "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", - "requires": { - "event-target-shim": "^5.0.0" - } - }, - "accepts": { - "version": "1.3.8", - "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", - "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", - "requires": { - "mime-types": "~2.1.34", - "negotiator": "0.6.3" - } - }, - "acorn": { - "version": "8.7.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.0.tgz", - "integrity": "sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ==", - "dev": true - }, - "acorn-walk": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", - "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", - "dev": true - }, - "agent-base": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", - "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", - "requires": { - "debug": "4" - } - }, - "amp": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/amp/-/amp-0.3.1.tgz", - "integrity": "sha1-at+NWKdPNh6CwfqNOJwHnhOfxH0=", - "dev": true - }, - "amp-message": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/amp-message/-/amp-message-0.1.2.tgz", - "integrity": "sha1-p48cmJlQh602GSpBKY5NtJ49/EU=", - "dev": true, - "requires": { - "amp": "0.3.1" - } - }, - "ansi-colors": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", - "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", - "dev": true - }, - "ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" - }, - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "requires": { - "color-convert": "^2.0.1" - } - }, - "anymatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", - "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", - "dev": true, - "requires": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" - } - }, - "archiver": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/archiver/-/archiver-5.3.0.tgz", - "integrity": "sha512-iUw+oDwK0fgNpvveEsdQ0Ase6IIKztBJU2U0E9MzszMfmVVUyv1QJhS2ITW9ZCqx8dktAxVAjWWkKehuZE8OPg==", - "requires": { - "archiver-utils": "^2.1.0", - "async": "^3.2.0", - "buffer-crc32": "^0.2.1", - "readable-stream": "^3.6.0", - "readdir-glob": "^1.0.0", - "tar-stream": "^2.2.0", - "zip-stream": "^4.1.0" - } - }, - "archiver-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-2.1.0.tgz", - "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", - "requires": { - "glob": "^7.1.4", - "graceful-fs": "^4.2.0", - "lazystream": "^1.0.0", - "lodash.defaults": "^4.2.0", - "lodash.difference": "^4.5.0", - "lodash.flatten": "^4.4.0", - "lodash.isplainobject": "^4.0.6", - "lodash.union": "^4.6.0", - "normalize-path": "^3.0.0", - "readable-stream": "^2.0.0" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "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" - } - }, - "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==" - }, - "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==", - "requires": { - "safe-buffer": "~5.1.0" - } - } - } - }, - "argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "dev": true, - "requires": { - "sprintf-js": "~1.0.2" - }, - "dependencies": { - "sprintf-js": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", - "dev": true - } - } - }, - "array-flatten": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", - "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" - }, - "arrify": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/arrify/-/arrify-2.0.1.tgz", - "integrity": "sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug==" - }, - "ast-types": { - "version": "0.13.4", - "resolved": "https://registry.npmjs.org/ast-types/-/ast-types-0.13.4.tgz", - "integrity": "sha512-x1FCFnFifvYDDzTaLII71vG5uvDwgtmDTEVWAxrgeiR8VjMONcCXJx7E+USjDtHlwFmt9MysbqgF9b9Vjr6w+w==", - "dev": true, - "requires": { - "tslib": "^2.0.1" - } - }, - "async": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/async/-/async-3.2.3.tgz", - "integrity": "sha512-spZRyzKL5l5BZQrr/6m/SqFdBN0q3OCI0f9rjfBzCMBIP4p75P620rR3gTmaksNOhmzgdxcaxdNfMy6anrbM0g==" - }, - "async-listener": { - "version": "0.6.10", - "resolved": "https://registry.npmjs.org/async-listener/-/async-listener-0.6.10.tgz", - "integrity": "sha512-gpuo6xOyF4D5DE5WvyqZdPA3NGhiT6Qf07l7DCB0wwDEsLvDIbCr6j9S5aj5Ch96dLace5tXVzWBZkxU/c5ohw==", - "dev": true, - "requires": { - "semver": "^5.3.0", - "shimmer": "^1.1.0" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } - }, - "axios": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz", - "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==", - "dev": true, - "requires": { - "follow-redirects": "^1.14.0" - } - }, - "balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" - }, - "base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" - }, - "base64id": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/base64id/-/base64id-2.0.0.tgz", - "integrity": "sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog==" - }, - "big-integer": { - "version": "1.6.51", - "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.51.tgz", - "integrity": "sha512-GPEid2Y9QU1Exl1rpO9B2IPJGHPSupF5GnVIP0blYvNOMer2bTvSWs1jGOUg04hTmu67nmLsQ9TBo1puaotBHg==" - }, - "bignumber.js": { - "version": "9.0.2", - "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.0.2.tgz", - "integrity": "sha512-GAcQvbpsM0pUb0zw1EI0KhQEZ+lRwR5fYaAp3vPOYuP7aDvGy6cVN6XHLauvF8SOga2y0dcLcjt3iQDTSEliyw==" - }, - "binary": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/binary/-/binary-0.3.0.tgz", - "integrity": "sha1-n2BVO8XOjDOG87VTz/R0Yq3sqnk=", - "requires": { - "buffers": "~0.1.1", - "chainsaw": "~0.1.0" - } - }, - "binary-extensions": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", - "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", - "dev": true - }, - "bl": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", - "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", - "requires": { - "buffer": "^5.5.0", - "inherits": "^2.0.4", - "readable-stream": "^3.4.0" - } - }, - "blessed": { - "version": "0.1.81", - "resolved": "https://registry.npmjs.org/blessed/-/blessed-0.1.81.tgz", - "integrity": "sha1-+WLWh+wsNpVwrnGvhDJW5tDKESk=", - "dev": true - }, - "bluebird": { - "version": "3.4.7", - "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.4.7.tgz", - "integrity": "sha1-9y12C+Cbf3bQjtj66Ysomo0F+rM=" - }, - "bodec": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/bodec/-/bodec-0.1.0.tgz", - "integrity": "sha1-vIUVVUMPI8n3ZQp172TGqUw0GMw=", - "dev": true - }, - "body-parser": { - "version": "1.19.2", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.2.tgz", - "integrity": "sha512-SAAwOxgoCKMGs9uUAUFHygfLAyaniaoun6I8mFY9pRAJL9+Kec34aU+oIjDhTycub1jozEfEwx1W1IuOYxVSFw==", - "requires": { - "bytes": "3.1.2", - "content-type": "~1.0.4", - "debug": "2.6.9", - "depd": "~1.1.2", - "http-errors": "1.8.1", - "iconv-lite": "0.4.24", - "on-finished": "~2.3.0", - "qs": "6.9.7", - "raw-body": "2.4.3", - "type-is": "~1.6.18" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "buffer": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", - "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", - "requires": { - "base64-js": "^1.3.1", - "ieee754": "^1.1.13" - } - }, - "buffer-crc32": { - "version": "0.2.13", - "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", - "integrity": "sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=" - }, - "buffer-equal-constant-time": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", - "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" - }, - "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 - }, - "buffer-indexof-polyfill": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/buffer-indexof-polyfill/-/buffer-indexof-polyfill-1.0.2.tgz", - "integrity": "sha512-I7wzHwA3t1/lwXQh+A5PbNvJxgfo5r3xulgpYDB5zckTu/Z9oUK9biouBKQUjEqzaz3HnAT6TYoovmE+GqSf7A==" - }, - "buffers": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/buffers/-/buffers-0.1.1.tgz", - "integrity": "sha1-skV5w77U1tOWru5tmorn9Ugqt7s=" - }, - "bytes": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", - "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==" - }, - "chainsaw": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/chainsaw/-/chainsaw-0.1.0.tgz", - "integrity": "sha1-XqtQsor+WAdNDVgpE4iCi15fvJg=", - "requires": { - "traverse": ">=0.3.0 <0.4" - } - }, - "chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "dev": true, - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - } - }, - "charm": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/charm/-/charm-0.1.2.tgz", - "integrity": "sha1-BsIe7RobBq62dVPNxT4jJ0usIpY=", - "dev": true - }, - "chokidar": { - "version": "3.5.3", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", - "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", - "dev": true, - "requires": { - "anymatch": "~3.1.2", - "braces": "~3.0.2", - "fsevents": "~2.3.2", - "glob-parent": "~5.1.2", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.6.0" - } - }, - "chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==" - }, - "cli-tableau": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/cli-tableau/-/cli-tableau-2.0.1.tgz", - "integrity": "sha512-he+WTicka9cl0Fg/y+YyxcN6/bfQ/1O3QmgxRXDhABKqLzvoOSM4fMzp39uMyLBulAFuywD2N7UaoQE7WaADxQ==", - "dev": true, - "requires": { - "chalk": "3.0.0" - } - }, - "cliui": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", - "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^7.0.0" - } - }, - "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==", - "requires": { - "color-name": "~1.1.4" - } - }, - "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==" - }, - "component-emitter": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", - "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==" - }, - "compress-commons": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/compress-commons/-/compress-commons-4.1.1.tgz", - "integrity": "sha512-QLdDLCKNV2dtoTorqgxngQCMA+gWXkM/Nwu7FpeBhk/RdkzimqC3jueb/FDmaZeXh+uby1jkBqE3xArsLBE5wQ==", - "requires": { - "buffer-crc32": "^0.2.13", - "crc32-stream": "^4.0.2", - "normalize-path": "^3.0.0", - "readable-stream": "^3.6.0" - } - }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" - }, - "content-disposition": { - "version": "0.5.4", - "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", - "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", - "requires": { - "safe-buffer": "5.2.1" - } - }, - "content-type": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", - "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" - }, - "continuation-local-storage": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/continuation-local-storage/-/continuation-local-storage-3.2.1.tgz", - "integrity": "sha512-jx44cconVqkCEEyLSKWwkvUXwO561jXMa3LPjTPsm5QR22PA0/mhe33FT4Xb5y74JDvt/Cq+5lm8S8rskLv9ZA==", - "dev": true, - "requires": { - "async-listener": "^0.6.0", - "emitter-listener": "^1.1.1" - } - }, - "cookie": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.1.tgz", - "integrity": "sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA==" - }, - "cookie-signature": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", - "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" - }, - "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==" - }, - "cors": { - "version": "2.8.5", - "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", - "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", - "requires": { - "object-assign": "^4", - "vary": "^1" - } - }, - "crc-32": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.1.tgz", - "integrity": "sha512-Dn/xm/1vFFgs3nfrpEVScHoIslO9NZRITWGz/1E/St6u4xw99vfZzVkW0OSnzx2h9egej9xwMCEut6sqwokM/w==", - "requires": { - "exit-on-epipe": "~1.0.1", - "printj": "~1.3.1" - } - }, - "crc32-stream": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/crc32-stream/-/crc32-stream-4.0.2.tgz", - "integrity": "sha512-DxFZ/Hk473b/muq1VJ///PMNLj0ZMnzye9thBpmjpJKCc5eMgB95aK8zCGrGfQ90cWo561Te6HK9D+j4KPdM6w==", - "requires": { - "crc-32": "^1.2.0", - "readable-stream": "^3.4.0" - } - }, - "croner": { - "version": "4.1.97", - "resolved": "https://registry.npmjs.org/croner/-/croner-4.1.97.tgz", - "integrity": "sha512-/f6gpQuxDaqXu+1kwQYSckUglPaOrHdbIlBAu0YuW8/Cdb45XwXYNUBXg3r/9Mo6n540Kn/smKcZWko5x99KrQ==", - "dev": true - }, - "cross-fetch": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.5.tgz", - "integrity": "sha512-lvb1SBsI0Z7GDwmuid+mU3kWVBwTVUbe7S0H52yaaAdQOXq2YktTCZdlAcNKFzE6QtRz0snpw9bNiPeOIkkQvw==", - "requires": { - "node-fetch": "2.6.7" - } - }, - "culvert": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/culvert/-/culvert-0.1.2.tgz", - "integrity": "sha1-lQL18BVKLVoioCPnn3HMk2+m728=", - "dev": true - }, - "data-uri-to-buffer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-3.0.1.tgz", - "integrity": "sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og==", - "dev": true - }, - "dayjs": { - "version": "1.10.7", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.10.7.tgz", - "integrity": "sha512-P6twpd70BcPK34K26uJ1KT3wlhpuOAPoMwJzpsIWUxHZ7wpmbdZL/hQqBDfz7hGurYSa5PhzdhDHtt319hL3ig==" - }, - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", - "requires": { - "ms": "2.1.2" - } - }, - "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 - }, - "degenerator": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/degenerator/-/degenerator-3.0.2.tgz", - "integrity": "sha512-c0mef3SNQo56t6urUU6tdQAs+ThoD0o9B9MJ8HEt7NQcGEILCRFqQb7ZbP9JAv+QF1Ky5plydhMR/IrqWDm+TQ==", - "dev": true, - "requires": { - "ast-types": "^0.13.2", - "escodegen": "^1.8.1", - "esprima": "^4.0.0", - "vm2": "^3.9.8" - } - }, - "depd": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", - "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" - }, - "destroy": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", - "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" - }, - "devtools-protocol": { - "version": "0.0.960912", - "resolved": "https://registry.npmjs.org/devtools-protocol/-/devtools-protocol-0.0.960912.tgz", - "integrity": "sha512-I3hWmV9rWHbdnUdmMKHF2NuYutIM2kXz2mdXW8ha7TbRlGTVs+PF+PsB5QWvpCek4Fy9B+msiispCfwlhG5Sqg==" - }, - "dotenv": { - "version": "11.0.0", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-11.0.0.tgz", - "integrity": "sha512-Fp/b504Y5W+e+FpCxTFMUZ7ZEQkQYF0rx+KZtmwixJxGQbLHrhCwo3FjZgNC8vIfrSi29PABNbMoCGD9YoiXbQ==" - }, - "duplexer2": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.1.4.tgz", - "integrity": "sha1-ixLauHjA1p4+eJEFFmKjL8a93ME=", - "requires": { - "readable-stream": "^2.0.2" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "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" - } - }, - "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==" - }, - "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==", - "requires": { - "safe-buffer": "~5.1.0" - } - } - } - }, - "duplexify": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-4.1.2.tgz", - "integrity": "sha512-fz3OjcNCHmRP12MJoZMPglx8m4rrFP8rovnk4vT8Fs+aonZoCwGg10dSsQsfP/E62eZcPTMSMP6686fu9Qlqtw==", - "requires": { - "end-of-stream": "^1.4.1", - "inherits": "^2.0.3", - "readable-stream": "^3.1.1", - "stream-shift": "^1.0.0" - } - }, - "ecdsa-sig-formatter": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", - "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", - "requires": { - "safe-buffer": "^5.0.1" - } - }, - "ee-first": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", - "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" - }, - "emitter-listener": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/emitter-listener/-/emitter-listener-1.1.2.tgz", - "integrity": "sha512-Bt1sBAGFHY9DKY+4/2cV6izcKJUf5T7/gkdmkxzX/qv9CcGH8xSwVRW5mtX03SWJtRTWSOpzCuWN9rBFYZepZQ==", - "dev": true, - "requires": { - "shimmer": "^1.2.0" - } - }, - "emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" - }, - "encodeurl": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", - "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" - }, - "end-of-stream": { - "version": "1.4.4", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", - "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", - "requires": { - "once": "^1.4.0" - } - }, - "engine.io": { - "version": "6.1.2", - "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-6.1.2.tgz", - "integrity": "sha512-v/7eGHxPvO2AWsksyx2PUsQvBafuvqs0jJJQ0FdmJG1b9qIvgSbqDRGwNhfk2XHaTTbTXiC4quRE8Q9nRjsrQQ==", - "requires": { - "@types/cookie": "^0.4.1", - "@types/cors": "^2.8.12", - "@types/node": ">=10.0.0", - "accepts": "~1.3.4", - "base64id": "2.0.0", - "cookie": "~0.4.1", - "cors": "~2.8.5", - "debug": "~4.3.1", - "engine.io-parser": "~5.0.0", - "ws": "~8.2.3" - } - }, - "engine.io-parser": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-5.0.3.tgz", - "integrity": "sha512-BtQxwF27XUNnSafQLvDi0dQ8s3i6VgzSoQMJacpIcGNrlUdfHSKbgm3jmjCVvQluGzqwujQMPAoMai3oYSTurg==", - "requires": { - "@socket.io/base64-arraybuffer": "~1.0.2" - } - }, - "enquirer": { - "version": "2.3.6", - "resolved": "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz", - "integrity": "sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg==", - "dev": true, - "requires": { - "ansi-colors": "^4.1.1" - } - }, - "escalade": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", - "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==" - }, - "escape-html": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", - "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" - }, - "escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true - }, - "escodegen": { - "version": "1.14.3", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.3.tgz", - "integrity": "sha512-qFcX0XJkdg+PB3xjZZG/wKSuT1PnQWx57+TVSjIMmILd2yC/6ByYElPwJnslDsuWuSAp4AwJGumarAAmJch5Kw==", - "dev": true, - "requires": { - "esprima": "^4.0.1", - "estraverse": "^4.2.0", - "esutils": "^2.0.2", - "optionator": "^0.8.1", - "source-map": "~0.6.1" - } - }, - "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 - }, - "estraverse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", - "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", - "dev": true - }, - "esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true - }, - "etag": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", - "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" - }, - "event-target-shim": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", - "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==" - }, - "eventemitter2": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-5.0.1.tgz", - "integrity": "sha1-YZegldX7a1folC9v1+qtY6CclFI=", - "dev": true - }, - "exceljs": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/exceljs/-/exceljs-4.3.0.tgz", - "integrity": "sha512-hTAeo5b5TPvf8Z02I2sKIT4kSfCnOO2bCxYX8ABqODCdAjppI3gI9VYiGCQQYVcBaBSKlFDMKlAQRqC+kV9O8w==", - "requires": { - "archiver": "^5.0.0", - "dayjs": "^1.8.34", - "fast-csv": "^4.3.1", - "jszip": "^3.5.0", - "readable-stream": "^3.6.0", - "saxes": "^5.0.1", - "tmp": "^0.2.0", - "unzipper": "^0.10.11", - "uuid": "^8.3.0" - } - }, - "exit-on-epipe": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/exit-on-epipe/-/exit-on-epipe-1.0.1.tgz", - "integrity": "sha512-h2z5mrROTxce56S+pnvAV890uu7ls7f1kEvVGJbw1OlFH3/mlJ5bkXu0KRyW94v37zzHPiUd55iLn3DA7TjWpw==" - }, - "express": { - "version": "4.17.3", - "resolved": "https://registry.npmjs.org/express/-/express-4.17.3.tgz", - "integrity": "sha512-yuSQpz5I+Ch7gFrPCk4/c+dIBKlQUxtgwqzph132bsT6qhuzss6I8cLJQz7B3rFblzd6wtcI0ZbGltH/C4LjUg==", - "requires": { - "accepts": "~1.3.8", - "array-flatten": "1.1.1", - "body-parser": "1.19.2", - "content-disposition": "0.5.4", - "content-type": "~1.0.4", - "cookie": "0.4.2", - "cookie-signature": "1.0.6", - "debug": "2.6.9", - "depd": "~1.1.2", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "etag": "~1.8.1", - "finalhandler": "~1.1.2", - "fresh": "0.5.2", - "merge-descriptors": "1.0.1", - "methods": "~1.1.2", - "on-finished": "~2.3.0", - "parseurl": "~1.3.3", - "path-to-regexp": "0.1.7", - "proxy-addr": "~2.0.7", - "qs": "6.9.7", - "range-parser": "~1.2.1", - "safe-buffer": "5.2.1", - "send": "0.17.2", - "serve-static": "1.14.2", - "setprototypeof": "1.2.0", - "statuses": "~1.5.0", - "type-is": "~1.6.18", - "utils-merge": "1.0.1", - "vary": "~1.1.2" - }, - "dependencies": { - "cookie": { - "version": "0.4.2", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.2.tgz", - "integrity": "sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==" - }, - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" - }, - "extract-zip": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-2.0.1.tgz", - "integrity": "sha512-GDhU9ntwuKyGXdZBUgTIe+vXnWj0fppUEtMDL0+idd5Sta8TGpHssn/eusA9mrPr9qNDym6SxAYZjNvCn/9RBg==", - "requires": { - "@types/yauzl": "^2.9.1", - "debug": "^4.1.1", - "get-stream": "^5.1.0", - "yauzl": "^2.10.0" - } - }, - "fast-csv": { - "version": "4.3.6", - "resolved": "https://registry.npmjs.org/fast-csv/-/fast-csv-4.3.6.tgz", - "integrity": "sha512-2RNSpuwwsJGP0frGsOmTb9oUF+VkFSM4SyLTDgwf2ciHWTarN0lQTC+F2f/t5J9QjW+c65VFIAAu85GsvMIusw==", - "requires": { - "@fast-csv/format": "4.3.5", - "@fast-csv/parse": "4.3.6" - } - }, - "fast-json-patch": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/fast-json-patch/-/fast-json-patch-3.1.0.tgz", - "integrity": "sha512-IhpytlsVTRndz0hU5t0/MGzS/etxLlfrpG5V5M9mVbuj9TrJLWaMfsox9REM5rkuGX0T+5qjpe8XA1o0gZ42nA==", - "dev": true - }, - "fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", - "dev": true - }, - "fast-text-encoding": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/fast-text-encoding/-/fast-text-encoding-1.0.3.tgz", - "integrity": "sha512-dtm4QZH9nZtcDt8qJiOH9fcQd1NAgi+K1O2DbE6GG1PPCK/BWfOH3idCTRQ4ImXRUOyopDEgDEnVEE7Y/2Wrig==" - }, - "fclone": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/fclone/-/fclone-1.0.11.tgz", - "integrity": "sha1-EOhdo4v+p/xZk0HClu4ddyZu5kA=", - "dev": true - }, - "fd-slicer": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.1.0.tgz", - "integrity": "sha1-JcfInLH5B3+IkbvmHY85Dq4lbx4=", - "requires": { - "pend": "~1.2.0" - } - }, - "file-type": { - "version": "16.5.3", - "resolved": "https://registry.npmjs.org/file-type/-/file-type-16.5.3.tgz", - "integrity": "sha512-uVsl7iFhHSOY4bEONLlTK47iAHtNsFHWP5YE4xJfZ4rnX7S1Q3wce09XgqSC7E/xh8Ncv/be1lNoyprlUH/x6A==", - "requires": { - "readable-web-to-node-stream": "^3.0.0", - "strtok3": "^6.2.4", - "token-types": "^4.1.1" - } - }, - "file-uri-to-path": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-2.0.0.tgz", - "integrity": "sha512-hjPFI8oE/2iQPVe4gbrJ73Pp+Xfub2+WI2LlXDbsaJBwT5wuMh35WNWVYYTpnz895shtwfyutMFLFywpQAFdLg==", - "dev": true - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "finalhandler": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", - "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", - "requires": { - "debug": "2.6.9", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "on-finished": "~2.3.0", - "parseurl": "~1.3.3", - "statuses": "~1.5.0", - "unpipe": "~1.0.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "fluent-ffmpeg": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/fluent-ffmpeg/-/fluent-ffmpeg-2.1.2.tgz", - "integrity": "sha1-yVLeIkD4EuvaCqgAbXd27irPfXQ=", - "requires": { - "async": ">=0.2.9", - "which": "^1.1.1" - } - }, - "follow-redirects": { - "version": "1.14.9", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.9.tgz", - "integrity": "sha512-MQDfihBQYMcyy5dhRDJUHcw7lb2Pv/TuE6xP1vyraLukNDHKbDxDNaOE3NbCAdKQApno+GPRyo1YAp89yCjK4w==", - "dev": true - }, - "forwarded": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", - "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" - }, - "fresh": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", - "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" - }, - "fs-constants": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", - "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==" - }, - "fs-extra": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", - "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", - "dev": true, - "requires": { - "graceful-fs": "^4.2.0", - "jsonfile": "^4.0.0", - "universalify": "^0.1.0" - } - }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" - }, - "fsevents": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", - "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", - "dev": true, - "optional": true - }, - "fstream": { - "version": "1.0.12", - "resolved": "https://registry.npmjs.org/fstream/-/fstream-1.0.12.tgz", - "integrity": "sha512-WvJ193OHa0GHPEL+AycEJgxvBEwyfRkN1vhjca23OaPVMCaLCXTd5qAu82AjTcgP1UJmytkOKb63Ypde7raDIg==", - "requires": { - "graceful-fs": "^4.1.2", - "inherits": "~2.0.0", - "mkdirp": ">=0.5 0", - "rimraf": "2" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "ftp": { - "version": "0.3.10", - "resolved": "https://registry.npmjs.org/ftp/-/ftp-0.3.10.tgz", - "integrity": "sha1-kZfYYa2BQvPmPVqDv+TFn3MwiF0=", - "dev": true, - "requires": { - "readable-stream": "1.1.x", - "xregexp": "2.0.0" - }, - "dependencies": { - "isarray": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", - "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=", - "dev": true - }, - "readable-stream": { - "version": "1.1.14", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", - "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", - "dev": true, - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.1", - "isarray": "0.0.1", - "string_decoder": "~0.10.x" - } - }, - "string_decoder": { - "version": "0.10.31", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", - "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=", - "dev": true - } - } - }, - "function-bind": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", - "dev": true - }, - "gaxios": { - "version": "4.3.2", - "resolved": "https://registry.npmjs.org/gaxios/-/gaxios-4.3.2.tgz", - "integrity": "sha512-T+ap6GM6UZ0c4E6yb1y/hy2UB6hTrqhglp3XfmU9qbLCGRYhLVV5aRPpC4EmoG8N8zOnkYCgoBz+ScvGAARY6Q==", - "requires": { - "abort-controller": "^3.0.0", - "extend": "^3.0.2", - "https-proxy-agent": "^5.0.0", - "is-stream": "^2.0.0", - "node-fetch": "^2.6.1" - } - }, - "gcp-metadata": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/gcp-metadata/-/gcp-metadata-4.3.1.tgz", - "integrity": "sha512-x850LS5N7V1F3UcV7PoupzGsyD6iVwTVvsh3tbXfkctZnBnjW5yu5z1/3k3SehF7TyoTIe78rJs02GMMy+LF+A==", - "requires": { - "gaxios": "^4.0.0", - "json-bigint": "^1.0.0" - } - }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==" - }, - "get-stream": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", - "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", - "requires": { - "pump": "^3.0.0" - } - }, - "get-uri": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/get-uri/-/get-uri-3.0.2.tgz", - "integrity": "sha512-+5s0SJbGoyiJTZZ2JTpFPLMPSch72KEqGOTvQsBqg0RBWvwhWUSYZFAtz3TPW0GXJuLBJPts1E241iHg+VRfhg==", - "dev": true, - "requires": { - "@tootallnate/once": "1", - "data-uri-to-buffer": "3", - "debug": "4", - "file-uri-to-path": "2", - "fs-extra": "^8.1.0", - "ftp": "^0.3.10" - } - }, - "git-node-fs": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/git-node-fs/-/git-node-fs-1.0.0.tgz", - "integrity": "sha1-SbIV4kLr5Dqkx1Ybu6SZUhdSCA8=", - "dev": true - }, - "git-sha1": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/git-sha1/-/git-sha1-0.1.2.tgz", - "integrity": "sha1-WZrBkrcYdYJeE6RF86bgURjC90U=", - "dev": true - }, - "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" - } - }, - "glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } - }, - "google-auth-library": { - "version": "7.12.0", - "resolved": "https://registry.npmjs.org/google-auth-library/-/google-auth-library-7.12.0.tgz", - "integrity": "sha512-RS/whvFPMoF1hQNxnoVET3DWKPBt1Xgqe2rY0k+Jn7TNhoHlwdnSe7Rlcbo2Nub3Mt2lUVz26X65aDQrWp6x8w==", - "requires": { - "arrify": "^2.0.0", - "base64-js": "^1.3.0", - "ecdsa-sig-formatter": "^1.0.11", - "fast-text-encoding": "^1.0.0", - "gaxios": "^4.0.0", - "gcp-metadata": "^4.2.0", - "gtoken": "^5.0.4", - "jws": "^4.0.0", - "lru-cache": "^6.0.0" - } - }, - "google-gax": { - "version": "2.29.7", - "resolved": "https://registry.npmjs.org/google-gax/-/google-gax-2.29.7.tgz", - "integrity": "sha512-yC0Q4OqX6s081jV72Idt9sxZnuRHOjg0SR0FFH15gT3LDE2u/78xqLjZwa3VuprBvaOLM29jzU19Vv4I7E8ETQ==", - "requires": { - "@grpc/grpc-js": "~1.5.0", - "@grpc/proto-loader": "^0.6.1", - "@types/long": "^4.0.0", - "abort-controller": "^3.0.0", - "duplexify": "^4.0.0", - "fast-text-encoding": "^1.0.3", - "google-auth-library": "^7.6.1", - "is-stream-ended": "^0.1.4", - "node-fetch": "^2.6.1", - "object-hash": "^2.1.1", - "proto3-json-serializer": "^0.1.8", - "protobufjs": "6.11.2", - "retry-request": "^4.0.0" - } - }, - "google-p12-pem": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/google-p12-pem/-/google-p12-pem-3.1.3.tgz", - "integrity": "sha512-MC0jISvzymxePDVembypNefkAQp+DRP7dBE+zNUPaIjEspIlYg0++OrsNr248V9tPbz6iqtZ7rX1hxWA5B8qBQ==", - "requires": { - "node-forge": "^1.0.0" - } - }, - "graceful-fs": { - "version": "4.2.9", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.9.tgz", - "integrity": "sha512-NtNxqUcXgpW2iMrfqSfR73Glt39K+BLwWsPs94yR63v45T0Wbej7eRmL5cWfwEgqXnmjQp3zaJTshdRW/qC2ZQ==" - }, - "gtoken": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/gtoken/-/gtoken-5.3.2.tgz", - "integrity": "sha512-gkvEKREW7dXWF8NV8pVrKfW7WqReAmjjkMBh6lNCCGOM4ucS0r0YyXXl0r/9Yj8wcW/32ISkfc8h5mPTDbtifQ==", - "requires": { - "gaxios": "^4.0.0", - "google-p12-pem": "^3.1.3", - "jws": "^4.0.0" - } - }, - "has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", - "dev": true, - "requires": { - "function-bind": "^1.1.1" - } - }, - "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 - }, - "http-errors": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.1.tgz", - "integrity": "sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g==", - "requires": { - "depd": "~1.1.2", - "inherits": "2.0.4", - "setprototypeof": "1.2.0", - "statuses": ">= 1.5.0 < 2", - "toidentifier": "1.0.1" - } - }, - "http-proxy-agent": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", - "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", - "dev": true, - "requires": { - "@tootallnate/once": "1", - "agent-base": "6", - "debug": "4" - } - }, - "https-proxy-agent": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", - "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", - "requires": { - "agent-base": "6", - "debug": "4" - } - }, - "iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", - "requires": { - "safer-buffer": ">= 2.1.2 < 3" - } - }, - "ieee754": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" - }, - "immediate": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", - "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=" - }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "requires": { - "once": "^1.3.0", - "wrappy": "1" - } - }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" - }, - "ini": { - "version": "1.3.8", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", - "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", - "dev": true - }, - "ip": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", - "integrity": "sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=", - "dev": true - }, - "ipaddr.js": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", - "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" - }, - "is-binary-path": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", - "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", - "dev": true, - "requires": { - "binary-extensions": "^2.0.0" - } - }, - "is-core-module": { - "version": "2.8.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.8.1.tgz", - "integrity": "sha512-SdNCUs284hr40hFTFP6l0IfZ/RSrMXF3qgoRHd3/79unUTvrFO/JoXwkGm+5J/Oe3E/b5GsnG330uUNgRpu1PA==", - "dev": true, - "requires": { - "has": "^1.0.3" - } - }, - "is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" - }, - "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, - "requires": { - "is-extglob": "^2.1.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "is-stream": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", - "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==" - }, - "is-stream-ended": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/is-stream-ended/-/is-stream-ended-0.1.4.tgz", - "integrity": "sha512-xj0XPvmr7bQFTvirqnFr50o0hQIh6ZItDqloxt5aJrR4NQsYeSsyFQERYGCAzfindAcnKjINnwEEgLx4IqVzQw==" - }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" - }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=" - }, - "js-git": { - "version": "0.7.8", - "resolved": "https://registry.npmjs.org/js-git/-/js-git-0.7.8.tgz", - "integrity": "sha1-UvplWrYYd9bxB578ZTS1VPMeVEQ=", - "dev": true, - "requires": { - "bodec": "^0.1.0", - "culvert": "^0.1.2", - "git-sha1": "^0.1.2", - "pako": "^0.2.5" - }, - "dependencies": { - "pako": { - "version": "0.2.9", - "resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz", - "integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU=", - "dev": true - } - } - }, - "json-bigint": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/json-bigint/-/json-bigint-1.0.0.tgz", - "integrity": "sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ==", - "requires": { - "bignumber.js": "^9.0.0" - } - }, - "json-stringify-safe": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", - "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", - "dev": true, - "optional": true - }, - "jsonfile": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", - "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.6" - } - }, - "jsqr": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/jsqr/-/jsqr-1.4.0.tgz", - "integrity": "sha512-dxLob7q65Xg2DvstYkRpkYtmKm2sPJ9oFhrhmudT1dZvNFFTlroai3AWSpLey/w5vMcLBXRgOJsbXpdN9HzU/A==" - }, - "jszip": { - "version": "3.7.1", - "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.7.1.tgz", - "integrity": "sha512-ghL0tz1XG9ZEmRMcEN2vt7xabrDdqHHeykgARpmZ0BiIctWxM47Vt63ZO2dnp4QYt/xJVLLy5Zv1l/xRdh2byg==", - "requires": { - "lie": "~3.3.0", - "pako": "~1.0.2", - "readable-stream": "~2.3.6", - "set-immediate-shim": "~1.0.1" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "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" - } - }, - "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==" - }, - "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==", - "requires": { - "safe-buffer": "~5.1.0" - } - } - } - }, - "jwa": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.0.tgz", - "integrity": "sha512-jrZ2Qx916EA+fq9cEAeCROWPTfCwi1IVHqT2tapuqLEVVDKFDENFw1oL+MwrTvH6msKxsd1YTDVw6uKEcsrLEA==", - "requires": { - "buffer-equal-constant-time": "1.0.1", - "ecdsa-sig-formatter": "1.0.11", - "safe-buffer": "^5.0.1" - } - }, - "jws": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/jws/-/jws-4.0.0.tgz", - "integrity": "sha512-KDncfTmOZoOMTFG4mBlG0qUIOlc03fmzH+ru6RgYVZhPkyiy/92Owlt/8UEN+a4TXR1FQetfIpJE8ApdvdVxTg==", - "requires": { - "jwa": "^2.0.0", - "safe-buffer": "^5.0.1" - } - }, - "lazy": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/lazy/-/lazy-1.0.11.tgz", - "integrity": "sha1-2qBoIGKCVCwIgojpdcKXwa53tpA=", - "dev": true - }, - "lazystream": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.1.tgz", - "integrity": "sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw==", - "requires": { - "readable-stream": "^2.0.5" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "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" - } - }, - "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==" - }, - "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==", - "requires": { - "safe-buffer": "~5.1.0" - } - } - } - }, - "levn": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", - "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", - "dev": true, - "requires": { - "prelude-ls": "~1.1.2", - "type-check": "~0.3.2" - } - }, - "lie": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", - "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", - "requires": { - "immediate": "~3.0.5" - } - }, - "listenercount": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/listenercount/-/listenercount-1.0.1.tgz", - "integrity": "sha1-hMinKrWcRyUyFIDJdeZQg0LnCTc=" - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "requires": { - "p-locate": "^4.1.0" - } - }, - "lodash": { - "version": "4.17.21", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", - "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", - "dev": true - }, - "lodash.camelcase": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz", - "integrity": "sha1-soqmKIorn8ZRA1x3EfZathkDMaY=" - }, - "lodash.defaults": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", - "integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw=" - }, - "lodash.difference": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", - "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=" - }, - "lodash.escaperegexp": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/lodash.escaperegexp/-/lodash.escaperegexp-4.1.2.tgz", - "integrity": "sha1-ZHYsSGGAglGKw99Mz11YhtriA0c=" - }, - "lodash.flatten": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", - "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8=" - }, - "lodash.groupby": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/lodash.groupby/-/lodash.groupby-4.6.0.tgz", - "integrity": "sha1-Cwih3PaDl8OXhVwyOXg4Mt90A9E=" - }, - "lodash.isboolean": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", - "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" - }, - "lodash.isequal": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", - "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=" - }, - "lodash.isfunction": { - "version": "3.0.9", - "resolved": "https://registry.npmjs.org/lodash.isfunction/-/lodash.isfunction-3.0.9.tgz", - "integrity": "sha512-AirXNj15uRIMMPihnkInB4i3NHeb4iBtNg9WRWuK2o31S+ePwwNmDPaTL3o7dTJ+VXNZim7rFs4rxN4YU1oUJw==" - }, - "lodash.isnil": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/lodash.isnil/-/lodash.isnil-4.0.0.tgz", - "integrity": "sha1-SeKM1VkBNFjIFMVHnTxmOiG/qmw=" - }, - "lodash.isplainobject": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", - "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" - }, - "lodash.isundefined": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/lodash.isundefined/-/lodash.isundefined-3.0.1.tgz", - "integrity": "sha1-I+89lTVWUgOmbO/VuDD4SJEa+0g=" - }, - "lodash.union": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/lodash.union/-/lodash.union-4.6.0.tgz", - "integrity": "sha1-SLtQiECfFvGCFmZkHETdGqrjzYg=" - }, - "lodash.uniq": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", - "integrity": "sha1-0CJTc662Uq3BvILklFM5qEJ1R3M=" - }, - "log-driver": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/log-driver/-/log-driver-1.2.7.tgz", - "integrity": "sha512-U7KCmLdqsGHBLeWqYlFA0V0Sl6P08EE1ZrmA9cxjUE0WVqT9qnyVDPz1kzpFEP0jdJuFnasWIfSd7fsaNXkpbg==", - "dev": true - }, - "long": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz", - "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA==" - }, - "lru-cache": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", - "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", - "requires": { - "yallist": "^4.0.0" - } - }, - "media-typer": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", - "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" - }, - "merge-descriptors": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", - "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" - }, - "methods": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", - "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" - }, - "mime": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", - "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" - }, - "mime-db": { - "version": "1.52.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", - "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" - }, - "mime-types": { - "version": "2.1.34", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.34.tgz", - "integrity": "sha512-6cP692WwGIs9XXdOO4++N+7qjqv0rqxxVvJ3VHPh/Sc9mVZcQP+ZGhkKiTvWMQRr2tbHkJP/Yn7Y0npb3ZBs4A==", - "requires": { - "mime-db": "1.51.0" - }, - "dependencies": { - "mime-db": { - "version": "1.51.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.51.0.tgz", - "integrity": "sha512-5y8A56jg7XVQx2mbv1lu49NR4dokRnhZYTtL+KGfaa27uq4pSTXkwQkFJl4pkRMyNFz/EtYDSkiiEHx3F7UN6g==" - } - } - }, - "minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", - "requires": { - "brace-expansion": "^1.1.7" - } - }, - "minimist": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", - "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" - }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", - "requires": { - "minimist": "^1.2.5" - } - }, - "mkdirp-classic": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", - "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==" - }, - "module-details-from-path": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/module-details-from-path/-/module-details-from-path-1.0.3.tgz", - "integrity": "sha1-EUyUlnPiqKNenTV4hSeqN7Z52is=", - "dev": true - }, - "moment": { - "version": "2.29.1", - "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", - "integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==" - }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" - }, - "mute-stream": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", - "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", - "dev": true - }, - "mysql": { - "version": "2.18.1", - "resolved": "https://registry.npmjs.org/mysql/-/mysql-2.18.1.tgz", - "integrity": "sha512-Bca+gk2YWmqp2Uf6k5NFEurwY/0td0cpebAucFpY/3jhrwrVGuxU2uQFCHjU19SJfje0yQvi+rVWdq78hR5lig==", - "requires": { - "bignumber.js": "9.0.0", - "readable-stream": "2.3.7", - "safe-buffer": "5.1.2", - "sqlstring": "2.3.1" - }, - "dependencies": { - "bignumber.js": { - "version": "9.0.0", - "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.0.0.tgz", - "integrity": "sha512-t/OYhhJ2SD+YGBQcjY8GzzDHEk9f3nerxjtfa6tlMXfe7frs/WozhvCNoGvpM0P3bNf3Gq5ZRMlGr5f3r4/N8A==" - }, - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "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" - } - }, - "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==" - }, - "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==", - "requires": { - "safe-buffer": "~5.1.0" - } - } - } - }, - "nanoid": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.1.tgz", - "integrity": "sha512-n6Vs/3KGyxPQd6uO0eH4Bv0ojGSUvuLlIHtC3Y0kEO23YRge8H9x1GCzLn28YX0H66pMkxuaeESFq4tKISKwdw==" - }, - "needle": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/needle/-/needle-2.4.0.tgz", - "integrity": "sha512-4Hnwzr3mi5L97hMYeNl8wRW/Onhy4nUKR/lVemJ8gJedxxUyBLm9kkrDColJvoSfwi0jCNhD+xCdOtiGDQiRZg==", - "dev": true, - "requires": { - "debug": "^3.2.6", - "iconv-lite": "^0.4.4", - "sax": "^1.2.4" - }, - "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - } - } - }, - "negotiator": { - "version": "0.6.3", - "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", - "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==" - }, - "netmask": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/netmask/-/netmask-2.0.2.tgz", - "integrity": "sha512-dBpDMdxv9Irdq66304OLfEmQ9tbNRFnFTuZiLo+bD+r332bBmMJ8GBLXklIXXgxd3+v9+KUnZaUR5PJMa75Gsg==", - "dev": true - }, - "node-fetch": { - "version": "2.6.7", - "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz", - "integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==", - "requires": { - "whatwg-url": "^5.0.0" - } - }, - "node-forge": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-1.2.1.tgz", - "integrity": "sha512-Fcvtbb+zBcZXbTTVwqGA5W+MKBj56UjVRevvchv5XrcyXbmNdesfZL37nlcWOfpgHhgmxApw3tQbTr4CqNmX4w==" - }, - "node-webpmux": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/node-webpmux/-/node-webpmux-3.1.1.tgz", - "integrity": "sha512-vG75BAe9zKghN+Y+XsJMPdOfVyesn1MmGvd/DMxeQ6gtpB3U053yCWXO1Gl2QWXTfU1++7flTihv/yB6EEdtKQ==" - }, - "normalize-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" - }, - "nssocket": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/nssocket/-/nssocket-0.6.0.tgz", - "integrity": "sha1-Wflvb/MhVm8zxw99vu7N/cBxVPo=", - "dev": true, - "requires": { - "eventemitter2": "~0.4.14", - "lazy": "~1.0.11" - }, - "dependencies": { - "eventemitter2": { - "version": "0.4.14", - "resolved": "https://registry.npmjs.org/eventemitter2/-/eventemitter2-0.4.14.tgz", - "integrity": "sha1-j2G3XN4BKy6esoTUVFWDtWQ7Yas=", - "dev": true - } - } - }, - "object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" - }, - "object-hash": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-2.2.0.tgz", - "integrity": "sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw==" - }, - "on-finished": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", - "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", - "requires": { - "ee-first": "1.1.1" - } - }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "requires": { - "wrappy": "1" - } - }, - "optionator": { - "version": "0.8.3", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", - "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", - "dev": true, - "requires": { - "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" - } - }, - "p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", - "requires": { - "p-try": "^2.0.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "requires": { - "p-limit": "^2.2.0" - } - }, - "p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==" - }, - "pac-proxy-agent": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/pac-proxy-agent/-/pac-proxy-agent-5.0.0.tgz", - "integrity": "sha512-CcFG3ZtnxO8McDigozwE3AqAw15zDvGH+OjXO4kzf7IkEKkQ4gxQ+3sdF50WmhQ4P/bVusXcqNE2S3XrNURwzQ==", - "dev": true, - "requires": { - "@tootallnate/once": "1", - "agent-base": "6", - "debug": "4", - "get-uri": "3", - "http-proxy-agent": "^4.0.1", - "https-proxy-agent": "5", - "pac-resolver": "^5.0.0", - "raw-body": "^2.2.0", - "socks-proxy-agent": "5" - } - }, - "pac-resolver": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/pac-resolver/-/pac-resolver-5.0.0.tgz", - "integrity": "sha512-H+/A6KitiHNNW+bxBKREk2MCGSxljfqRX76NjummWEYIat7ldVXRU3dhRIE3iXZ0nvGBk6smv3nntxKkzRL8NA==", - "dev": true, - "requires": { - "degenerator": "^3.0.1", - "ip": "^1.1.5", - "netmask": "^2.0.1" - } - }, - "pako": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", - "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==" - }, - "parseurl": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", - "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" - }, - "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==" - }, - "path-is-absolute": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" - }, - "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 - }, - "path-to-regexp": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", - "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" - }, - "peek-readable": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/peek-readable/-/peek-readable-4.1.0.tgz", - "integrity": "sha512-ZI3LnwUv5nOGbQzD9c2iDG6toheuXSZP5esSHBjopsXH4dg19soufvpUGA3uohi5anFtGb2lhAVdHzH6R/Evvg==" - }, - "pend": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", - "integrity": "sha1-elfrVQpng/kRUzH89GY9XI4AelA=" - }, - "picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true - }, - "pidusage": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pidusage/-/pidusage-3.0.0.tgz", - "integrity": "sha512-8VJLToXhj+RYZGNVw8oxc7dS54iCQXUJ+MDFHezQ/fwF5B8W4OWodAMboc1wb08S/4LiHwAmkT4ohf/d3YPPsw==", - "dev": true, - "requires": { - "safe-buffer": "^5.2.1" - } - }, - "pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", - "requires": { - "find-up": "^4.0.0" - } - }, - "pm2": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/pm2/-/pm2-5.2.0.tgz", - "integrity": "sha512-PO5hMVhQ85cTszFM++6v07Me9hPJMkFbHjkFigtMMk+La8ty2wCi2dlBTeZYJDhPUSjK8Ccltpq2buNRcyMOTw==", - "dev": true, - "requires": { - "@pm2/agent": "~2.0.0", - "@pm2/io": "~5.0.0", - "@pm2/js-api": "~0.6.7", - "@pm2/pm2-version-check": "^1.0.4", - "async": "~3.2.0", - "blessed": "0.1.81", - "chalk": "3.0.0", - "chokidar": "^3.5.1", - "cli-tableau": "^2.0.0", - "commander": "2.15.1", - "croner": "~4.1.92", - "dayjs": "~1.8.25", - "debug": "^4.3.1", - "enquirer": "2.3.6", - "eventemitter2": "5.0.1", - "fclone": "1.0.11", - "mkdirp": "1.0.4", - "needle": "2.4.0", - "pidusage": "~3.0", - "pm2-axon": "~4.0.1", - "pm2-axon-rpc": "~0.7.1", - "pm2-deploy": "~1.0.2", - "pm2-multimeter": "^0.1.2", - "pm2-sysmonit": "^1.2.8", - "promptly": "^2", - "semver": "^7.2", - "source-map-support": "0.5.19", - "sprintf-js": "1.1.2", - "vizion": "~2.2.1", - "yamljs": "0.3.0" - }, - "dependencies": { - "@pm2/pm2-version-check": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@pm2/pm2-version-check/-/pm2-version-check-1.0.4.tgz", - "integrity": "sha512-SXsM27SGH3yTWKc2fKR4SYNxsmnvuBQ9dd6QHtEWmiZ/VqaOYPAIlS8+vMcn27YLtAEBGvNRSh3TPNvtjZgfqA==", - "dev": true, - "requires": { - "debug": "^4.3.1" - } - }, - "commander": { - "version": "2.15.1", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.15.1.tgz", - "integrity": "sha512-VlfT9F3V0v+jr4yxPc5gg9s62/fIVWsd2Bk2iD435um1NlGMYdVCq+MjcXnhYq2icNOizHr1kK+5TI6H0Hy0ag==", - "dev": true - }, - "dayjs": { - "version": "1.8.36", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.8.36.tgz", - "integrity": "sha512-3VmRXEtw7RZKAf+4Tv1Ym9AGeo8r8+CjDi26x+7SYQil1UqtqdaokhzoEJohqlzt0m5kacJSDhJQkG/LWhpRBw==", - "dev": true - }, - "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", - "dev": true - } - } - }, - "pm2-axon": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/pm2-axon/-/pm2-axon-4.0.1.tgz", - "integrity": "sha512-kES/PeSLS8orT8dR5jMlNl+Yu4Ty3nbvZRmaAtROuVm9nYYGiaoXqqKQqQYzWQzMYWUKHMQTvBlirjE5GIIxqg==", - "dev": true, - "requires": { - "amp": "~0.3.1", - "amp-message": "~0.1.1", - "debug": "^4.3.1", - "escape-string-regexp": "^4.0.0" - } - }, - "pm2-axon-rpc": { - "version": "0.7.1", - "resolved": "https://registry.npmjs.org/pm2-axon-rpc/-/pm2-axon-rpc-0.7.1.tgz", - "integrity": "sha512-FbLvW60w+vEyvMjP/xom2UPhUN/2bVpdtLfKJeYM3gwzYhoTEEChCOICfFzxkxuoEleOlnpjie+n1nue91bDQw==", - "dev": true, - "requires": { - "debug": "^4.3.1" - } - }, - "pm2-deploy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/pm2-deploy/-/pm2-deploy-1.0.2.tgz", - "integrity": "sha512-YJx6RXKrVrWaphEYf++EdOOx9EH18vM8RSZN/P1Y+NokTKqYAca/ejXwVLyiEpNju4HPZEk3Y2uZouwMqUlcgg==", - "dev": true, - "requires": { - "run-series": "^1.1.8", - "tv4": "^1.3.0" - } - }, - "pm2-multimeter": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/pm2-multimeter/-/pm2-multimeter-0.1.2.tgz", - "integrity": "sha1-Gh5VFT1BoFU0zqI8/oYKuqDrSs4=", - "dev": true, - "requires": { - "charm": "~0.1.1" - } - }, - "pm2-sysmonit": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/pm2-sysmonit/-/pm2-sysmonit-1.2.8.tgz", - "integrity": "sha512-ACOhlONEXdCTVwKieBIQLSi2tQZ8eKinhcr9JpZSUAL8Qy0ajIgRtsLxG/lwPOW3JEKqPyw/UaHmTWhUzpP4kA==", - "dev": true, - "optional": true, - "requires": { - "async": "^3.2.0", - "debug": "^4.3.1", - "pidusage": "^2.0.21", - "systeminformation": "^5.7", - "tx2": "~1.0.4" - }, - "dependencies": { - "pidusage": { - "version": "2.0.21", - "resolved": "https://registry.npmjs.org/pidusage/-/pidusage-2.0.21.tgz", - "integrity": "sha512-cv3xAQos+pugVX+BfXpHsbyz/dLzX+lr44zNMsYiGxUw+kV5sgQCIcLd1z+0vq+KyC7dJ+/ts2PsfgWfSC3WXA==", - "dev": true, - "optional": true, - "requires": { - "safe-buffer": "^5.2.1" - } - } - } - }, - "prelude-ls": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", - "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", - "dev": true - }, - "prettier": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.5.1.tgz", - "integrity": "sha512-vBZcPRUR5MZJwoyi3ZoyQlc1rXeEck8KgeC9AwwOn+exuxLxq5toTRDTSaVrXHxelDMHy9zlicw8u66yxoSUFg==", - "dev": true - }, - "printj": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/printj/-/printj-1.3.1.tgz", - "integrity": "sha512-GA3TdL8szPK4AQ2YnOe/b+Y1jUFwmmGMMK/qbY7VcE3Z7FU8JstbKiKRzO6CIiAKPhTO8m01NoQ0V5f3jc4OGg==" - }, - "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==" - }, - "progress": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", - "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==" - }, - "promptly": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/promptly/-/promptly-2.2.0.tgz", - "integrity": "sha1-KhP6BjaIoqWYOxYf/wEIoH0m/HQ=", - "dev": true, - "requires": { - "read": "^1.0.4" - } - }, - "proto3-json-serializer": { - "version": "0.1.8", - "resolved": "https://registry.npmjs.org/proto3-json-serializer/-/proto3-json-serializer-0.1.8.tgz", - "integrity": "sha512-ACilkB6s1U1gWnl5jtICpnDai4VCxmI9GFxuEaYdxtDG2oVI3sVFIUsvUZcQbJgtPM6p+zqKbjTKQZp6Y4FpQw==", - "requires": { - "protobufjs": "^6.11.2" - } - }, - "protobufjs": { - "version": "6.11.2", - "resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-6.11.2.tgz", - "integrity": "sha512-4BQJoPooKJl2G9j3XftkIXjoC9C0Av2NOrWmbLWT1vH32GcSUHjM0Arra6UfTsVyfMAuFzaLucXn1sadxJydAw==", - "requires": { - "@protobufjs/aspromise": "^1.1.2", - "@protobufjs/base64": "^1.1.2", - "@protobufjs/codegen": "^2.0.4", - "@protobufjs/eventemitter": "^1.1.0", - "@protobufjs/fetch": "^1.1.0", - "@protobufjs/float": "^1.0.2", - "@protobufjs/inquire": "^1.1.0", - "@protobufjs/path": "^1.1.2", - "@protobufjs/pool": "^1.1.0", - "@protobufjs/utf8": "^1.1.0", - "@types/long": "^4.0.1", - "@types/node": ">=13.7.0", - "long": "^4.0.0" - } - }, - "proxy-addr": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", - "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", - "requires": { - "forwarded": "0.2.0", - "ipaddr.js": "1.9.1" - } - }, - "proxy-agent": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/proxy-agent/-/proxy-agent-5.0.0.tgz", - "integrity": "sha512-gkH7BkvLVkSfX9Dk27W6TyNOWWZWRilRfk1XxGNWOYJ2TuedAv1yFpCaU9QSBmBe716XOTNpYNOzhysyw8xn7g==", - "dev": true, - "requires": { - "agent-base": "^6.0.0", - "debug": "4", - "http-proxy-agent": "^4.0.0", - "https-proxy-agent": "^5.0.0", - "lru-cache": "^5.1.1", - "pac-proxy-agent": "^5.0.0", - "proxy-from-env": "^1.0.0", - "socks-proxy-agent": "^5.0.0" - }, - "dependencies": { - "lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "requires": { - "yallist": "^3.0.2" - } - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - } - } - }, - "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==" - }, - "pump": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", - "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", - "requires": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" - } - }, - "puppeteer": { - "version": "13.4.0", - "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-13.4.0.tgz", - "integrity": "sha512-WrHtFF2WpYC6KWFP4OCPOHWCjW4f8tFk+FkYZeNQ8/lHn+asjXBEXiIWauune8CY2xIHBVExGas+WI6Ay8/MgQ==", - "requires": { - "cross-fetch": "3.1.5", - "debug": "4.3.3", - "devtools-protocol": "0.0.960912", - "extract-zip": "2.0.1", - "https-proxy-agent": "5.0.0", - "pkg-dir": "4.2.0", - "progress": "2.0.3", - "proxy-from-env": "1.1.0", - "rimraf": "3.0.2", - "tar-fs": "2.1.1", - "unbzip2-stream": "1.4.3", - "ws": "8.5.0" - }, - "dependencies": { - "ws": { - "version": "8.5.0", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.5.0.tgz", - "integrity": "sha512-BWX0SWVgLPzYwF8lTzEy1egjhS4S4OEAHfsO8o65WOVsrnSRGaSiUaa9e0ggGlkMTtBlmOpEXiie9RUcBO86qg==" - } - } - }, - "qr-image": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/qr-image/-/qr-image-3.2.0.tgz", - "integrity": "sha1-n6gpW+rlDEoUnPn5CaHbRkqGcug=" - }, - "qrcode-terminal": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/qrcode-terminal/-/qrcode-terminal-0.12.0.tgz", - "integrity": "sha512-EXtzRZmC+YGmGlDFbXKxQiMZNwCLEO6BANKXG4iCtSIM0yqc/pappSx3RIKr4r0uh5JsBckOXeKrB3Iz7mdQpQ==" - }, - "qs": { - "version": "6.9.7", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.9.7.tgz", - "integrity": "sha512-IhMFgUmuNpyRfxA90umL7ByLlgRXu6tIfKPpF5TmcfRLlLCckfP/g3IQmju6jjpu+Hh8rA+2p6A27ZSPOOHdKw==" - }, - "range-parser": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", - "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" - }, - "raw-body": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.3.tgz", - "integrity": "sha512-UlTNLIcu0uzb4D2f4WltY6cVjLi+/jEN4lgEUj3E04tpMDpUlkBo/eSn6zou9hum2VMNpCCUone0O0WeJim07g==", - "requires": { - "bytes": "3.1.2", - "http-errors": "1.8.1", - "iconv-lite": "0.4.24", - "unpipe": "1.0.0" - } - }, - "read": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/read/-/read-1.0.7.tgz", - "integrity": "sha1-s9oZvQUkMal2cdRKQmNK33ELQMQ=", - "dev": true, - "requires": { - "mute-stream": "~0.0.4" - } - }, - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - }, - "readable-web-to-node-stream": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/readable-web-to-node-stream/-/readable-web-to-node-stream-3.0.2.tgz", - "integrity": "sha512-ePeK6cc1EcKLEhJFt/AebMCLL+GgSKhuygrZ/GLaKZYEecIgIECf4UaUuaByiGtzckwR4ain9VzUh95T1exYGw==", - "requires": { - "readable-stream": "^3.6.0" - } - }, - "readdir-glob": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/readdir-glob/-/readdir-glob-1.1.1.tgz", - "integrity": "sha512-91/k1EzZwDx6HbERR+zucygRFfiPl2zkIYZtv3Jjr6Mn7SkKcVct8aVO+sSRiGMc6fLf72du3d92/uY63YPdEA==", - "requires": { - "minimatch": "^3.0.4" - } - }, - "readdirp": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", - "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", - "dev": true, - "requires": { - "picomatch": "^2.2.1" - } - }, - "require-directory": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", - "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=" - }, - "require-in-the-middle": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/require-in-the-middle/-/require-in-the-middle-5.1.0.tgz", - "integrity": "sha512-M2rLKVupQfJ5lf9OvqFGIT+9iVLnTmjgbOmpil12hiSQNn5zJTKGPoIisETNjfK+09vP3rpm1zJajmErpr2sEQ==", - "dev": true, - "requires": { - "debug": "^4.1.1", - "module-details-from-path": "^1.0.3", - "resolve": "^1.12.0" - } - }, - "resolve": { - "version": "1.22.0", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.0.tgz", - "integrity": "sha512-Hhtrw0nLeSrFQ7phPp4OOcVjLPIeMnRlr5mcnVuMe7M/7eBn98A3hmFRLoFo3DLZkivSYwhRUJTyPyWAk56WLw==", - "dev": true, - "requires": { - "is-core-module": "^2.8.1", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - } - }, - "retry-request": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/retry-request/-/retry-request-4.2.2.tgz", - "integrity": "sha512-xA93uxUD/rogV7BV59agW/JHPGXeREMWiZc9jhcwY4YdZ7QOtC7qbomYg0n4wyk2lJhggjvKvhNX8wln/Aldhg==", - "requires": { - "debug": "^4.1.1", - "extend": "^3.0.2" - } - }, - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "requires": { - "glob": "^7.1.3" - } - }, - "run-series": { - "version": "1.1.9", - "resolved": "https://registry.npmjs.org/run-series/-/run-series-1.1.9.tgz", - "integrity": "sha512-Arc4hUN896vjkqCYrUXquBFtRZdv1PfLbTYP71efP6butxyQ0kWpiNJyAgsxscmQg1cqvHY32/UCBzXedTpU2g==", - "dev": true - }, - "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==" - }, - "safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" - }, - "sax": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", - "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==", - "dev": true - }, - "saxes": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", - "integrity": "sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw==", - "requires": { - "xmlchars": "^2.2.0" - } - }, - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", - "dev": true, - "requires": { - "lru-cache": "^6.0.0" - } - }, - "send": { - "version": "0.17.2", - "resolved": "https://registry.npmjs.org/send/-/send-0.17.2.tgz", - "integrity": "sha512-UJYB6wFSJE3G00nEivR5rgWp8c2xXvJ3OPWPhmuteU0IKj8nKbG3DrjiOmLwpnHGYWAVwA69zmTm++YG0Hmwww==", - "requires": { - "debug": "2.6.9", - "depd": "~1.1.2", - "destroy": "~1.0.4", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "etag": "~1.8.1", - "fresh": "0.5.2", - "http-errors": "1.8.1", - "mime": "1.6.0", - "ms": "2.1.3", - "on-finished": "~2.3.0", - "range-parser": "~1.2.1", - "statuses": "~1.5.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - }, - "dependencies": { - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" - } - } - }, - "serve-static": { - "version": "1.14.2", - "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.2.tgz", - "integrity": "sha512-+TMNA9AFxUEGuC0z2mevogSnn9MXKb4fa7ngeRMJaaGv8vTwnIEkKi+QGvPt33HSnf8pRS+WGM0EbMtCJLKMBQ==", - "requires": { - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "parseurl": "~1.3.3", - "send": "0.17.2" - } - }, - "set-immediate-shim": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz", - "integrity": "sha1-SysbJ+uAip+NzEgaWOXlb1mfP2E=" - }, - "setimmediate": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", - "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=" - }, - "setprototypeof": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", - "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" - }, - "shimmer": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/shimmer/-/shimmer-1.2.1.tgz", - "integrity": "sha512-sQTKC1Re/rM6XyFM6fIAGHRPVGvyXfgzIDvzoq608vM+jeyVD0Tu1E6Np0Kc2zAIFWIj963V2800iF/9LPieQw==", - "dev": true - }, - "signal-exit": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", - "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", - "dev": true - }, - "smart-buffer": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", - "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", - "dev": true - }, - "socket.io": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.4.1.tgz", - "integrity": "sha512-s04vrBswdQBUmuWJuuNTmXUVJhP0cVky8bBDhdkf8y0Ptsu7fKU2LuLbts9g+pdmAdyMMn8F/9Mf1/wbtUN0fg==", - "requires": { - "accepts": "~1.3.4", - "base64id": "~2.0.0", - "debug": "~4.3.2", - "engine.io": "~6.1.0", - "socket.io-adapter": "~2.3.3", - "socket.io-parser": "~4.0.4" - } - }, - "socket.io-adapter": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.3.3.tgz", - "integrity": "sha512-Qd/iwn3VskrpNO60BeRyCyr8ZWw9CPZyitW4AQwmRZ8zCiyDiL+znRnWX6tDHXnWn1sJrM1+b6Mn6wEDJJ4aYQ==" - }, - "socket.io-parser": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.0.4.tgz", - "integrity": "sha512-t+b0SS+IxG7Rxzda2EVvyBZbvFPBCjJoyHuE0P//7OAsN23GItzDRdWa6ALxZI/8R5ygK7jAR6t028/z+7295g==", - "requires": { - "@types/component-emitter": "^1.2.10", - "component-emitter": "~1.3.0", - "debug": "~4.3.1" - } - }, - "socks": { - "version": "2.6.2", - "resolved": "https://registry.npmjs.org/socks/-/socks-2.6.2.tgz", - "integrity": "sha512-zDZhHhZRY9PxRruRMR7kMhnf3I8hDs4S3f9RecfnGxvcBHQcKcIH/oUcEWffsfl1XxdYlA7nnlGbbTvPz9D8gA==", - "dev": true, - "requires": { - "ip": "^1.1.5", - "smart-buffer": "^4.2.0" - } - }, - "socks-proxy-agent": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-5.0.1.tgz", - "integrity": "sha512-vZdmnjb9a2Tz6WEQVIurybSwElwPxMZaIc7PzqbJTrezcKNznv6giT7J7tZDZ1BojVaa1jvO/UiUdhDVB0ACoQ==", - "dev": true, - "requires": { - "agent-base": "^6.0.2", - "debug": "4", - "socks": "^2.3.3" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "source-map-support": { - "version": "0.5.19", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", - "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", - "dev": true, - "requires": { - "buffer-from": "^1.0.0", - "source-map": "^0.6.0" - } - }, - "sprintf-js": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", - "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==", - "dev": true - }, - "sqlstring": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/sqlstring/-/sqlstring-2.3.1.tgz", - "integrity": "sha1-R1OT/56RR5rqYtyvDKPRSYOn+0A=" - }, - "statuses": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", - "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" - }, - "stormdb": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/stormdb/-/stormdb-0.5.2.tgz", - "integrity": "sha512-qvjjceJnt95CkIr8e/mhSqFxw/b787vRKDysQ7P6hmXyavROiw6X3A6li4AoF2QXL/xrwA0hdajCtuiu2WGe6g==" - }, - "stream-shift": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/stream-shift/-/stream-shift-1.0.1.tgz", - "integrity": "sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ==" - }, - "string-width": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", - "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.1" - } - }, - "string_decoder": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", - "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", - "requires": { - "safe-buffer": "~5.2.0" - } - }, - "strip-ansi": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", - "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "requires": { - "ansi-regex": "^5.0.1" - } - }, - "strtok3": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/strtok3/-/strtok3-6.3.0.tgz", - "integrity": "sha512-fZtbhtvI9I48xDSywd/somNqgUHl2L2cstmXCCif0itOf96jeW18MBSyrLuNicYQVkvpOxkZtkzujiTJ9LW5Jw==", - "requires": { - "@tokenizer/token": "^0.3.0", - "peek-readable": "^4.1.0" - } - }, - "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, - "requires": { - "has-flag": "^4.0.0" - } - }, - "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 - }, - "systeminformation": { - "version": "5.11.5", - "resolved": "https://registry.npmjs.org/systeminformation/-/systeminformation-5.11.5.tgz", - "integrity": "sha512-AtJreEPwOAnMstMRtVbQwq8V+dOgOJa8fqzjXjrb46JnVIqrRHRP6+L8VU4WvDbPgdRHA7Nr0l6jH42dmGoACQ==", - "dev": true, - "optional": true - }, - "tar-fs": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz", - "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==", - "requires": { - "chownr": "^1.1.1", - "mkdirp-classic": "^0.5.2", - "pump": "^3.0.0", - "tar-stream": "^2.1.4" - } - }, - "tar-stream": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", - "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", - "requires": { - "bl": "^4.0.3", - "end-of-stream": "^1.4.1", - "fs-constants": "^1.0.0", - "inherits": "^2.0.3", - "readable-stream": "^3.1.1" - } - }, - "through": { - "version": "2.3.8", - "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", - "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=" - }, - "tmp": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz", - "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==", - "requires": { - "rimraf": "^3.0.0" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - }, - "toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==" - }, - "token-types": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/token-types/-/token-types-4.1.1.tgz", - "integrity": "sha512-hD+QyuUAyI2spzsI0B7gf/jJ2ggR4RjkAo37j3StuePhApJUwcWDjnHDOFdIWYSwNR28H14hpwm4EI+V1Ted1w==", - "requires": { - "@tokenizer/token": "^0.3.0", - "ieee754": "^1.2.1" - } - }, - "tr46": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", - "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=" - }, - "traverse": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/traverse/-/traverse-0.3.9.tgz", - "integrity": "sha1-cXuPIgzAu3tE5AUUwisui7xw2Lk=" - }, - "tslib": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.3.1.tgz", - "integrity": "sha512-77EbyPPpMz+FRFRuAFlWMtmgUWGe9UOG2Z25NqCwiIjRhOf5iKGuzSe5P2w1laq+FkRy4p+PCuVkJSGkzTEKVw==", - "dev": true - }, - "tv4": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/tv4/-/tv4-1.3.0.tgz", - "integrity": "sha1-0CDIRvrdUMhVq7JeuuzGj8EPeWM=", - "dev": true - }, - "tx2": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/tx2/-/tx2-1.0.5.tgz", - "integrity": "sha512-sJ24w0y03Md/bxzK4FU8J8JveYYUbSs2FViLJ2D/8bytSiyPRbuE3DyL/9UKYXTZlV3yXq0L8GLlhobTnekCVg==", - "dev": true, - "optional": true, - "requires": { - "json-stringify-safe": "^5.0.1" - } - }, - "type-check": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", - "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", - "dev": true, - "requires": { - "prelude-ls": "~1.1.2" - } - }, - "type-is": { - "version": "1.6.18", - "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", - "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", - "requires": { - "media-typer": "0.3.0", - "mime-types": "~2.1.24" - } - }, - "unbzip2-stream": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/unbzip2-stream/-/unbzip2-stream-1.4.3.tgz", - "integrity": "sha512-mlExGW4w71ebDJviH16lQLtZS32VKqsSfk80GCfUlwT/4/hNRFsoscrF/c++9xinkMzECL1uL9DDwXqFWkruPg==", - "requires": { - "buffer": "^5.2.1", - "through": "^2.3.8" - } - }, - "universalify": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", - "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", - "dev": true - }, - "unpipe": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", - "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" - }, - "unzipper": { - "version": "0.10.11", - "resolved": "https://registry.npmjs.org/unzipper/-/unzipper-0.10.11.tgz", - "integrity": "sha512-+BrAq2oFqWod5IESRjL3S8baohbevGcVA+teAIOYWM3pDVdseogqbzhhvvmiyQrUNKFUnDMtELW3X8ykbyDCJw==", - "requires": { - "big-integer": "^1.6.17", - "binary": "~0.3.0", - "bluebird": "~3.4.1", - "buffer-indexof-polyfill": "~1.0.0", - "duplexer2": "~0.1.4", - "fstream": "^1.0.12", - "graceful-fs": "^4.2.2", - "listenercount": "~1.0.1", - "readable-stream": "~2.3.6", - "setimmediate": "~1.0.4" - }, - "dependencies": { - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "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" - } - }, - "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==" - }, - "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==", - "requires": { - "safe-buffer": "~5.1.0" - } - } - } - }, - "util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" - }, - "utils-merge": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", - "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" - }, - "uuid": { - "version": "8.3.2", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", - "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==" - }, - "vary": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", - "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" - }, - "vizion": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/vizion/-/vizion-2.2.1.tgz", - "integrity": "sha512-sfAcO2yeSU0CSPFI/DmZp3FsFE9T+8913nv1xWBOyzODv13fwkn6Vl7HqxGpkr9F608M+8SuFId3s+BlZqfXww==", - "dev": true, - "requires": { - "async": "^2.6.3", - "git-node-fs": "^1.0.0", - "ini": "^1.3.5", - "js-git": "^0.7.8" - }, - "dependencies": { - "async": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", - "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", - "dev": true, - "requires": { - "lodash": "^4.17.14" - } - } - } - }, - "vm2": { - "version": "3.9.9", - "resolved": "https://registry.npmjs.org/vm2/-/vm2-3.9.9.tgz", - "integrity": "sha512-xwTm7NLh/uOjARRBs8/95H0e8fT3Ukw5D/JJWhxMbhKzNh1Nu981jQKvkep9iKYNxzlVrdzD0mlBGkDKZWprlw==", - "dev": true, - "requires": { - "acorn": "^8.7.0", - "acorn-walk": "^8.2.0" - } - }, - "webidl-conversions": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", - "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=" - }, - "whatsapp-web.js": { - "version": "1.16.4", - "resolved": "https://registry.npmjs.org/whatsapp-web.js/-/whatsapp-web.js-1.16.4.tgz", - "integrity": "sha512-cN4ncwqnR3aMmltXtPRwQTfjwQGNmSFDnfaQ4pm3EpqbHWAzvxBlxaWFCWcZQnbUmPRvnVCEky1C9LvV+ePq3Q==", - "requires": { - "@pedroslopez/moduleraid": "^5.0.2", - "fluent-ffmpeg": "^2.1.2", - "jsqr": "^1.3.1", - "mime": "^3.0.0", - "node-fetch": "^2.6.5", - "node-webpmux": "^3.1.0", - "puppeteer": "^13.0.0" - }, - "dependencies": { - "mime": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/mime/-/mime-3.0.0.tgz", - "integrity": "sha512-jSCU7/VB1loIWBZe14aEYHU/+1UMEHoaO7qxCOVJOw9GgH72VAWppxNcjU+x9a2k3GSIBXNKxXQFqRvvZ7vr3A==" - } - } - }, - "whatwg-url": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", - "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", - "requires": { - "tr46": "~0.0.3", - "webidl-conversions": "^3.0.0" - } - }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", - "requires": { - "isexe": "^2.0.0" - } - }, - "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 - }, - "wrap-ansi": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - } - }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" - }, - "ws": { - "version": "8.2.3", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.2.3.tgz", - "integrity": "sha512-wBuoj1BDpC6ZQ1B7DWQBYVLphPWkm8i9Y0/3YdHjHKHiohOJ1ws+3OccDWtH+PoC9DZD5WOTrJvNbWvjS6JWaA==" - }, - "xmlchars": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", - "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==" - }, - "xregexp": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/xregexp/-/xregexp-2.0.0.tgz", - "integrity": "sha1-UqY+VsoLhKfzpfPWGHLxJq16WUM=", - "dev": true - }, - "y18n": { - "version": "5.0.8", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", - "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==" - }, - "yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" - }, - "yamljs": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/yamljs/-/yamljs-0.3.0.tgz", - "integrity": "sha512-C/FsVVhht4iPQYXOInoxUM/1ELSf9EsgKH34FofQOp6hwCPrW4vG4w5++TED3xRUo8gD7l0P1J1dLlDYzODsTQ==", - "dev": true, - "requires": { - "argparse": "^1.0.7", - "glob": "^7.0.5" - } - }, - "yargs": { - "version": "16.2.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", - "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", - "requires": { - "cliui": "^7.0.2", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.0", - "y18n": "^5.0.5", - "yargs-parser": "^20.2.2" - } - }, - "yargs-parser": { - "version": "20.2.9", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", - "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==" - }, - "yauzl": { - "version": "2.10.0", - "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.10.0.tgz", - "integrity": "sha1-x+sXyT4RLLEIb6bY5R+wZnt5pfk=", - "requires": { - "buffer-crc32": "~0.2.3", - "fd-slicer": "~1.1.0" - } - }, - "zip-stream": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/zip-stream/-/zip-stream-4.1.0.tgz", - "integrity": "sha512-zshzwQW7gG7hjpBlgeQP9RuyPGNxvJdzR8SUM3QhxCnLjWN2E7j3dOvpeDcQoETfHx0urRS7EtmVToql7YpU4A==", - "requires": { - "archiver-utils": "^2.1.0", - "compress-commons": "^4.1.0", - "readable-stream": "^3.6.0" - } - } - } -} diff --git a/package.json b/package.json index 554dd25..dbc3846 100644 --- a/package.json +++ b/package.json @@ -1,37 +1,92 @@ { - "name": "test-ws-bot", - "version": "1.0.0", - "description": "", - "main": "app.js", - "scripts": { - "start": "node ./app.js", - "test": "echo \"Error: no test specified\" && exit 1" - }, - "keywords": [], - "author": "", - "license": "ISC", - "dependencies": { - "@google-cloud/dialogflow": "^4.6.0", - "cors": "^2.8.5", - "dotenv": "^11.0.0", - "exceljs": "^4.3.0", - "express": "^4.17.2", - "file-type": "^16.5.3", - "mime-db": "^1.51.0", - "moment": "^2.29.1", - "mysql": "^2.18.1", - "nanoid": "^3.1.32", - "qr-image": "^3.2.0", - "qrcode-terminal": "^0.12.0", - "socket.io": "^4.4.1", - "stormdb": "^0.5.2", - "whatsapp-web.js": "^1.16.4" - }, - "devDependencies": { - "pm2": "^5.1.2", - "prettier": "2.5.1" - }, - "engines": { - "node": "14.x" - } + "name": "@bot-whatsapp/root", + "version": "0.1.1", + "description": "Bot de wahtsapp open source para MVP o pequeños negocios", + "main": "app.js", + "private": true, + "scripts": { + "commit": "git-cz", + "cli:rollup": "rollup --config ./packages/cli/rollup-cli.config.js ", + "bot:rollup": "rollup --config ./packages/bot/rollup-bot.config.js", + "provider:rollup": "rollup --config ./packages/provider/rollup-provider.config.js ", + "database:rollup": "rollup --config ./packages/database/rollup-database.config.js", + "format:check": "prettier --check ./packages", + "format:write": "prettier --write ./packages", + "lint:check": "eslint ./packages", + "lint:fix": "eslint --fix ./packages", + "build": "yarn run cli:rollup && yarn run bot:rollup && yarn run provider:rollup && yarn run database:rollup", + "link.dist": "cd packages/bot && npm link && cd ../provider && npm link && cd ../cli && npm link", + "test.unit": "node ./node_modules/uvu/bin.js packages test", + "test.coverage": "node ./node_modules/c8/bin/c8.js npm run test.unit", + "test": "npm run test.coverage", + "cli": "node ./packages/cli/bin/cli.js", + "dev:debug": "node --inspect ./example-app/app.js", + "dev": "node ./example-app/app.js", + "prepare": "npx husky install", + "preinstall": "npx only-allow yarn", + "release": "standard-version" + }, + "workspaces": [ + "packages/bot", + "packages/cli", + "packages/database", + "packages/provider", + "packages/docs" + ], + "keywords": [ + "whatsapp", + "bot-whatsapp", + "node-bot-whatsapp" + ], + "contributors": [ + { + "email": "leifer33@gmail.com", + "name": "Leifer Mendez", + "url": "https://leifermendez.github.io" + }, + { + "name": "aurik3", + "email": "aurik3@aurik3.com", + "url": "https://github.com/aurik3" + } + ], + "repository": "https://github.com/leifermendez/bot-whatsapp", + "license": "ISC", + "devDependencies": { + "@commitlint/cli": "^17.3.0", + "@commitlint/config-conventional": "^17.3.0", + "@rollup/plugin-commonjs": "^23.0.2", + "@rollup/plugin-json": "^5.0.1", + "@rollup/plugin-node-resolve": "^15.0.1", + "@rollup/plugin-replace": "^5.0.1", + "c8": "^7.12.0", + "commitizen": "^4.2.5", + "cross-env": "^7.0.3", + "cz-conventional-changelog": "^3.3.0", + "eslint": "^8.26.0", + "eslint-config-prettier": "^8.5.0", + "git-cz": "^4.9.0", + "husky": "^8.0.2", + "only-allow": "^1.1.1", + "prettier": "^2.7.1", + "prompts": "^2.4.2", + "rimraf": "^3.0.2", + "rollup": "^3.2.3", + "rollup-plugin-cleanup": "^3.2.1", + "rollup-plugin-copy": "^3.4.0", + "standard-version": "^9.5.0", + "uvu": "^0.5.6" + }, + "packageManager": "yarn@3.3.0", + "engines": { + "node": ">=16", + "npm": "please-use-yarn", + "yarn": ">=1" + }, + "author": "Leifer Mendez ", + "config": { + "commitizen": { + "path": "git-cz" + } + } } diff --git a/packages/bot/USES_CASES.md b/packages/bot/USES_CASES.md new file mode 100644 index 0000000..f3d8729 --- /dev/null +++ b/packages/bot/USES_CASES.md @@ -0,0 +1,110 @@ +# @bot-whatsapp/io + +### Caso de uso + +> Una persona escribe `hola` + +**addKeyword** recibe `string | string[]` + +> `sensitive` false _default_ + +- [x] addKeyword +- [x] addAnswer +- [x] addKeyword: Opciones +- [x] addAnswer: Opciones, media, buttons +- [x] Retornar JSON (options) +- [ ] Recibir JSON + +```js +// bootstrap.js Como iniciar el provider +const { inout, provider, database } = require('@bot-whatsapp') + +/** + * async whatsapp-web, twilio, meta + * */ + +const bootstrap = async () => { + console.log(`Iniciando....`) + const client = await provider.start() + /** + * - QR + * - Endpoint + * - Check Token Meta, Twilio + * - Return events? on message + * */ + console.log(`Fin...`) + // Esto es opcional ? no deberia ser necesario + client.on('message', ({number, body,...}) => { + // Incoming message + }) +} + +``` + +```js +// flow.js Como agregar keywords y respuestas +const { inout, provider, database } = require('@bot-whatsapp') + +await inout + .addKeyword('hola') + .addAnswer('Bienvenido a tu tienda 🥲') + .addAnswer('escribe *catalogo* o *ofertas*') + +await inout + .addKeyword(['catalogo', 'ofertas']) + .addAnswer('Este es nuestro CATALOGO mas reciente!', { + buttons: [{ body: 'Xiaomi' }, { body: 'Samsung' }], + }) + +await inout + .addKeyword('Xiaomi') + .addAnswer('Estos son nuestro productos XIAOMI ....', { + media: 'https://....', + }) + .addAnswer('Si quieres mas info escrbie *info*') + +await inout + .addKeyword('chao!') + .addAnswer('bye!') + .addAnswer('Recuerda que tengo esta promo', { + media: 'https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif', + }) + +await inout + .addKeyword('Modelo C', { sensitive: false }) + .addAnswer('100USD', { media: 'http//:...' }) + +await inout + .addKeyword('hola!', { sensitive: false }) + .addAnswer('Bievenido Escribe *productos*') + +await inout + .addKeyword('productos', { sensitive: false }) + .addAnswer('Esto son los mas vendidos') + .addAnswer('*PC1* Precio 10USD', { media: 'https://....' }) + .addAnswer('*PC2* Precio 10USD', { media: 'https://....' }) + +await inout + .addKeyword('PC1', { sensitive: false }) + .addAnswer('Bievenido Escribe *productos*') + +const answerOne = await inout.addAnswer({ + message: 'Como estas!', + media: 'https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif', +}) + +const otherAnswer = await inout.addAnswer('Aprovecho para decirte!') + +answerOne.push(otherAnswer) + +inout.addKeywords(['hola', 'hi', 'ola']) +``` + +**Comunidad** + +> Forma parte de este proyecto. + +- [Discord](https://link.codigoencasa.com/DISCORD) +- [Twitter](https://twitter.com/leifermendez) +- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR) +- [Telegram](https://t.me/leifermendez) diff --git a/packages/bot/core/core.class.js b/packages/bot/core/core.class.js new file mode 100644 index 0000000..106b9c9 --- /dev/null +++ b/packages/bot/core/core.class.js @@ -0,0 +1,128 @@ +const { toCtx } = require('../io/methods') +const { printer } = require('../utils/interactive') + +/** + * [ ] Escuchar eventos del provider asegurarte que los provider emitan eventos + * [ ] Guardar historial en db + * [ ] Buscar mensaje en flow + * + */ +class CoreClass { + flowClass + databaseClass + providerClass + constructor(_flow, _database, _provider) { + this.flowClass = _flow + this.databaseClass = _database + this.providerClass = _provider + + for (const { event, func } of this.listenerBusEvents()) { + this.providerClass.on(event, func) + } + } + + listenerBusEvents = () => [ + { + event: 'require_action', + func: ({ instructions, title = '⚡⚡ ACCION REQUERIDA ⚡⚡' }) => + printer(instructions, title), + }, + { + event: 'ready', + func: () => printer('Provider conectado y listo'), + }, + { + event: 'auth_failure', + func: ({ instructions }) => + printer(instructions, '⚡⚡ ERROR AUTH ⚡⚡'), + }, + + { + event: 'message', + func: (msg) => this.handleMsg(msg), + }, + ] + + /** + * @private + * @param {*} ctxMessage + */ + handleMsg = async (messageInComming) => { + const { body, from } = messageInComming + let msgToSend = [] + + //Consultamos mensaje previo en DB + const prevMsg = await this.databaseClass.getPrevByNumber(from) + //Consultamos for refSerializada en el flow actual + const refToContinue = this.flowClass.findBySerialize( + prevMsg?.refSerialize + ) + + if (prevMsg?.ref) { + const ctxByNumber = toCtx({ + body, + from, + prevRef: prevMsg.refSerialize, + }) + this.databaseClass.save(ctxByNumber) + } + + //Si se tiene un callback se ejecuta + if (refToContinue && prevMsg?.options?.callback) { + const indexFlow = this.flowClass.findIndexByRef(refToContinue?.ref) + this.flowClass.allCallbacks[indexFlow].callback(messageInComming) + } + + //Si se tiene anidaciones de flows, si tienes anidados obligatoriamente capture:true + if (prevMsg?.options?.nested?.length) { + const nestedRef = prevMsg.options.nested + const flowStandalone = nestedRef.map((f) => ({ + ...nestedRef.find((r) => r.refSerialize === f.refSerialize), + })) + + msgToSend = this.flowClass.find(body, false, flowStandalone) || [] + this.sendFlow(msgToSend, from) + return + } + + //Consultamos si se espera respuesta por parte de cliente "Ejemplo: Dime tu nombre" + if (!prevMsg?.options?.nested?.length && prevMsg?.options?.capture) { + msgToSend = this.flowClass.find(refToContinue?.ref, true) || [] + } else { + msgToSend = this.flowClass.find(body) || [] + } + + this.sendFlow(msgToSend, from) + } + + sendProviderAndSave = (numberOrId, ctxMessage) => { + const { answer } = ctxMessage + return Promise.all([ + this.providerClass.sendMessage(numberOrId, answer), + this.databaseClass.save({ ...ctxMessage, from: numberOrId }), + ]) + } + + sendFlow = (messageToSend, numberOrId) => { + const queue = [] + for (const ctxMessage of messageToSend) { + queue.push(this.sendProviderAndSave(numberOrId, ctxMessage)) + } + return Promise.all(queue) + } + + /** + * @private + * @param {*} message + * @param {*} ref + */ + continue = (message, ref = false) => { + const responde = this.flowClass.find(message, ref) + if (responde) { + this.providerClass.sendMessage(responde.answer) + this.databaseClass.saveLog(responde.answer) + this.continue(null, responde.ref) + } + } +} +module.exports = CoreClass diff --git a/packages/bot/index.js b/packages/bot/index.js new file mode 100644 index 0000000..e5b6e1c --- /dev/null +++ b/packages/bot/index.js @@ -0,0 +1,44 @@ +const CoreClass = require('./core/core.class') +const ProviderClass = require('./provider/provider.class') +const FlowClass = require('./io/flow.class') +const { addKeyword, addAnswer, toSerialize } = require('./io/methods') + +/** + * Crear instancia de clase Bot + * @param {*} args + * @returns + */ +const createBot = async ({ flow, database, provider }) => + new CoreClass(flow, database, provider) + +/** + * Crear instancia de clase Io (Flow) + * @param {*} args + * @returns + */ +const createFlow = (args) => { + return new FlowClass(args) +} + +/** + * Crear instancia de clase Provider + * @param {*} args + * @returns + */ +const createProvider = (providerClass = class {}) => { + const providerInstance = new providerClass() + if (!providerClass.prototype instanceof ProviderClass) + throw new Error('El provider no implementa ProviderClass') + return providerInstance +} + +module.exports = { + createBot, + createFlow, + createProvider, + addKeyword, + addAnswer, + toSerialize, + ProviderClass, + CoreClass, +} diff --git a/packages/bot/io/flow.class.js b/packages/bot/io/flow.class.js new file mode 100644 index 0000000..5ed8eb4 --- /dev/null +++ b/packages/bot/io/flow.class.js @@ -0,0 +1,71 @@ +const { toSerialize } = require('./methods/toSerialize') + +class FlowClass { + allCallbacks = [] + flowSerialize = [] + flowRaw = [] + constructor(_flow) { + if (!Array.isArray(_flow)) throw new Error('Esto debe ser un ARRAY') + this.flowRaw = _flow + + this.allCallbacks = _flow + .map((cbIn) => cbIn.ctx.callbacks) + .flat(2) + .map((c, i) => ({ callback: c?.callback, index: i })) + + const mergeToJsonSerialize = Object.keys(_flow) + .map((indexObjectFlow) => _flow[indexObjectFlow].toJson()) + .flat(2) + + this.flowSerialize = toSerialize(mergeToJsonSerialize) + } + + find = (keyOrWord, symbol = false, overFlow = null) => { + let capture = false + let messages = [] + let refSymbol = null + overFlow = overFlow ?? this.flowSerialize + + const mapSensitiveString = (str, flag = false) => { + if (!flag && Array.isArray(str)) { + return str.map((c) => c.toLowerCase()) + } + + if (!flag && typeof str === 'string') { + return str.toLowerCase() + } + + return str + } + + const findIn = (keyOrWord, symbol = false, flow = overFlow) => { + const sensitive = refSymbol?.options?.sensitive || false + capture = refSymbol?.options?.capture || false + + keyOrWord = mapSensitiveString(keyOrWord, sensitive) + + if (capture) return messages + + if (symbol) { + refSymbol = flow.find((c) => c.keyword === keyOrWord) + if (refSymbol?.answer) messages.push(refSymbol) + if (refSymbol?.ref) findIn(refSymbol.ref, true) + } else { + refSymbol = flow.find((c) => + mapSensitiveString(c.keyword, sensitive).includes(keyOrWord) + ) + if (refSymbol?.ref) findIn(refSymbol.ref, true) + return messages + } + } + findIn(keyOrWord, symbol) + return messages + } + + findBySerialize = (refSerialize) => + this.flowSerialize.find((r) => r.refSerialize === refSerialize) + + findIndexByRef = (ref) => this.flowSerialize.findIndex((r) => r.ref === ref) +} + +module.exports = FlowClass diff --git a/packages/bot/io/methods/addAnswer.js b/packages/bot/io/methods/addAnswer.js new file mode 100644 index 0000000..2300ff4 --- /dev/null +++ b/packages/bot/io/methods/addAnswer.js @@ -0,0 +1,91 @@ +const { generateRef } = require('../../utils/hash') +const { toJson } = require('./toJson') +const { toSerialize } = require('./toSerialize') +/** + * + * @param answer string + * @param options {media:string, buttons:[], capture:true default false} + * @returns + */ +const addAnswer = + (inCtx) => + (answer, options, cb = null, nested = []) => { + /** + * Todas las opciones referentes a el mensaje en concreto options:{} + * @returns + */ + const getAnswerOptions = () => ({ + media: + typeof options?.media === 'string' ? `${options?.media}` : null, + buttons: Array.isArray(options?.buttons) ? options.buttons : [], + capture: + typeof options?.capture === 'boolean' + ? options?.capture + : false, + child: + typeof options?.child === 'string' ? `${options?.child}` : null, + }) + + const getNested = () => ({ + nested: Array.isArray(nested) ? nested : [], + }) + + const callback = + typeof cb === 'function' + ? cb + : () => console.log('Callback no definida') + + const lastCtx = inCtx.hasOwnProperty('ctx') ? inCtx.ctx : inCtx + + /** + * Esta funcion se encarga de mapear y transformar todo antes + * de retornar + * @returns + */ + const ctxAnswer = () => { + const ref = `ans_${generateRef()}` + + const options = { + ...getAnswerOptions(), + ...getNested(), + keyword: {}, + callback: !!cb, + } + + const json = [].concat(inCtx.json).concat([ + { + ref, + keyword: lastCtx.ref, + answer, + options, + }, + ]) + + const callbacks = [].concat(inCtx.callbacks).concat([ + { + ref: lastCtx.ref, + callback, + }, + ]) + + return { + ...lastCtx, + ref, + answer, + json, + options, + callbacks, + } + } + + const ctx = ctxAnswer() + + return { + ctx, + ref: ctx.ref, + addAnswer: addAnswer(ctx), + toJson: toJson(ctx), + } + } + +module.exports = { addAnswer } diff --git a/packages/bot/io/methods/addKeyword.js b/packages/bot/io/methods/addKeyword.js new file mode 100644 index 0000000..e17f9be --- /dev/null +++ b/packages/bot/io/methods/addKeyword.js @@ -0,0 +1,49 @@ +const { generateRef } = require('../../utils/hash') +const { addAnswer } = require('./addAnswer') +const { toJson } = require('./toJson') + +/** + * + * @param {*} message `string | string[]` + * @param {*} options {sensitive:boolean} default false + */ +const addKeyword = (keyword, options) => { + const parseOptions = () => { + const defaultProperties = { + sensitive: + typeof options?.sensitive === 'boolean' + ? options?.sensitive + : false, + } + + return defaultProperties + } + + const ctxAddKeyword = () => { + const ref = `key_${generateRef()}` + const options = parseOptions() + const json = [ + { + ref, + keyword, + options, + }, + ] + /** + * Se guarda en db + */ + + return { ref, keyword, options, json } + } + + const ctx = ctxAddKeyword() + + return { + ctx, + ref: ctx.ref, + addAnswer: addAnswer(ctx), + toJson: toJson(ctx), + } +} + +module.exports = { addKeyword } diff --git a/packages/bot/io/methods/index.js b/packages/bot/io/methods/index.js new file mode 100644 index 0000000..875a49c --- /dev/null +++ b/packages/bot/io/methods/index.js @@ -0,0 +1,7 @@ +const { addAnswer } = require('./addAnswer') +const { addKeyword } = require('./addKeyword') +const { toSerialize } = require('./toSerialize') +const { toCtx } = require('./toCtx') +const { toJson } = require('./toJson') + +module.exports = { addAnswer, addKeyword, toCtx, toJson, toSerialize } diff --git a/packages/bot/io/methods/toCtx.js b/packages/bot/io/methods/toCtx.js new file mode 100644 index 0000000..d29295e --- /dev/null +++ b/packages/bot/io/methods/toCtx.js @@ -0,0 +1,19 @@ +const { generateRef, generateRefSerialize } = require('../../utils/hash') +/** + * @deprecate + * @param answer string + * @param options {media:string, buttons:[], capture:true default false} + * @returns + */ +const toCtx = ({ body, from, prevRef, index }) => { + return { + ref: generateRef(), + keyword: prevRef, + answer: body, + options: {}, + from, + refSerialize: generateRefSerialize({ index, answer: body }), + } +} + +module.exports = { toCtx } diff --git a/packages/bot/io/methods/toJson.js b/packages/bot/io/methods/toJson.js new file mode 100644 index 0000000..121748e --- /dev/null +++ b/packages/bot/io/methods/toJson.js @@ -0,0 +1,6 @@ +const toJson = (inCtx) => () => { + const lastCtx = inCtx.hasOwnProperty('ctx') ? inCtx.ctx : inCtx + return lastCtx.json +} + +module.exports = { toJson } diff --git a/packages/bot/io/methods/toSerialize.js b/packages/bot/io/methods/toSerialize.js new file mode 100644 index 0000000..21ebf74 --- /dev/null +++ b/packages/bot/io/methods/toSerialize.js @@ -0,0 +1,23 @@ +const { generateRefSerialize } = require('../../utils/hash') + +/** + * Crear referencia serializada + * @param {*} flowJson + * @returns array[] + */ +const toSerialize = (flowJson) => { + if (!Array.isArray(flowJson)) throw new Error('Esto debe ser un ARRAY') + + const jsonToSerialize = flowJson.map((row, index) => ({ + ...row, + refSerialize: `${generateRefSerialize({ + index, + keyword: row.keyword, + answer: row.answer, + })}`, + })) + + return jsonToSerialize +} + +module.exports = { toSerialize } diff --git a/packages/bot/io/rollup-cli.config.js b/packages/bot/io/rollup-cli.config.js new file mode 100644 index 0000000..e58aa0c --- /dev/null +++ b/packages/bot/io/rollup-cli.config.js @@ -0,0 +1,14 @@ +const commonjs = require('@rollup/plugin-commonjs') +const { nodeResolve } = require('@rollup/plugin-node-resolve') +const { join } = require('path') + +const PATH = join(__dirname, 'lib', 'io', 'bundle.io.cjs') + +module.exports = { + input: 'index.js', + output: { + file: PATH, + format: 'cjs', + }, + plugins: [commonjs(), nodeResolve()], +} diff --git a/packages/bot/package.json b/packages/bot/package.json new file mode 100644 index 0000000..f6affee --- /dev/null +++ b/packages/bot/package.json @@ -0,0 +1,33 @@ +{ + "name": "@bot-whatsapp/bot", + "version": "0.0.1", + "description": "", + "main": "./lib/bundle.bot.cjs", + "private": true, + "scripts": { + "bot:rollup": "node ../../node_modules/.bin/rollup index.js --config ./rollup-cli.config.js", + "format:check": "prettier --check .", + "format:write": "prettier --write .", + "lint:check": "eslint .", + "lint:fix": "eslint --fix .", + "test.unit": "cross-env NODE_ENV=test node ../../node_modules/uvu/bin.js tests" + }, + "keywords": [], + "files": [ + "./lib/bundle.bot.cjs", + "./provider/*", + "./core/*", + "./io/*" + ], + "author": "", + "license": "ISC", + "devDependencies": { + "@bot-whatsapp/cli": "*", + "@bot-whatsapp/database": "*", + "@bot-whatsapp/provider": "*", + "kleur": "^4.1.5" + }, + "dependencies": { + "dotenv": "^16.0.3" + } +} diff --git a/packages/bot/provider/provider.class.js b/packages/bot/provider/provider.class.js new file mode 100644 index 0000000..212f042 --- /dev/null +++ b/packages/bot/provider/provider.class.js @@ -0,0 +1,29 @@ +const { EventEmitter } = require('node:events') +/** + * Esta clase debe siempre proporcionar los siguietes metodos + * sendMessage = Para enviar un mensaje + * + * @important + * Esta clase extiende de la clase del provider OJO + * Eventos + * - message + * - ready + * - error + * - require_action + */ + +const NODE_ENV = process.env.NODE_ENV || 'dev' +class ProviderClass extends EventEmitter { + /** + * events: message | auth | auth_error | ... + * + */ + + sendMessage = async (userId, message) => { + if (NODE_ENV !== 'production') + console.log('[sendMessage]', { userId, message }) + return message + } +} + +module.exports = ProviderClass diff --git a/packages/bot/rollup-bot.config.js b/packages/bot/rollup-bot.config.js new file mode 100644 index 0000000..0af60e4 --- /dev/null +++ b/packages/bot/rollup-bot.config.js @@ -0,0 +1,14 @@ +const commonjs = require('@rollup/plugin-commonjs') +const { nodeResolve } = require('@rollup/plugin-node-resolve') +const { join } = require('path') + +const PATH = join(__dirname, 'lib', 'bundle.bot.cjs') + +module.exports = { + input: join(__dirname, 'index.js'), + output: { + file: PATH, + format: 'cjs', + }, + plugins: [commonjs(), nodeResolve()], +} diff --git a/packages/bot/tests/bot.class.test.js b/packages/bot/tests/bot.class.test.js new file mode 100644 index 0000000..74fdb79 --- /dev/null +++ b/packages/bot/tests/bot.class.test.js @@ -0,0 +1,279 @@ +const { test } = require('uvu') +const assert = require('uvu/assert') +const FlowClass = require('../io/flow.class') +const MockProvider = require('../../../__mocks__/mock.provider') +const { + createBot, + CoreClass, + createFlow, + createProvider, + ProviderClass, +} = require('../index') + +class MockFlow { + allCallbacks = [{ callback: () => console.log('') }] + flowSerialize = [] + flowRaw = [] + find = (arg) => { + if (arg) { + return [{ answer: 'answer', ref: 'ref' }] + } else { + return null + } + } + findBySerialize = () => ({}) + findIndexByRef = () => 0 +} + +class MockDBA { + listHistory = [] + save = () => {} + getPrevByNumber = () => {} +} + +class MockDBB { + listHistory = [] + save = () => {} + getPrevByNumber = () => ({ + refSerialize: 'xxxxx', + ref: 'xxxx', + options: { callback: true }, + }) +} + +class MockDBC { + listHistory = [] + save = () => {} + getPrevByNumber = () => ({ + refSerialize: 'xxxxx', + ref: 'xxxx', + options: { callback: true, nested: ['1', '2'] }, + }) + saveLog = () => {} +} + +test(`[CoreClass] Probando instanciamiento de clase`, async () => { + const setting = { + flow: new MockFlow(), + database: new MockDBA(), + provider: new MockProvider(), + } + const bot = await createBot(setting) + assert.is(bot instanceof CoreClass, true) +}) + +test(`[CoreClass createFlow] Probando instanciamiento de clase`, async () => { + const mockCreateFlow = createFlow([]) + assert.is(mockCreateFlow instanceof FlowClass, true) +}) + +test(`[CoreClass createProvider] Probando instanciamiento de clase`, async () => { + const mockCreateProvider = createProvider(MockProvider) + assert.is(mockCreateProvider instanceof ProviderClass, true) +}) + +test(`[Bot] Eventos 'require_action,ready,auth_failure,message '`, async () => { + let responseEvents = {} + + const MOCK_EVENTS = { + require_action: { + instructions: 'Debes...', + }, + ready: true, + auth_failure: { + instructions: 'Error...', + }, + message: { + from: 'XXXXXX', + body: 'hola', + hasMedia: false, + }, + } + + const mockProvider = new MockProvider() + + const setting = { + flow: new MockFlow(), + database: new MockDBA(), + provider: mockProvider, + } + await createBot(setting) + + /// Escuchamos eventos + mockProvider.on( + 'require_action', + (r) => (responseEvents['require_action'] = r) + ) + mockProvider.on('ready', (r) => (responseEvents['ready'] = r)) + mockProvider.on('auth_failure', (r) => (responseEvents['auth_failure'] = r)) + mockProvider.on('message', (r) => (responseEvents['message'] = r)) + + /// Emitimos eventos + mockProvider.delaySendMessage( + 0, + 'require_action', + MOCK_EVENTS.require_action + ) + mockProvider.delaySendMessage(0, 'ready', MOCK_EVENTS.ready) + mockProvider.delaySendMessage(0, 'auth_failure', MOCK_EVENTS.auth_failure) + mockProvider.delaySendMessage(0, 'message', MOCK_EVENTS.message) + + await delay(0) + + /// Testeamos eventos + assert.is( + JSON.stringify(responseEvents.require_action), + JSON.stringify(MOCK_EVENTS.require_action) + ) + assert.is(responseEvents.ready, MOCK_EVENTS.ready) + + assert.is( + JSON.stringify(responseEvents.auth_failure), + JSON.stringify(MOCK_EVENTS.auth_failure) + ) + + assert.is( + JSON.stringify(responseEvents.message), + JSON.stringify(MOCK_EVENTS.message) + ) +}) + +test(`[Bot] Probando Flujos Internos`, async () => { + let responseEvents = {} + + const MOCK_EVENTS = { + require_action: { + instructions: 'Debes...', + }, + ready: true, + auth_failure: { + instructions: 'Error...', + }, + message: { + from: 'XXXXXX', + body: 'hola', + hasMedia: false, + }, + } + + const mockProvider = new MockProvider() + + const setting = { + flow: new MockFlow(), + database: new MockDBB(), + provider: mockProvider, + } + await createBot(setting) + + /// Escuchamos eventos + mockProvider.on( + 'require_action', + (r) => (responseEvents['require_action'] = r) + ) + mockProvider.on('ready', (r) => (responseEvents['ready'] = r)) + mockProvider.on('auth_failure', (r) => (responseEvents['auth_failure'] = r)) + mockProvider.on('message', (r) => (responseEvents['message'] = r)) + + /// Emitimos eventos + mockProvider.delaySendMessage( + 0, + 'require_action', + MOCK_EVENTS.require_action + ) + mockProvider.delaySendMessage(0, 'ready', MOCK_EVENTS.ready) + mockProvider.delaySendMessage(0, 'auth_failure', MOCK_EVENTS.auth_failure) + mockProvider.delaySendMessage(0, 'message', MOCK_EVENTS.message) + + await delay(0) + + /// Testeamos eventos + assert.is( + JSON.stringify(responseEvents.require_action), + JSON.stringify(MOCK_EVENTS.require_action) + ) + assert.is(responseEvents.ready, MOCK_EVENTS.ready) + + assert.is( + JSON.stringify(responseEvents.auth_failure), + JSON.stringify(MOCK_EVENTS.auth_failure) + ) + + assert.is( + JSON.stringify(responseEvents.message), + JSON.stringify(MOCK_EVENTS.message) + ) +}) + +test(`[Bot] Probando Flujos Nested`, async () => { + let responseEvents = {} + + const MOCK_EVENTS = { + require_action: { + instructions: 'Debes...', + }, + ready: true, + auth_failure: { + instructions: 'Error...', + }, + message: { + from: 'XXXXXX', + body: 'hola', + hasMedia: false, + }, + } + + const mockProvider = new MockProvider() + + const setting = { + flow: new MockFlow(), + database: new MockDBC(), + provider: mockProvider, + } + const botInstance = await createBot(setting) + + botInstance.sendProviderAndSave('xxxxx', 'xxxxx') + botInstance.continue('xxxxx', 'xxxxx') + /// Escuchamos eventos + mockProvider.on( + 'require_action', + (r) => (responseEvents['require_action'] = r) + ) + mockProvider.on('ready', (r) => (responseEvents['ready'] = r)) + mockProvider.on('auth_failure', (r) => (responseEvents['auth_failure'] = r)) + mockProvider.on('message', (r) => (responseEvents['message'] = r)) + + /// Emitimos eventos + mockProvider.delaySendMessage( + 0, + 'require_action', + MOCK_EVENTS.require_action + ) + mockProvider.delaySendMessage(0, 'ready', MOCK_EVENTS.ready) + mockProvider.delaySendMessage(0, 'auth_failure', MOCK_EVENTS.auth_failure) + mockProvider.delaySendMessage(0, 'message', MOCK_EVENTS.message) + + await delay(0) + + /// Testeamos eventos + assert.is( + JSON.stringify(responseEvents.require_action), + JSON.stringify(MOCK_EVENTS.require_action) + ) + assert.is(responseEvents.ready, MOCK_EVENTS.ready) + + assert.is( + JSON.stringify(responseEvents.auth_failure), + JSON.stringify(MOCK_EVENTS.auth_failure) + ) + + assert.is( + JSON.stringify(responseEvents.message), + JSON.stringify(MOCK_EVENTS.message) + ) +}) + +test.run() + +function delay(ms) { + return new Promise((res) => setTimeout(res, ms)) +} diff --git a/packages/bot/tests/methods.test.js b/packages/bot/tests/methods.test.js new file mode 100644 index 0000000..9f9452e --- /dev/null +++ b/packages/bot/tests/methods.test.js @@ -0,0 +1,152 @@ +const { test } = require('uvu') +const assert = require('uvu/assert') +const { generateRefSerialize } = require('../utils/hash') +const { addKeyword, addAnswer, toSerialize } = require('../io/methods') + +test('Debere probar las propeidades', () => { + const ARRANGE = { + keyword: 'hola!', + } + const MAIN_CTX = addKeyword(ARRANGE.keyword) + + assert.type(MAIN_CTX.addAnswer, 'function') + assert.is(MAIN_CTX.ctx.keyword, ARRANGE.keyword) +}) + +test('Debere probar las propeidades array', () => { + const ARRANGE = { + keyword: ['hola!', 'ole'], + } + const MAIN_CTX = addKeyword(ARRANGE.keyword) + + assert.is(MAIN_CTX.ctx.keyword, ARRANGE.keyword) +}) + +test('Debere probar toSerialize', () => { + const ARRANGE = { + keyword: ['hola!', 'ole'], + } + const MAIN_CTX = addKeyword(ARRANGE.keyword) + .addAnswer('Segundo!') + .addAnswer('Segundo!') + .toJson() + + const [ANSWER_A] = MAIN_CTX + + assert.is( + toSerialize(MAIN_CTX)[0].refSerialize, + generateRefSerialize({ + index: 0, + answer: ANSWER_A.answer, + keyword: ANSWER_A.keyword, + }) + ) +}) + +test('Debere probar el paso de contexto', () => { + const ARRANGE = { + keyword: 'hola!', + answer: 'Bienvenido', + } + const CTX_A = addKeyword(ARRANGE.keyword) + const CTX_B = addAnswer(CTX_A)(ARRANGE.answer) + + assert.is(CTX_A.ctx.keyword, ARRANGE.keyword) + assert.is(CTX_B.ctx.keyword, ARRANGE.keyword) + assert.is(CTX_B.ctx.answer, ARRANGE.answer) +}) + +test('Debere probar la anidación', () => { + const ARRANGE = { + keyword: 'hola!', + answer_A: 'Bienvenido', + answer_B: 'Continuar', + } + const MAIN_CTX = addKeyword(ARRANGE.keyword) + .addAnswer(ARRANGE.answer_A) + .addAnswer(ARRANGE.answer_B) + + assert.is(MAIN_CTX.ctx.answer, ARRANGE.answer_B) +}) + +test('Debere probar las poptions', () => { + const MAIN_CTX = addKeyword('etc', { sensitive: false }) + + assert.is(MAIN_CTX.ctx.options.sensitive, false) +}) + +test('Debere probar las addAnswer', () => { + const MOCK_OPT = { + media: 'http://image.mock/mock.png', + buttons: [1], + } + const MAIN_CTX = addKeyword('hola').addAnswer('etc', MOCK_OPT) + + assert.is(MAIN_CTX.ctx.options.media, MOCK_OPT.media) + assert.is(MAIN_CTX.ctx.options.buttons.length, 1) +}) + +test('Debere probar error las addAnswer', () => { + const MOCK_OPT = { + media: { a: 1, b: [] }, + buttons: 'test', + } + const MAIN_CTX = addKeyword('hola').addAnswer('etc', MOCK_OPT) + + assert.is(MAIN_CTX.ctx.options.media, null) + assert.is(MAIN_CTX.ctx.options.buttons.length, 0) +}) + +test('Obtener toJson', () => { + const [ctxA, ctxB, ctxC] = addKeyword('hola') + .addAnswer('pera!') + .addAnswer('chao') + .toJson() + + assert.is(ctxA.keyword, 'hola') + assert.match(ctxA.ref, /^key_/) + + assert.is(ctxB.answer, 'pera!') + assert.match(ctxB.ref, /^ans_/) + + assert.is(ctxC.answer, 'chao') + assert.match(ctxC.ref, /^ans_/) +}) + +test('addKeyword toJson con sensitive', () => { + const [ctxA] = addKeyword('hola').toJson() + assert.is(ctxA.options.sensitive, false) + const [ctxB] = addKeyword('hola', { sensitive: true }).toJson() + assert.is(ctxB.options.sensitive, true) +}) + +test('addAnswer toJson con IMG', () => { + const [, ctxB, ctxC] = addKeyword('hola') + .addAnswer('bye!', { + media: 'http://mock.img/file-a.png', + }) + .addAnswer('otro!', { + media: 'http://mock.img/file-b.png', + }) + .toJson() + + assert.is(ctxB.options.media, 'http://mock.img/file-a.png') + assert.is(ctxC.options.media, 'http://mock.img/file-b.png') +}) + +test('addAnswer toJson con BUTTONS', () => { + const [, ctxB] = addKeyword('hola') + .addAnswer('mis opciones!', { + buttons: [{ body: 'BTN_1' }, { body: 'BTN_2' }], + }) + .toJson() + + assert.is(ctxB.options.buttons.length, 2) + + const [btnA, btnB] = ctxB.options.buttons + + assert.is(btnA.body, 'BTN_1') + assert.is(btnB.body, 'BTN_2') +}) + +test.run() diff --git a/packages/bot/utils/hash.js b/packages/bot/utils/hash.js new file mode 100644 index 0000000..84a25db --- /dev/null +++ b/packages/bot/utils/hash.js @@ -0,0 +1,24 @@ +const crypto = require('crypto') + +/** + * Generamos un UUID unico con posibilidad de tener un prefijo + * @param {*} prefix + * @returns + */ +const generateRef = (prefix = false) => { + const id = crypto.randomUUID() + return prefix ? `${prefix}_${id}` : id +} + +/** + * Genera un HASH MD5 + * @param {*} param0 + * @returns + */ +const generateRefSerialize = ({ index, answer, keyword }) => + crypto + .createHash('md5') + .update(JSON.stringify({ index, answer, keyword })) + .digest('hex') + +module.exports = { generateRef, generateRefSerialize } diff --git a/packages/bot/utils/interactive.js b/packages/bot/utils/interactive.js new file mode 100644 index 0000000..c317a7f --- /dev/null +++ b/packages/bot/utils/interactive.js @@ -0,0 +1,14 @@ +const { yellow, bgRed } = require('kleur') +const NODE_ENV = process.env.NODE_ENV || 'dev' +const printer = (message, title) => { + if (NODE_ENV !== 'test') { + // console.clear() + if (title) console.log(bgRed(`${title}`)) + console.log( + yellow(Array.isArray(message) ? message.join('\n') : message) + ) + console.log(``) + } +} + +module.exports = { printer } diff --git a/packages/cli/README.md b/packages/cli/README.md new file mode 100644 index 0000000..9923c6c --- /dev/null +++ b/packages/cli/README.md @@ -0,0 +1,20 @@ +# @bot-whatsapp/cli + +- [x] Revisar version de NODE +- [x] Revisar OS +- [x] Obtener Package Manager +- [x] Revisar las libreria de WhatsappWeb para obtener version reciente +- [x] Opcion interactiva de limpiar session +- [x] Opcion de generar `json` con la configuracion +- [x] Agregar `rollup` para limpiar el codigo + +--- + +**Comunidad** + +> Forma parte de este proyecto. + +- [Discord](https://link.codigoencasa.com/DISCORD) +- [Twitter](https://twitter.com/leifermendez) +- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR) +- [Telegram](https://t.me/leifermendez) diff --git a/packages/cli/bin/cli.js b/packages/cli/bin/cli.js new file mode 100644 index 0000000..1a1e282 --- /dev/null +++ b/packages/cli/bin/cli.js @@ -0,0 +1,3 @@ +#!/usr/bin/env node +const index = require('../lib/cli/bundle.cli.cjs') +index.startInteractive() diff --git a/packages/cli/check/index.js b/packages/cli/check/index.js new file mode 100644 index 0000000..ec4f235 --- /dev/null +++ b/packages/cli/check/index.js @@ -0,0 +1,38 @@ +const { red, yellow, green, bgCyan } = require('kleur') + +const checkNodeVersion = () => { + console.log(bgCyan('🚀 Revisando tu Node.js')) + const version = process.version + const majorVersion = parseInt(version.replace('v', '').split('.').shift()) + if (majorVersion < 16) { + console.error( + red( + `🔴 Se require Node.js 16 o superior. Actualmente esta ejecutando Node.js ${version}` + ) + ) + process.exit(1) + } + console.log(green(`Node.js combatible ${version}`)) + console.log(``) +} + +const checkOs = () => { + console.log(bgCyan('🙂 Revisando tu Sistema Operativo')) + const os = process.platform + if (!os.includes('win32')) { + const messages = [ + `El sistema operativo actual (${os}) posiblemente requiera`, + `una confiuración adicional referente al puppeter`, + ``, + `Recuerda pasar por el WIKI`, + `🔗 https://github.com/leifermendez/bot-whatsapp/wiki/Instalaci%C3%B3n`, + ``, + ] + + console.log(yellow(messages.join(' \n'))) + } + + console.log(``) +} + +module.exports = { checkNodeVersion, checkOs } diff --git a/packages/cli/clean/index.js b/packages/cli/clean/index.js new file mode 100644 index 0000000..6b7f6fd --- /dev/null +++ b/packages/cli/clean/index.js @@ -0,0 +1,19 @@ +const rimraf = require('rimraf') +const { yellow } = require('kleur') +const { join } = require('path') + +const PATH_WW = [ + join(process.cwd(), '.wwebjs_auth'), + join(process.cwd(), 'session.json'), +] + +const cleanSession = () => { + const queue = [] + for (const PATH of PATH_WW) { + console.log(yellow(`😬 Eliminando: ${PATH}`)) + queue.push(rimraf(PATH, () => Promise.resolve())) + } + return Promise.all(queue) +} + +module.exports = { cleanSession } diff --git a/packages/cli/configuration/index.js b/packages/cli/configuration/index.js new file mode 100644 index 0000000..50565d9 --- /dev/null +++ b/packages/cli/configuration/index.js @@ -0,0 +1,33 @@ +const { writeFile } = require('fs').promises +const { join } = require('path') + +/** + * JSON_TEMPLATE = {[key:string]{...pros}} + */ +const JSON_TEMPLATE = { + provider: { + vendor: '', + }, + database: { + host: '', + password: '', + port: '', + username: '', + db: '', + }, + io: { + vendor: '', + }, +} + +const PATH_CONFIG = join(process.cwd(), 'config.json') + +const jsonConfig = () => { + return writeFile( + PATH_CONFIG, + JSON.stringify(JSON_TEMPLATE, null, 2), + 'utf-8' + ) +} + +module.exports = { jsonConfig } diff --git a/chats/.gitkeep b/packages/cli/db/index.js similarity index 100% rename from chats/.gitkeep rename to packages/cli/db/index.js diff --git a/packages/cli/index.js b/packages/cli/index.js new file mode 100644 index 0000000..70785fc --- /dev/null +++ b/packages/cli/index.js @@ -0,0 +1,3 @@ +const { startInteractive } = require('./interactive') +if (process.env.NODE_ENV === 'dev') startInteractive() +module.exports = { startInteractive } diff --git a/packages/cli/install/index.js b/packages/cli/install/index.js new file mode 100644 index 0000000..0c18761 --- /dev/null +++ b/packages/cli/install/index.js @@ -0,0 +1,24 @@ +const { readFileSync, existsSync } = require('fs') +const { join } = require('path') +const { installDeps, getPkgManage } = require('./tool') + +const PATHS_DIR = [ + join(__dirname, 'pkg-to-update.json'), + join(__dirname, '..', 'pkg-to-update.json'), + join(__dirname, '..', '..', 'pkg-to-update.json'), +] + +const PKG_TO_UPDATE = () => { + const PATH_INDEX = PATHS_DIR.findIndex((a) => existsSync(a)) + const data = readFileSync(PATHS_DIR[PATH_INDEX], 'utf-8') + const dataParse = JSON.parse(data) + const pkg = Object.keys(dataParse).map((n) => `${n}@${dataParse[n]}`) + return pkg +} + +const installAll = async () => { + const pkg = await getPkgManage() + installDeps(pkg, PKG_TO_UPDATE()).runInstall() +} + +module.exports = { installAll } diff --git a/packages/cli/install/tool.js b/packages/cli/install/tool.js new file mode 100644 index 0000000..6ec176c --- /dev/null +++ b/packages/cli/install/tool.js @@ -0,0 +1,68 @@ +const { red } = require('kleur') +const spawn = require('cross-spawn') +// const { detect } = require('detect-package-manager') +const PKG_OPTION = { + npm: 'install', + yarn: 'add', + pnpm: 'add', +} + +const getPkgManage = async () => { + // const pkg = await detect() + // return pkg + return 'npm' +} + +const installDeps = (pkgManager, packageList) => { + const errorMessage = `Ocurrio un error instalando ${packageList}` + let childProcess = [] + + const installSingle = (pkgInstall) => () => { + new Promise((resolve) => { + try { + childProcess = spawn( + pkgManager, + [PKG_OPTION[pkgManager], pkgInstall], + { + stdio: 'inherit', + } + ) + + childProcess.on('error', (e) => { + console.error(e) + console.error(red(errorMessage)) + resolve() + }) + + childProcess.on('close', (code) => { + if (code === 0) { + resolve() + } else { + console.error(code) + console.error(red(errorMessage)) + } + }) + + resolve() + } catch (e) { + console.error(e) + console.error(red(errorMessage)) + } + }) + } + + if (typeof packageList === 'string') { + childProcess.push(installSingle(packageList)) + } else { + for (const pkg of packageList) { + childProcess.push(installSingle(pkg)) + } + } + + const runInstall = () => { + return Promise.all(childProcess.map((i) => i())) + } + return { runInstall } +} + +module.exports = { getPkgManage, installDeps } diff --git a/packages/cli/interactive/index.js b/packages/cli/interactive/index.js new file mode 100644 index 0000000..c86a568 --- /dev/null +++ b/packages/cli/interactive/index.js @@ -0,0 +1,127 @@ +const prompts = require('prompts') +const { yellow, red } = require('kleur') +const { installAll } = require('../install') +const { cleanSession } = require('../clean') +const { checkNodeVersion, checkOs } = require('../check') +const { jsonConfig } = require('../configuration') + +const startInteractive = async () => { + const questions = [ + { + type: 'text', + name: 'dependencies', + message: + 'Quieres actualizar las librerias "whatsapp-web.js"? (Y/n)', + }, + { + type: 'text', + name: 'cleanTmp', + message: 'Quieres limpiar la session del bot? (Y/n)', + }, + { + type: 'multiselect', + name: 'providerWs', + message: 'Proveedor de Whatsapp', + choices: [ + { title: 'whatsapp-web.js', value: 'whatsapp-web.js' }, + { title: 'API Oficial (Meta)', value: 'meta', disabled: true }, + { title: 'Twilio', value: 'twilio', disabled: true }, + ], + max: 1, + hint: 'Espacio para selecionar', + instructions: '↑/↓', + }, + { + type: 'multiselect', + name: 'providerDb', + message: 'Cual base de datos quieres usar', + choices: [ + { title: 'JSONFile', value: 'json' }, + { title: 'MySQL', value: 'mysql', disabled: true }, + { title: 'Mongo', value: 'mongo', disabled: true }, + ], + max: 1, + hint: 'Espacio para selecionar', + instructions: '↑/↓', + }, + ] + + console.clear() + checkNodeVersion() + checkOs() + const onCancel = () => { + console.log('Proceso cancelado!') + return true + } + const response = await prompts(questions, { onCancel }) + const { + dependencies = '', + cleanTmp = '', + providerDb = [], + providerWs = [], + } = response + /** + * Question #1 + * @returns + */ + const installOrUdpateDep = async () => { + const answer = dependencies.toLowerCase() || 'n' + if (answer.includes('n')) return true + + if (answer.includes('y')) { + await installAll() + return true + } + } + + /** + * Question #2 + * @returns + */ + const cleanAllSession = async () => { + const answer = cleanTmp.toLowerCase() || 'n' + if (answer.includes('n')) return true + + if (answer.includes('y')) { + await cleanSession() + return true + } + } + + const vendorProvider = async () => { + if (!providerWs.length) { + console.log( + red( + `Debes de seleccionar una WS Provider. Tecla [Space] para seleccionar` + ) + ) + process.exit(1) + } + console.log(yellow(`'Deberia crer una carpeta en root/provider'`)) + return true + } + + const dbProvider = async () => { + const answer = providerDb + if (!providerDb.length) { + console.log( + red( + `Debes de seleccionar una DB Provider. Tecla [Space] para seleccionar` + ) + ) + process.exit(1) + } + if (answer === 'json') { + console.log('Deberia crer una carpeta en root/data') + return 1 + } + } + + await installOrUdpateDep() + await cleanAllSession() + await vendorProvider() + await dbProvider() + await jsonConfig() +} + +module.exports = { startInteractive } diff --git a/packages/cli/package.json b/packages/cli/package.json new file mode 100644 index 0000000..fd9b7f7 --- /dev/null +++ b/packages/cli/package.json @@ -0,0 +1,19 @@ +{ + "name": "@bot-whatsapp/cli", + "version": "0.0.1", + "description": "", + "main": "index.js", + "private": true, + "devDependencies": { + "cross-env": "^7.0.3", + "cross-spawn": "^7.0.3", + "detect-package-manager": "^2.0.1", + "kleur": "^4.1.5" + }, + "files": [ + "./lib/cli/bundle.cli.cjs" + ], + "bin": { + "bot": "./bin/cli.js" + } +} diff --git a/packages/cli/pkg-to-update.json b/packages/cli/pkg-to-update.json new file mode 100644 index 0000000..4a13049 --- /dev/null +++ b/packages/cli/pkg-to-update.json @@ -0,0 +1,3 @@ +{ + "whatsapp-web.js": "latest" +} diff --git a/media/.gitkeep b/packages/cli/provider/index.js similarity index 100% rename from media/.gitkeep rename to packages/cli/provider/index.js diff --git a/packages/cli/rollup-cli.config.js b/packages/cli/rollup-cli.config.js new file mode 100644 index 0000000..047909c --- /dev/null +++ b/packages/cli/rollup-cli.config.js @@ -0,0 +1,14 @@ +const commonjs = require('@rollup/plugin-commonjs') +const { nodeResolve } = require('@rollup/plugin-node-resolve') +const { join } = require('path') + +const PATH = join(__dirname, 'lib', 'cli', 'bundle.cli.cjs') + +module.exports = { + input: join(__dirname, 'index.js'), + output: { + file: PATH, + format: 'cjs', + }, + plugins: [commonjs(), nodeResolve()], +} diff --git a/packages/database/README.md b/packages/database/README.md new file mode 100644 index 0000000..e226f15 --- /dev/null +++ b/packages/database/README.md @@ -0,0 +1,41 @@ +### 🚀 Package (@bot-whatsapp/database) + +Este package tiene como reponsabilidad proveer de diferentes adaptadores para la capa de datos. +La idea es brindar multiples opciones como un adaptador de MySQL, Mongo, entre otros. + +Ejemplo de como se implementaria: + + +```js +const MongoAdapter = require('@bot-whatsapp/database/mongo') +/// o +const MySQLAdapter = require('@bot-whatsapp/database/mysql') + +const main = async () => { + + const adapterDB = new MongoAdapter() + const adapterFlow = createFlow([flujoBot]) + const adapterProvider = createProvider(WebWhatsappProvider) + + createBot({ + flow: adapterFlow, + provider: adapterProvider, + database: adapterDB, + }) +} +``` + +#### Video + +> Video explicando como debes de agregar nuevos adaptadores +[![Video](https://i.imgur.com/DlxJIKV.gif)](https://youtu.be/Sjzkpg1OJuY) +--- + +**Comunidad** + +> Forma parte de este proyecto. + +- [Discord](https://link.codigoencasa.com/DISCORD) +- [Twitter](https://twitter.com/leifermendez) +- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR) +- [Telegram](https://t.me/leifermendez) diff --git a/packages/database/package.json b/packages/database/package.json new file mode 100644 index 0000000..d2bcd3c --- /dev/null +++ b/packages/database/package.json @@ -0,0 +1,18 @@ +{ + "name": "@bot-whatsapp/database", + "version": "0.0.1", + "description": "Esto es el conector a mysql, pg, mongo", + "main": "./lib/mock/index.cjs", + "private": true, + "keywords": [], + "author": "", + "license": "ISC", + "dependencies": { + "dotenv": "^16.0.3", + "mongodb": "^4.11.0" + }, + "exports": { + "./mock": "./lib/mock/index.cjs", + "./mongo": "./lib/mongo/index.cjs" + } +} diff --git a/packages/database/rollup-database.config.js b/packages/database/rollup-database.config.js new file mode 100644 index 0000000..234425e --- /dev/null +++ b/packages/database/rollup-database.config.js @@ -0,0 +1,21 @@ +const commonjs = require('@rollup/plugin-commonjs') +const { join } = require('path') + +module.exports = [ + { + input: join(__dirname, 'src', 'mock', 'index.js'), + output: { + file: join(__dirname, 'lib', 'mock', 'index.cjs'), + format: 'cjs', + }, + plugins: [commonjs()], + }, + { + input: join(__dirname, 'src', 'mongo', 'index.js'), + output: { + file: join(__dirname, 'lib', 'mongo', 'index.cjs'), + format: 'cjs', + }, + plugins: [commonjs()], + }, +] diff --git a/packages/database/src/mock/index.js b/packages/database/src/mock/index.js new file mode 100644 index 0000000..7422687 --- /dev/null +++ b/packages/database/src/mock/index.js @@ -0,0 +1,17 @@ +class MockDatabase { + listHistory = [] + + constructor() { + /** + * Se debe cargar listHistory con historial de mensajes + * para que se pueda continuar el flow + */ + } + + save = (ctx) => { + console.log('Guardando DB...', ctx) + this.listHistory.push(ctx) + } +} + +module.exports = MockDatabase diff --git a/packages/database/src/mongo/index.js b/packages/database/src/mongo/index.js new file mode 100644 index 0000000..21b7cf1 --- /dev/null +++ b/packages/database/src/mongo/index.js @@ -0,0 +1,46 @@ +require('dotenv').config() +const { MongoClient } = require('mongodb') + +const DB_URI = process.env.DB_URI || 'mongodb://0.0.0.0:27017' +const DB_NAME = process.env.DB_NAME || 'db_bot' + +class MongoAdapter { + db + listHistory = [] + + constructor() { + this.init().then() + } + + init = async () => { + try { + const client = new MongoClient(DB_URI, {}) + await client.connect() + console.log('🆗 Conexión Correcta DB') + const db = client.db(DB_NAME) + this.db = db + return true + } catch (e) { + console.log('Error', e) + return + } + } + + getPrevByNumber = async (from) => { + const result = await this.db + .collection('history') + .find({ from }) + .sort({ _id: -1 }) + .limit(1) + .toArray() + return result[0] + } + + save = async (ctx) => { + await this.db.collection('history').insert(ctx) + console.log('Guardando DB...', ctx) + this.listHistory.push(ctx) + } +} + +module.exports = MongoAdapter diff --git a/packages/docs/.eslintignore b/packages/docs/.eslintignore new file mode 100644 index 0000000..73d2683 --- /dev/null +++ b/packages/docs/.eslintignore @@ -0,0 +1,33 @@ +**/*.log +**/.DS_Store +*. +.vscode/settings.json +.history +.yarn +bazel-* +bazel-bin +bazel-out +bazel-qwik +bazel-testlogs +dist +dist-dev +lib +lib-types +etc +external +node_modules +temp +tsc-out +tsdoc-metadata.json +target +output +rollup.config.js +build +.cache +.vscode +.rollup.cache +dist +tsconfig.tsbuildinfo +vite.config.ts +*.spec.tsx +*.spec.ts diff --git a/packages/docs/.eslintrc.cjs b/packages/docs/.eslintrc.cjs new file mode 100644 index 0000000..c31c7a9 --- /dev/null +++ b/packages/docs/.eslintrc.cjs @@ -0,0 +1,40 @@ +module.exports = { + root: true, + env: { + browser: true, + es2021: true, + node: true, + }, + extends: [ + 'eslint:recommended', + 'plugin:@typescript-eslint/recommended', + 'plugin:qwik/recommended', + ], + parser: '@typescript-eslint/parser', + parserOptions: { + tsconfigRootDir: __dirname, + project: ['./tsconfig.json'], + ecmaVersion: 2021, + sourceType: 'module', + ecmaFeatures: { + jsx: true, + }, + }, + plugins: ['@typescript-eslint'], + rules: { + '@typescript-eslint/no-explicit-any': 'off', + '@typescript-eslint/explicit-module-boundary-types': 'off', + '@typescript-eslint/no-inferrable-types': 'off', + '@typescript-eslint/no-non-null-assertion': 'off', + '@typescript-eslint/no-empty-interface': 'off', + '@typescript-eslint/no-namespace': 'off', + '@typescript-eslint/no-empty-function': 'off', + '@typescript-eslint/no-this-alias': 'off', + '@typescript-eslint/ban-types': 'off', + '@typescript-eslint/ban-ts-comment': 'off', + 'prefer-spread': 'off', + 'no-case-declarations': 'off', + 'no-console': 'off', + '@typescript-eslint/no-unused-vars': ['error'], + }, +}; diff --git a/packages/docs/.gitignore b/packages/docs/.gitignore new file mode 100644 index 0000000..b29623f --- /dev/null +++ b/packages/docs/.gitignore @@ -0,0 +1,41 @@ +# Build +/dist +/lib +/lib-types +/server + +# Development +node_modules + +# Cache +.cache +.mf +.vscode +.rollup.cache +tsconfig.tsbuildinfo + +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +# Editor +!.vscode/extensions.json +.idea +.DS_Store +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? + +# Yarn +.yarn/* +!.yarn/releases + +# Cloudflare +functions/**/*.js diff --git a/packages/docs/.node-version b/packages/docs/.node-version new file mode 100644 index 0000000..b6a7d89 --- /dev/null +++ b/packages/docs/.node-version @@ -0,0 +1 @@ +16 diff --git a/packages/docs/.prettierignore b/packages/docs/.prettierignore new file mode 100644 index 0000000..1592248 --- /dev/null +++ b/packages/docs/.prettierignore @@ -0,0 +1,6 @@ +# Files Prettier should not format +**/*.log +**/.DS_Store +*. +dist +node_modules diff --git a/packages/docs/README.md b/packages/docs/README.md new file mode 100644 index 0000000..9948f9c --- /dev/null +++ b/packages/docs/README.md @@ -0,0 +1,11 @@ +### 😎 Documentación Bot-Whatsapp + +👉 [https://bot-whatsapp.pages.dev/](https://bot-whatsapp.pages.dev/) + +Se esta iniciando una documentación oficial sobre como usar e implementar los diferentes funcionalidades del bot-wahtsapp + + +La idea es cada usuario pueda ir aportando a la documentacion y formar parte de este proyecto. + + +##### ¿Como agregar documentación? [Video] \ No newline at end of file diff --git a/packages/docs/adaptors/cloudflare-pages/vite.config.ts b/packages/docs/adaptors/cloudflare-pages/vite.config.ts new file mode 100644 index 0000000..b03956f --- /dev/null +++ b/packages/docs/adaptors/cloudflare-pages/vite.config.ts @@ -0,0 +1,19 @@ +import { cloudflarePagesAdaptor } from '@builder.io/qwik-city/adaptors/cloudflare-pages/vite'; +import { extendConfig } from '@builder.io/qwik-city/vite'; +import baseConfig from '../../vite.config'; + +export default extendConfig(baseConfig, () => { + return { + build: { + ssr: true, + rollupOptions: { + input: ['src/entry.cloudflare-pages.tsx', '@qwik-city-plan'], + }, + }, + plugins: [ + cloudflarePagesAdaptor({ + staticGenerate: true, + }), + ], + }; +}); diff --git a/packages/docs/functions/[[path]].ts b/packages/docs/functions/[[path]].ts new file mode 100644 index 0000000..66013ad --- /dev/null +++ b/packages/docs/functions/[[path]].ts @@ -0,0 +1,5 @@ +// @ts-ignore + +// Cloudflare Pages Functions +// https://developers.cloudflare.com/pages/platform/functions/ +export { onRequest } from '../server/entry.cloudflare-pages'; diff --git a/packages/docs/package.json b/packages/docs/package.json new file mode 100644 index 0000000..a4425cc --- /dev/null +++ b/packages/docs/package.json @@ -0,0 +1,41 @@ +{ + "name": "bot-whatsapp-docs", + "version": "0.0.1", + "description": "Basic start point to build a docs site with Qwik", + "engines": { + "node": ">=15.0.0" + }, + "private": true, + "scripts": { + "build": "qwik build", + "build.client": "vite build", + "build.preview": "vite build --ssr src/entry.preview.tsx", + "build.server": "vite build -c adaptors/cloudflare-pages/vite.config.ts", + "build.types": "tsc --incremental --noEmit", + "deploy": "wrangler pages dev ./dist", + "dev": "vite --mode ssr", + "dev.debug": "node --inspect-brk ./node_modules/vite/bin/vite.js --mode ssr --force", + "fmt": "prettier --write .", + "fmt.check": "prettier --check .", + "lint": "eslint \"src/**/*.ts*\"", + "preview": "qwik build preview && vite preview --open", + "start": "vite --open --mode ssr", + "qwik": "qwik" + }, + "devDependencies": { + "@builder.io/qwik": "0.14.1", + "@builder.io/qwik-city": "0.0.127", + "@types/eslint": "8.4.10", + "@types/node": "latest", + "@typescript-eslint/eslint-plugin": "5.43.0", + "@typescript-eslint/parser": "5.43.0", + "eslint": "8.28.0", + "eslint-plugin-qwik": "0.14.1", + "node-fetch": "3.3.0", + "prettier": "2.7.1", + "typescript": "4.9.3", + "vite": "3.2.4", + "vite-tsconfig-paths": "3.5.0", + "wrangler": "latest" + } +} diff --git a/packages/docs/public/_headers b/packages/docs/public/_headers new file mode 100644 index 0000000..0690cb4 --- /dev/null +++ b/packages/docs/public/_headers @@ -0,0 +1,4 @@ +# https://developers.cloudflare.com/pages/platform/headers/ + +/build/* + Cache-Control: public, max-age=31536000, s-maxage=31536000, immutable diff --git a/packages/docs/public/_redirects b/packages/docs/public/_redirects new file mode 100644 index 0000000..e274610 --- /dev/null +++ b/packages/docs/public/_redirects @@ -0,0 +1 @@ +# https://developers.cloudflare.com/pages/platform/redirects/ diff --git a/packages/docs/public/favicon.svg b/packages/docs/public/favicon.svg new file mode 100644 index 0000000..0ded7c1 --- /dev/null +++ b/packages/docs/public/favicon.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/packages/docs/public/manifest.json b/packages/docs/public/manifest.json new file mode 100644 index 0000000..c18e75f --- /dev/null +++ b/packages/docs/public/manifest.json @@ -0,0 +1,9 @@ +{ + "$schema": "https://json.schemastore.org/web-manifest-combined.json", + "name": "qwik-project-name", + "short_name": "Welcome to Qwik", + "start_url": ".", + "display": "standalone", + "background_color": "#fff", + "description": "A Qwik project app." +} diff --git a/mediaSend/.gitkeep b/packages/docs/public/robots.txt similarity index 100% rename from mediaSend/.gitkeep rename to packages/docs/public/robots.txt diff --git a/packages/docs/src/components/breadcrumbs/breadcrumbs.css b/packages/docs/src/components/breadcrumbs/breadcrumbs.css new file mode 100644 index 0000000..c278f51 --- /dev/null +++ b/packages/docs/src/components/breadcrumbs/breadcrumbs.css @@ -0,0 +1,25 @@ +nav.breadcrumbs { + padding: 5px; + border-bottom: 1px solid #ddd; +} + +nav.breadcrumbs > span { + display: inline-block; + padding: 5px 0; + font-size: 12px; +} + +nav.breadcrumbs > span a { + text-decoration: none; + color: inherit; +} + +nav.breadcrumbs > span::after { + content: '>'; + padding: 0 5px; + opacity: 0.4; +} + +nav.breadcrumbs > span:last-child::after { + display: none; +} diff --git a/packages/docs/src/components/breadcrumbs/breadcrumbs.tsx b/packages/docs/src/components/breadcrumbs/breadcrumbs.tsx new file mode 100644 index 0000000..36de82f --- /dev/null +++ b/packages/docs/src/components/breadcrumbs/breadcrumbs.tsx @@ -0,0 +1,74 @@ +import { component$, useStyles$ } from '@builder.io/qwik'; +import { useContent, useLocation, ContentMenu } from '@builder.io/qwik-city'; +import styles from './breadcrumbs.css?inline'; + +export const Breadcrumbs = component$(() => { + useStyles$(styles); + + const { menu } = useContent(); + const loc = useLocation(); + + const breadcrumbs = createBreadcrumbs(menu, loc.pathname); + if (breadcrumbs.length === 0) { + return null; + } + + return ( + + ); +}); + +export function createBreadcrumbs(menu: ContentMenu | undefined, pathname: string) { + if (menu?.items) { + for (const indexA of menu.items) { + const breadcrumbA: ContentBreadcrumb = { + text: indexA.text, + }; + if (typeof indexA.href === 'string') { + breadcrumbA.href = indexA.href; + } + if (indexA.href === pathname) { + return [breadcrumbA]; + } + + if (indexA.items) { + for (const indexB of indexA.items) { + const breadcrumbB: ContentBreadcrumb = { + text: indexB.text, + }; + if (typeof indexB.href === 'string') { + breadcrumbB.href = indexB.href; + } + if (indexB.href === pathname) { + return [breadcrumbA, breadcrumbB]; + } + + if (indexB.items) { + for (const indexC of indexB.items) { + const breadcrumbC: ContentBreadcrumb = { + text: indexC.text, + }; + if (typeof indexC.href === 'string') { + breadcrumbC.href = indexC.href; + } + if (indexC.href === pathname) { + return [breadcrumbA, breadcrumbB, breadcrumbC]; + } + } + } + } + } + } + } + + return []; +} + +interface ContentBreadcrumb { + text: string; + href?: string; +} diff --git a/packages/docs/src/components/footer/footer.css b/packages/docs/src/components/footer/footer.css new file mode 100644 index 0000000..2e037e8 --- /dev/null +++ b/packages/docs/src/components/footer/footer.css @@ -0,0 +1,22 @@ +footer { + border-top: 0.5px solid #ddd; + margin-top: 40px; + padding: 20px; + text-align: center; +} + +footer a { + color: #9e9e9e; + font-size: 12px; +} + +footer ul { + list-style: none; + margin: 0; + padding: 0; +} + +footer li { + display: inline-block; + padding: 6px 12px; +} diff --git a/packages/docs/src/components/footer/footer.tsx b/packages/docs/src/components/footer/footer.tsx new file mode 100644 index 0000000..ce84ac1 --- /dev/null +++ b/packages/docs/src/components/footer/footer.tsx @@ -0,0 +1,36 @@ +import { component$, useStyles$ } from '@builder.io/qwik'; +import styles from './footer.css?inline'; + +export default component$(() => { + useStyles$(styles); + + return ( + + ); +}); diff --git a/packages/docs/src/components/header/header.css b/packages/docs/src/components/header/header.css new file mode 100644 index 0000000..fe3a1f6 --- /dev/null +++ b/packages/docs/src/components/header/header.css @@ -0,0 +1,34 @@ +header { + position: sticky; + top: 0; + z-index: 11; + display: grid; + grid-template-columns: minmax(130px, auto) 1fr; + gap: 30px; + height: 80px; + width: 100%; + padding: 10px; + background-color: white; + overflow: hidden; +} + +header a.logo { + display: block; +} + +header a { + text-decoration: none; +} + +header nav { + text-align: right; +} + +header nav a { + display: inline-block; + padding: 5px 15px; +} + +header nav a:hover { + text-decoration: underline; +} diff --git a/packages/docs/src/components/header/header.tsx b/packages/docs/src/components/header/header.tsx new file mode 100644 index 0000000..a4c3798 --- /dev/null +++ b/packages/docs/src/components/header/header.tsx @@ -0,0 +1,26 @@ +import { component$, useStyles$ } from '@builder.io/qwik'; +import { useLocation } from '@builder.io/qwik-city'; +import { QwikLogo } from '../icons/qwik'; +import styles from './header.css?inline'; + +export default component$(() => { + useStyles$(styles); + + const { pathname } = useLocation(); + + return ( +
+ + +
+ ); +}); diff --git a/packages/docs/src/components/icons/qwik.tsx b/packages/docs/src/components/icons/qwik.tsx new file mode 100644 index 0000000..984eb66 --- /dev/null +++ b/packages/docs/src/components/icons/qwik.tsx @@ -0,0 +1,20 @@ +export const QwikLogo = () => ( + + + + + + +); diff --git a/packages/docs/src/components/menu/menu.css b/packages/docs/src/components/menu/menu.css new file mode 100644 index 0000000..6b786ae --- /dev/null +++ b/packages/docs/src/components/menu/menu.css @@ -0,0 +1,13 @@ +.menu { + background: #eee; + padding: 20px 10px; +} + +.menu h5 { + margin: 0; +} + +.menu ul { + padding-left: 20px; + margin: 5px 0 25px 0; +} diff --git a/packages/docs/src/components/menu/menu.tsx b/packages/docs/src/components/menu/menu.tsx new file mode 100644 index 0000000..8ced9e0 --- /dev/null +++ b/packages/docs/src/components/menu/menu.tsx @@ -0,0 +1,36 @@ +import { component$, useStyles$ } from '@builder.io/qwik'; +import { useContent, Link, useLocation } from '@builder.io/qwik-city'; +import styles from './menu.css?inline'; + +export default component$(() => { + useStyles$(styles); + + const { menu } = useContent(); + const loc = useLocation(); + + return ( + + ); +}); diff --git a/packages/docs/src/components/on-this-page/on-this-page.css b/packages/docs/src/components/on-this-page/on-this-page.css new file mode 100644 index 0000000..e54fe02 --- /dev/null +++ b/packages/docs/src/components/on-this-page/on-this-page.css @@ -0,0 +1,33 @@ +.on-this-page { + padding-bottom: 20px; + font-size: 0.9em; +} + +.on-this-page h6 { + margin: 10px 0; + font-weight: bold; + text-transform: uppercase; +} + +.on-this-page ul { + margin: 0; + padding: 0 0 20px 0; + list-style: none; +} + +.on-this-page a { + position: relative; + display: block; + border: 0 solid #ddd; + border-left-width: 2px; + padding: 4px 2px 4px 8px; + text-decoration: none; +} + +.on-this-page a.indent { + padding-left: 30px; +} + +.on-this-page a:hover { + border-color: var(--theme-accent); +} diff --git a/packages/docs/src/components/on-this-page/on-this-page.tsx b/packages/docs/src/components/on-this-page/on-this-page.tsx new file mode 100644 index 0000000..8c8e137 --- /dev/null +++ b/packages/docs/src/components/on-this-page/on-this-page.tsx @@ -0,0 +1,62 @@ +import { useContent, useLocation } from '@builder.io/qwik-city'; +import { component$, useStyles$ } from '@builder.io/qwik'; +import styles from './on-this-page.css?inline'; + +export default component$(() => { + useStyles$(styles); + + const { headings } = useContent(); + const contentHeadings = headings?.filter((h) => h.level === 2 || h.level === 3) || []; + + const { pathname } = useLocation(); + const editUrl = `#update-your-edit-url-for-${pathname}`; + + return ( + + ); +}); diff --git a/packages/docs/src/components/router-head/router-head.tsx b/packages/docs/src/components/router-head/router-head.tsx new file mode 100644 index 0000000..9e70703 --- /dev/null +++ b/packages/docs/src/components/router-head/router-head.tsx @@ -0,0 +1,32 @@ +import { component$ } from '@builder.io/qwik'; +import { useDocumentHead, useLocation } from '@builder.io/qwik-city'; + +/** + * The RouterHead component is placed inside of the document `` element. + */ +export const RouterHead = component$(() => { + const head = useDocumentHead(); + const loc = useLocation(); + + return ( + <> + {head.title} + + + + + + {head.meta.map((m) => ( + + ))} + + {head.links.map((l) => ( + + ))} + + {head.styles.map((s) => ( +