Compare commits

...

41 Commits

Author SHA1 Message Date
3f004d840d docs: env 2023-01-15 20:06:25 -06:00
307dd2b6a0 docs: env 2023-01-15 20:05:07 -06:00
df1fd5cb75 Update spam.json 2023-01-06 04:28:48 -06:00
c9284a9382 Update README.md 2023-01-06 04:23:27 -06:00
232b2c790a Update README.md 2023-01-06 04:10:18 -06:00
b6d272aa4f Update README.md 2023-01-06 04:09:31 -06:00
efe7a1537c Update README.md 2023-01-06 04:08:45 -06:00
2145e850f5 Update README.md 2023-01-06 04:08:11 -06:00
aaf543c3d5 Update README.md 2023-01-06 04:07:07 -06:00
cf5efe7a4b Update README.md 2023-01-06 04:04:44 -06:00
9e9f97bae8 Update README.md 2023-01-06 01:16:12 -06:00
915fbf1b8d Update README.md 2023-01-06 01:12:46 -06:00
eae1a87cd6 Update README.md 2023-01-06 01:09:30 -06:00
03c21a997a Update README.md 2023-01-06 01:06:39 -06:00
c432a97aef Update README.md 2023-01-06 01:06:03 -06:00
bc9ebc3642 Update README.md 2023-01-06 01:05:13 -06:00
b4e5c956d9 docs: cambios 2023-01-06 00:57:29 -06:00
36d0f3d0de mods a response.json 2023-01-06 00:52:45 -06:00
1f92c0802a se agregaron ejemplos de regExp 2023-01-06 00:39:46 -06:00
7b9906f18f docs: 2022-12-27 02:35:45 -06:00
629b13de23 fix: check commits and changes 2022-12-27 02:16:50 -06:00
b7d89411b3 Merge branch 'Mod-V2' of https://github.com/cheveguerra/bot-whatsapp into Mod-V2 2022-12-26 16:04:35 -06:00
2ca2eefb9a fix: fix media sin extension 2022-12-24 22:33:53 -06:00
b625787a9a Merge branch 'Mod-V2' of https://github.com/cheveguerra/bot-whatsapp into Mod-V2 2022-12-24 22:07:05 -06:00
c54111a636 feat: se agregaron ejemplos de listas y botones
Se agregaron ejemplos de listas y botones
2022-12-24 22:07:01 -06:00
dad6941a70 Merge branch 'codigoencasa:main' into Mod-V2 2022-12-23 02:55:56 -06:00
bbc6b689ac Merge branch 'codigoencasa:main' into Mod-V2 2022-12-18 12:23:32 -06:00
f8451adf94 Merge branch 'Mod-V2' of https://github.com/cheveguerra/bot-whatsapp into Mod-V2 2022-12-16 13:40:11 -06:00
aa8d8baa18 feat: se agrega %nombre% a remplazos
Se agrega %nombre% y %primer_nombre% a remplazos
2022-12-16 13:40:07 -06:00
8a678f252f Merge branch 'leifermendez:main' into Mod-V2 2022-12-15 03:59:31 -06:00
5266c7050f feat: cambios en response.json 2022-12-12 23:00:21 -06:00
a774ff23de feat: cambio en el mensaje inicial. 2022-12-12 22:56:32 -06:00
26b3fb0787 feat: cambios en response.json 2022-12-12 22:53:39 -06:00
85670e0d4e docs: cambios en README 2022-12-12 22:26:16 -06:00
ff561fce4b feat: cambios en intial y response 2022-12-12 22:16:24 -06:00
fd2954f1cd docs: cambios a README 2022-12-12 21:45:18 -06:00
9ffc9bac74 docs: cambios a README 2022-12-12 21:44:29 -06:00
ac6cba2265 Merge branch 'leifermendez:main' into Mod-V2 2022-12-12 15:23:59 -06:00
d30a44fc3e test: t1 2022-12-11 06:21:32 -06:00
ac8c09c665 Initial Mod v2 2022-12-11 03:41:41 -06:00
4334c13cca test: testting
testing
2022-12-11 03:26:08 -06:00
22 changed files with 1335 additions and 4531 deletions

View File

2
.gitattributes vendored Normal file
View File

@@ -0,0 +1,2 @@
# Auto detect text files and perform LF normalization
* text=auto

1
.gitignore vendored
View File

@@ -8,7 +8,6 @@ media/*
mediaSend/* mediaSend/*
!mediaSend/.gitkeep !mediaSend/.gitkeep
!mediaSend/nota-de-voz.mp3 !mediaSend/nota-de-voz.mp3
.env
.wwebjs_auth .wwebjs_auth
backup backup
backup/* backup/*

104
README.md
View File

@@ -1,4 +1,84 @@
## Chatbot Whatsapp (OpenSource) ## Chatbot Whatsapp (OpenSource)
#### Actualizado Diciembre 2022
Este proyecto es un fork de la **version 1** (legacy) de [Leifer Mendez](https://github.com/leifermendez/bot-whatsapp) y tiene las siguientes modificaciones:
- Permite **submenus**.
- Un submenú es una regla que **sólo se dispara** cuando la regla anterior es la especificada, los submenus se definen agregando el parametro "```pasoRequerido```" en el **response.json**.
```json
"menu":{
"replyMessage":[
"%saludo%\nHoy es %dia_semana%.\n"
],
"media":null,
"trigger":null
},
"submenu":{
"replyMessage":[
"Este submenu solo se dispara si **ANTES** se disparó la regla 'menu'"
],
"media":null,
"trigger":null,
"pasoRequerido":"menu"
}
```
- Permite **expresiones regulares** en las palabras predefinidas en el initial.json.
- Si queremos usar RegExp, en los "keywords" de **inital.json**, en lugar de un arreglo, debemos usar un string ( quitamos los [] )
y en él usamos "*" para significar cualquier texto y "|" para significar "OR", esto nos permite ser mas flexibles
con los "keywords", por ejemplo, si queremos que el mensaje pueda decir:
"Hola quiero info del paquete" o "Requiero más información"
Ponemos ```*info*``` y la regla se va a disparar porque los dos contienen "info".
Si queremos que se dispare con:
"Quiero info del paquete numero 3" o "Me gustó el paquete de Angular"
Ponemos ```*paquete*3*|*paquete*angular*``` y la regla se va a disparar porque contiene "paquete" y "3" -O- "paquete" y "angular".
```json
{
"keywords": "*pak*3*|*pak*angular*|*paquete*3*|*paquete*angular*",
"key": "paq3"
}
```
- Permite **remplazos** en el texto de los mensajes por ejemplo:
- __%saludo%__ para que aparezca "Buenos días, tardes o noches" dependiendo de la hora.
- __%primer_nombre%__ para que aparezca el nombre (hasta el primer espacio) del remitente.
- __%dia_semana%__ para que aparezca "lunes, martes, miercoles, etc" dependiendo del día de la semana.
- __%msjant_XX%__ para que aparezca el mensaje xx anterior, es decir, si quieres mostrar el texto de 2 mensajes anteriores se pone %msjant_2%.
- etc, etc, se pueden agregar mas remplazos en la funcion "remplazos" en el archivo "adapter\index.js".
- Las modificaciones están enfocadas al uso de los archivos __initial.json__ y __response.json__, yo no uso MySQL o DialogFlow, así que no sé si las modificaciones funcionen con esos modulos, en particular el __remplazo %msjant_XX%__ depende de los archivos __JSON__ que se crean en el directorio "chats".
- Tiene agregado el parche de **botones y listas**, así que funcionan sin problema (las listas no funcionan si el bot esta ligado a un número que use **Whatsapp Business**).
- Tiene los ultimos parches de **DialogFlow** (27-dic-2022) (When Dialogflow asks for an Image, then **Upload it to Google Drive** and then generate Shared Link)
## INICIA DOCUMENTACION DEL PROYECTO ORIGINAL
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 🔴
> 💥💥 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
> 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)
### Busco colaboradores ⭐
Hola amigos me gusta mucho este proyecto pero por cuestiones de tiempo se me dificulta mantener las actualizaciones si alguno quieres participar en el proyecto escribeme a leifer.contacto@gmail.com
#### 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 #### Actualización
@@ -58,6 +138,7 @@ await sendMediaVoiceNote(client, from, 'PTT-20220223-WA0000.opus')
## Instruciones ## Instruciones
__Descargar o Clonar repositorio__ __Descargar o Clonar repositorio__
![](https://i.imgur.com/dSpUbFz.png)
__Usas ¿Ubuntu / Linux?__ __Usas ¿Ubuntu / Linux?__
> Asegurate de instalar los siguientes paquetes > Asegurate de instalar los siguientes paquetes
@@ -69,9 +150,9 @@ sudo apt install -y gconf-service libasound2 libatk1.0-0 libc6 libcairo2 libcups
__Instalar dependencias (npm install)__ __Instalar dependencias (npm install)__
> Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando > Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando
``` `npm install`
npm i
``` ![](https://i.imgur.com/BJuMjGR.png)
__Configurar .env__ __Configurar .env__
> Con el editor de texto crea un archivo `.env` el cual debes de guiarte del archivo `.env.example` > Con el editor de texto crea un archivo `.env` el cual debes de guiarte del archivo `.env.example`
@@ -90,14 +171,21 @@ SQL_PASS=
SQL_DATABASE= SQL_DATABASE=
``` ```
![](https://i.imgur.com/9poNnW0.png)
__Ejecutar el script__ __Ejecutar el script__
> Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando > Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando
`npm run start` `npm run start`
![](https://i.imgur.com/eMkBkuJ.png)
__Whatsapp en tu celular__ __Whatsapp en tu celular__
> Ahora abre la aplicación de Whatsapp en tu dispositivo y escanea el código QR > Ahora abre la aplicación de Whatsapp en tu dispositivo y escanea el código QR
<img src="https://i.imgur.com/RSbPtat.png" width="500" /> <img src="https://i.imgur.com/RSbPtat.png" width="500" />
Tambien puedes visitar la pagina http://127.0.0.1:3000/qr Visitar la pagina
`http://localhost:3000/qr`
![](https://i.imgur.com/Q3JEDlP.png)
__Listo 😎__ __Listo 😎__
> Cuando sale este mensaje tu BOT está __listo__ para trabajar! > Cuando sale este mensaje tu BOT está __listo__ para trabajar!
@@ -114,3 +202,11 @@ __Listo 😎__
> Ahora deberías obtener un arespuesta por parte del BOT como la siguiente, ademas de esto tambien se crea un archivo excel > 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 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)

1
Whatsapp-Bot.ps1 Normal file
View File

@@ -0,0 +1 @@
npm start

103
adapter/gdrive,.js Normal file
View File

@@ -0,0 +1,103 @@
require('dotenv').config({ path: `${__dirname}/../.env` });
const { google } = require('googleapis');
const path = require('path');
const fs = require('fs');
//const clientEmail = require(`${__dirname}/../chatbot-account.json`);
/**
* La funcion 'generatePublicUrl' genera un error muy menor al enviar el 'requestBody'
* siempre y cuando necesites que el acceso sea restringido y solo ciertos usuarios puedan acceder.
* Esto se logra con la combinacion requerida: 'reader', 'user' y 'emailAddress':
* requestBody: {
* role: 'reader',
* type: 'user',
* emailAddress: usuario@gmail.com,
* },
* Segun la documentacion https://developers.google.com/drive/api/v3/reference/permissions/create#request-body,
* los datos se envian correctamente, pero la respuesta del API regresa este error:
* Bad Request. User message: "You cannot share this item because it has been flagged as inappropriate."
* Al parecer, es un error conocido en stackoverflow.com entre varios usuarios del API.
*/
if (process.env.DATABASE === 'dialogflow') {
/**
* Debes de tener tu archivo con el nombre "chatbot-account.json" en la raíz del proyecto
*/
const KEYFILEPATH = path.join(`${__dirname}/../chatbot-account.json`);
const SCOPES = ['https://www.googleapis.com/auth/drive'];
const auth = new google.auth.GoogleAuth({
keyFile: KEYFILEPATH,
scopes: SCOPES,
});
const drive = google.drive({
version: 'v3',
auth,
});
const uploadSingleFile = async (fileName, filePath) => {
const folderId = process.env.GDRIVE_FOLDER_ID;
const { data: { id, name } = {} } = await drive.files.create({
resource: {
name: fileName,
parents: [folderId],
},
media: {
mimeType: 'image/jpg',
body: fs.createReadStream(filePath),
},
fields: 'id,name',
});
generatePublicUrl(id).then(() => {
console.log(`Se generó enlace https://drive.google.com/open?id=${id} para el archivo ${name}`);
});
return `https://drive.google.com/open?id=${id}`
};
const scanFolderForFiles = async (folderPath) => {
const folder = await fs.promises.opendir(folderPath);
for await (const dirent of folder) {
if (dirent.isFile() && dirent.name.endsWith('.jpeg')) {
await uploadSingleFile(dirent.name, path.join(folderPath, dirent.name));
await fs.promises.rm(filePath);
}
}
};
async function generatePublicUrl(id) {
try {
const fileId = id;
await drive.permissions.create({
fileId: fileId,
supportsAllDrives: true,
requestBody: {
role: 'reader',
type: 'domain', // 'anyone' da acceso al publico vía enlace https://drive.google.com...
domain: 'gserviceaccount.com', // Si tu cuenta esta bajo un dominio (usuario@empresa.com) y no bajo gmail.com
allowFileDiscovery: false,
},
});
/*
webViewLink: Ver el archivo en el navegador
webContentLink: Enlace de descarga directa
*/
const result = await drive.files.get({
fileId: fileId,
fields: 'webViewLink, webContentLink',
});
console.log(result.data);
} catch (error) {
//console.log(error.message); // Imprime 'Internal Error', pero aún así genera el enlace
console.error = () => { }; // No muestra el error anterior
}
}
module.exports = { uploadSingleFile, scanFolderForFiles }
} else {
console.log(`Actualmente, la base de datos es:\n\t'DATABASE=${process.env.DATABASE}'\nPara usar Google Drive, cambiar a:\n\t'DATABASE=dialogflow'`);
}

View File

@@ -3,17 +3,114 @@ const { saveMessageJson } = require('./jsonDb')
const { getDataIa } = require('./diaglogflow') const { getDataIa } = require('./diaglogflow')
const stepsInitial = require('../flow/initial.json') const stepsInitial = require('../flow/initial.json')
const stepsReponse = require('../flow/response.json') const stepsReponse = require('../flow/response.json')
const { isUndefined } = require('util');
var msjsRecibidos = [];
var ultimoStep; //MOD by CHV -
var pasoAnterior = []; //MOD by CHV - Para guardar el paso anterior de cada número.
var pasoRequerido; //MOD by CHV -
var vamosA = ""; //MOD by CHV -
var VA = ""; //MOD by CHV -
var elNum; //MOD by CHV -
var cumplePasoPrevio; //MOD by CHV -
const resps = require('../flow/response.json'); //MOD by CHV - Agregamos para traer las respuestas.
const { appendFile } = require('fs')
const get = (message, num) => new Promise((resolve, reject) => { //MOD by CHV - Agregamos parametro "num" para recibir el número de "app.js"
// console.log(num)
elNum = num //MOD by CHV -
if(siguientePaso.find(k => k.numero.includes(elNum))){
console.log("siguientePaso="+siguientePaso.find(k => k.numero.includes(elNum))["numero"], siguientePaso.find(k => k.numero.includes(elNum))["va"])
// ultimoStep = siguientePaso.find(k => k.numero.includes(elNum))["va"]
pasoAnterior[elNum] = siguientePaso.find(k => k.numero.includes(elNum))["va"] //Asignamos pasoAnterior al número.
siguientePaso.splice(siguientePaso.indexOf(elNum), 1)
console.log("******************** "+siguientePaso.find(k => k.numero.includes(elNum)))
}
if(siguientePaso.length>1){console.log(siguientePaso[1]["numero"], siguientePaso[1]["va"])}
const get = (message) => new Promise((resolve, reject) => {
/** /**
* Si no estas usando un gesto de base de datos * Si no estas usando un gesto de base de datos
*/ */
if (process.env.DATABASE === 'none') { if (process.env.DATABASE === 'none') {
const { key } = stepsInitial.find(k => k.keywords.includes(message)) || { key: null } var { key } = stepsInitial.find(k => k.keywords.includes(message)) || { key: null }
const response = key || null
resolve(response) /* ############################################### * REGEXP * ####################################################
Si queremos usar RegExp, en los "keywords" de inital.json, en lugar de un arreglo usamos un string (quitamos los [])
y en él usamos "*" para significar cualquier texto y "|" para significar "OR", esto nos permite ser mas flexibles
con los "keywords", por ejemplo, si queremos que el mensaje pueda decir:
"Hola quiero info del paquete" o "Requiero mas informacion"
ponemos "*info*" y la regla se va a disparar porque los dos contienen "info", o si queremos que se dispare con:
"Quiero info del paquete numero 3" o "Me gusto el paquete de Angular"
ponemos "paquete*3|paquete*angular" y la regla se dispara porque contiene "paquete" Y "3" -O- "paquete" Y "angular".
#####################################################################################################################
*/
var {keywords} = stepsInitial.find(k => k.key.includes(key)) || { keywords: null }
if(!Array.isArray(keywords)){key=null;}//Si "keywords" no es arreglo entonces ponemos "key" en null y usamos REGEXP para buscar reglas.
if(key == null && message.length > 0){
console.log("======= KEY ES NULO USAMOS REGEXP =======");
for (i=0; i<stepsInitial.length;i++){
// console.log(i, stepsInitial[i].keywords, message.match(stepsInitial[i].keywords.toString().replaceAll("*",".*")));
if(!Array.isArray(stepsInitial[i].keywords)){// Si "Keywords" NO es arreglo entonces ...
x = null;
console.log("KEY=|" + stepsInitial[i].key.toString() + "|" )
// if(resps[stepsInitial[i].key.toString()].pasoRequerido != undefined){pr = resps[stepsInitial[i].key].pasoRequerido};
// console.log(resps[stepsInitial[i].key.toString()].pasoRequerido== ultimoStep)
console.log("Esta Key=" + stepsInitial[i].key.toString() + " - pasoReq=" + resps[stepsInitial[i].key.toString()].pasoRequerido + " - PasoAnt=" + ultimoStep+"|"+pasoAnterior[elNum])
if(resps[stepsInitial[i].key.toString()].pasoRequerido == undefined || resps[stepsInitial[i].key.toString()].pasoRequerido == pasoAnterior[elNum]){
var tempKeyword = "";
if (stepsInitial[i].keywords == "%solo_correos%"){
console.log("solo_correos")
tempKeyword = "[a-zA-Z0-9]+[_a-zA-Z0-9\.-]*[a-zA-Z0-9]+@[a-zA-Z0-9-]+(?:\.[a-zA-Z0-9-]+[\.][a-zA-Z]{2,12})"}
else{tempKeyword = stepsInitial[i].keywords.toString().replaceAll("*",".*")}
x = message.match(tempKeyword);
// console.log("Keywords="+stepsInitial[i].keywords + " - key=" + stepsInitial[i].key + " - pasoReq=" + resps[stepsInitial[i].key].pasoRequerido + " - PasoAnt=" + ultimoStep)
// console.log("x:"+x+" - ultimoStep="+ultimoStep+" - pasoReq="+resps[stepsInitial[i].key].pasoRequerido);
// console.log(resps[stepsInitial[i].key].replyMessage.toString())
if (x != null){
key = stepsInitial[i].key;
if(resps[stepsInitial[i].key].pasoRequerido != null && resps[stepsInitial[i].key].pasoRequerido != pasoAnterior[elNum]){key=null;}
// console.log("KEY="+key+" - X="+x);
if(resps[stepsInitial[i].key].replyMessage.toString().search("/URL") > -1){
console.log("**************** HAY URL ****************")
} }
break;
}
else{x = null;}
} else
{ console.log("NO CUMPLE PASO REQ");
// console.log("pasoReq=" + resps[stepsInitial[i].key.toString()].pasoRequerido + " - PasoAnt=" + ultimoStep)
}
}
}
// console.log("<<<<<<<<< "+key);
// cumplePasoRequerido(key)
// ultimoPaso = pasoRequerido;
// ultimoStep = key;
}
const response = key || null
// if(key != null){remplazos(resps[key].replyMessage.join(''));}
if(resps[key]!=undefined){VA = resps[key].goto}else{VA=null}
cumplePasoRequerido(key);
vamosA = VA;
// console.log(elNum)
if(vamosA != "" && vamosA != undefined && cumplePasoPrevio == true){
console.log("ASIGNAMOS VAMOSA = " + vamosA);
pasoAnterior[elNum] = vamosA;
}
// console.log("ULTIMOSTEP="+ultimoStep)
vamosA = "";
// console.log("MESSAGE: "+message);
// console.log("KEY: "+key);
// console.log("RESPONSE: "+response);
if(cumplePasoPrevio) {resolve(response);}
}
/** /**
* Si usas MYSQL * Si usas MYSQL
*/ */
@@ -25,7 +122,6 @@ const get = (message) => new Promise((resolve, reject) => {
}) })
const reply = (step) => new Promise((resolve, reject) => { const reply = (step) => new Promise((resolve, reject) => {
/** /**
* Si no estas usando un gesto de base de datos * Si no estas usando un gesto de base de datos
@@ -52,13 +148,13 @@ const reply = (step) => new Promise((resolve, reject) => {
} }
}) })
const getIA = (message, sessionId) => new Promise((resolve, reject) => { const getIA = (message) => new Promise((resolve, reject) => {
/** /**
* Si usas dialogflow * Si usas dialogflow
*/ */
if (process.env.DATABASE === 'dialogflow') { if (process.env.DATABASE === 'dialogflow') {
let resData = { replyMessage: '', media: null, trigger: null } let resData = { replyMessage: '', media: null, trigger: null }
getDataIa(message, sessionId, (dt) => { getDataIa(message,(dt) => {
resData = { ...resData, ...dt } resData = { ...resData, ...dt }
resolve(resData) resolve(resData)
}) })
@@ -73,13 +169,14 @@ const getIA = (message, sessionId) => new Promise((resolve, reject) => {
* @param {*} number * @param {*} number
* @returns * @returns
*/ */
const saveMessage = ( message, trigger, number ) => new Promise( async (resolve, reject) => { const saveMessage = ( message, trigger, number, regla ) => new Promise( async (resolve, reject) => { //MOD by CHV - Agregamos el partametro "regla" para poder guardarlo en "chats/numero.json"
switch ( process.env.DATABASE ) { switch ( process.env.DATABASE ) {
case 'mysql': case 'mysql':
resolve( await saveMessageMysql( message, trigger, number ) ) resolve( await saveMessageMysql( message, trigger, number ) )
break; break;
case 'none': case 'none':
resolve( await saveMessageJson( message, trigger, number ) ) resolve( await saveMessageJson( message, trigger, number, regla) ) //MOD by CHV - Agregamos el paranetro "regla"
// console.log("REGLA DESDE APP.JS="+regla)
break; break;
default: default:
resolve(true) resolve(true)
@@ -87,4 +184,163 @@ const saveMessage = ( message, trigger, number ) => new Promise( async (resolve
} }
}) })
module.exports = { get, reply, getIA, saveMessage } module.exports = { get, reply, getIA, saveMessage, remplazos, stepsInitial } //MOD by CHV - Agregamos "remplazos" y "stepsInitial" para usarlos en "apps.js"
/**
* Reemplaza texto en la respuesta con variables predefinidas.
*/
function remplazos(elTexto, extraInfo){
if(elTexto == null){elTexto = '';}
laLista = elTexto.toString().split(' ');
// console.log(laLista);
// console.log('============= remplazos ============');
for (var i = 0; i < laLista.length; i++) {
// console.log('Revisamos: '+laLista[i]);
if (laLista[i].search('%dia_semana%')>-1){//Remplaza con el dia de hoy.
var dia = new Date().getDay();
if(dia==0){diaSemana='domingo';}
else if(dia==1){diaSemana='lunes';}
else if(dia==2){diaSemana='martes';}
else if(dia==3){diaSemana='miercoles';}
else if(dia==4){diaSemana='jueves';}
else if(dia==5){diaSemana='viernes';}
else {diaSemana='sábado';}
elTexto = elTexto.replace('%dia_semana%', diaSemana);
}
if (laLista[i].search('%saludo%')>-1){//Remplaza con "Buenos dias, tardes o noches" dependiendo de la hora.
var hora = new Date().getHours()
if(hora>0 && hora < 12){saludo='Buenos días';}
else if(hora>11 && hora < 19){saludo='Buenas tardes';}
else {saludo='Buenas noches';}
elTexto = elTexto.toString().replace('%saludo%', saludo);
}
if (laLista[i].search('%hora24%')>-1){//Remplaza con la hora a 24 hrs.
var hora = new Date().getHours();
if (hora.toString().length < 2){hora = "0" + hora;}
elTexto = elTexto.toString().replace('%hora24%', hora);
}
if (laLista[i].search('%hora12%')>-1){//Remplaza con la hora a 12 hrs.
var hora = new Date().getHours();
var ampm = hora >= 12 ? 'pm' : 'am';
hora = hora % 12;
hora = hora ? hora : 12; // the hour '0' should be '12'
if (hora.toString().length < 2){hora = "0" + hora;}
elTexto = elTexto.toString().replace('%hora12%', hora);
}
if (laLista[i].search('%minutos%')>-1){//Remplaza con los minutos de la hora actual.
var mins = new Date().getMinutes();
if (mins.toString().length < 2){mins = "0" + mins;}
elTexto = elTexto.toString().replace('%minutos%', mins);
}
if (laLista[i].search('%ampm%')>-1){//Remplaza con am o pm.
var hours = new Date().getHours();
var ampm = hours >= 12 ? 'pm' : 'am';
elTexto = elTexto.toString().replace('%ampm%', ampm);
}
if (laLista[i].search('%rnd_')>-1){//Remplaza con opción al azar dentro de las especificadas.
var inicio = laLista[i].search('%rnd_');
var final = laLista[i].indexOf("%", inicio+1);
// console.log(inicio, final);
var subStr = laLista[i].substring(inicio, final+1);
// console.log("El substring="+subStr);
var partes = subStr.toString().split('_');
if(partes.length > 1){
var opciones = partes[1].toString().substring(0,partes[1].toString().length-1).split(",");
var elRand = Math.floor(Math.random() * (opciones.length));
if(elRand == opciones.length){elRand = elRand - 1;}
// console.log(opciones.length, elRand, opciones[elRand]);
elTexto = elTexto.toString().replace(subStr, opciones[elRand]);
}
else{
elTexto = elTexto.toString().replace(subStr, "");
}
}
if(laLista[i].search('%msjant_')>-1){//Remplaza con el mensaje anterior especificado.
var histlMsjs = {};
console.log("entramos a msjant")
// var hayHistorial = (chkFile(`${__dirname}/chats/`+from+".json"));
if(chkFile(`${__dirname}/../chats/`+elNum+".json")){
let rawdata = fs.readFileSync(`./chats/${elNum}.json`);
let elHistorial0 = JSON.parse(rawdata);
elHistorial = elHistorial0["messages"];
var inicio = laLista[i].search('%msjant_');
var final = laLista[i].indexOf("%", inicio+1);
var subStr = laLista[i].substring(inicio, final+1);
console.log("Substr = |" + subStr + "|");
var partes = subStr.toString().split('_');
if(partes.length > 1){
console.log("Partes[1] = |" + partes[1] + "|");
let posicion0 = partes[1].substring(0, partes[1].length-1)
console.log("Posicion0 = |" + posicion0 + "|");
posicion = ((posicion0*1) + 1);
console.log("Posicion = " + posicion);
console.log( elHistorial.length );
console.log((elHistorial.length*1)-posicion);
console.log("Mensaje="+elHistorial[elHistorial.length - posicion]["message"])
elTexto = elTexto.toString().replace(subStr, elHistorial[elHistorial.length - posicion]["message"]);
}
// histlMsjs = elHistorial["messages"];
// totalMsjs = histlMsjs.length-1;
// ultimoMensaje = histlMsjs[histlMsjs.length-1];
// let mensajeAnterior = elHistorial["messages"][totalMsjs-1];
// console.log("Mensajes:"+totalMsjs+", Ultimo:"+JSON.stringify(ultimoMensaje));
// console.log("Anterior:"+JSON.stringify(mensajeAnterior));
}
// return histlMsjs;
}
if (laLista[i].search('%nombre%')>-1){//Remplaza con el nombre del remitente.
if(typeof extraInfo !== undefined){
const {theMsg} = extraInfo;
if(theMsg['_data']['notifyName'] !== undefined){
elTexto = elTexto.toString().replace('%nombre%', theMsg['_data']['notifyName']);
}
}
}
if (laLista[i].search('%primer_nombre%')>-1){//Remplaza con el nombre del remitente.
if(typeof extraInfo !== undefined){
const {theMsg} = extraInfo;
if(theMsg['_data']['notifyName'] !== undefined){
elTexto = elTexto.toString().replace('%primer_nombre%', theMsg['_data']['notifyName'].split(' ')[0]);
}
}
}
}
// console.log("EL TEXTO="+elTexto);
return elTexto
}
/**
* Revisa si la regla especificada depende (es submenu) de otra regla, y cambia la variable "cumplePasoPrevio" a verdadero o falso.
*/
function cumplePasoRequerido(step){
//Traemos las respuestas para obtener el "pasoRequerido".
if(resps[step]!=undefined){pasoRequerido=resps[step].pasoRequerido}else{pasoRequerido=null}
if((pasoRequerido != null && pasoRequerido == ultimoStep)){
// console.log("REQUIERE PASO PREVIO Y CUMPLE");
cumplePasoPrevio = true;
}
else if((pasoRequerido != null && pasoRequerido != pasoAnterior[elNum])){
// console.log("REQUIERE PASO PREVIO Y NO LO CUMPLE");
cumplePasoPrevio = false;
}
else{
// console.log("NO REQUIERE PASO PREVIO")
cumplePasoPrevio = true;
}
pasoAnterior[elNum] = step
ultimoPaso = pasoRequerido;
}
const fs = require('fs');
function chkFile(theFile){ //MOD by CHV - Agregamos para revisar que exista el archivo "chats/numero.json"
if (fs.existsSync(theFile)) {
// console.log("Si existe el archivo "+ theFile);
var h = true;
}
else{
// console.log("No existe el archivo "+ theFile);
var h = false;
}
return h;
}

View File

@@ -1,14 +1,14 @@
const Path = require('path') const Path = require('path')
const StormDB = require("stormdb"); const StormDB = require("stormdb");
const date = new Date().toISOString(); const date = new Date().toISOString();
const saveMessageJson = (message, trigger, number) => new Promise( async(resolve,reject) =>{ const saveMessageJson = (message, trigger, number, regla) => new Promise( async(resolve,reject) =>{
try { try {
const engine = new StormDB.localFileEngine( Path.join(__dirname, `/../chats/${number}.json`) ); const engine = new StormDB.localFileEngine( Path.join(__dirname, `/../chats/${number}.json`) );
const db = new StormDB(engine); const db = new StormDB(engine);
// set default db value if db is empty // set default db value if db is empty
db.default({ messages: [] }); db.default({ messages: [] });
// add new users entry // add new users entry
db.get("messages").push({ message, date, trigger }); db.get("messages").push({ message, date, trigger, regla});
db.save(); db.save();
resolve('Saved') resolve('Saved')
} catch (error) { } catch (error) {

395
app.js
View File

@@ -4,45 +4,67 @@
require('dotenv').config() require('dotenv').config()
const fs = require('fs'); const fs = require('fs');
const express = require('express'); const express = require('express');
global.siguientePaso = [{"numero":"1", "va":"XXX"}]; //MOD by CHV - Agregamos para pasar el VAMOSA a "index.js"
const cors = require('cors') const cors = require('cors')
const axios = require('axios').default;//MOD by CHV - Agregamos para el get del "/URL"
const qrcode = require('qrcode-terminal'); const qrcode = require('qrcode-terminal');
const { Client, LocalAuth } = require('whatsapp-web.js'); const { Client, LocalAuth, Buttons, List } = require('whatsapp-web.js');
const mysqlConnection = require('./config/mysql') const mysqlConnection = require('./config/mysql')
const { middlewareClient } = require('./middleware/client') const { middlewareClient } = require('./middleware/client')
const { generateImage, cleanNumber, checkEnvFile, createClient, isValidNumber } = require('./controllers/handle') const { generateImage, cleanNumber, checkEnvFile, createClient, isValidNumber } = require('./controllers/handle')
const { connectionReady, connectionLost } = require('./controllers/connection') const { connectionReady, connectionLost } = require('./controllers/connection')
const { saveMedia, saveMediaToGoogleDrive } = require('./controllers/save') const { saveMedia, saveMediaToGoogleDrive } = require('./controllers/save')
const { getMessages, responseMessages, bothResponse, waitFor } = require('./controllers/flows') const { getMessages, responseMessages, bothResponse, waitFor } = require('./controllers/flows')
const { sendMedia, sendMessage, lastTrigger, sendMessageButton, readChat } = require('./controllers/send') const { sendMedia, sendMessage, lastTrigger, sendMessageButton, sendMessageList, readChat } = require('./controllers/send');
const { remplazos, stepsInitial} = require('./adapter/index');//MOD by CHV - Agregamos para utilizar remplazos y stepsInitial
const { isUndefined } = require('util');
const { isSet } = require('util/types');
const { Console } = require('console');
const { ClientRequest } = require('http');
const app = express(); const app = express();
app.use(cors()) app.use(cors())
app.use(express.json()) app.use(express.json())
const MULTI_DEVICE = process.env.MULTI_DEVICE || 'true'; const MULTI_DEVICE = process.env.MULTI_DEVICE || 'true';
const server = require('http').Server(app) const server = require('http').Server(app)
const port = process.env.PORT || 3000 const port = process.env.PORT || 3000
var client; var client;
var dialogflowFilter = false; var dialogflowFilter = false;
var totalMsjs; //MOD by CHV -
var vamosA = ""; //MOD by CHV -
var newBody; //MOD by CHV -
var nuevaRespuesta; //MOD by CHV - Se agrego para los remplazos
app.use('/', require('./routes/web')) app.use('/', require('./routes/web'))
/** /**
* Escuchamos cuando entre un mensaje * Escuchamos cuando entre un mensaje
*/ */
const listenMessage = () => client.on('message', async msg => { const listenMessage = () => client.on('message', async msg => {
const { from, body, hasMedia } = msg; const { from, body, hasMedia } = msg;
// console.log("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++");
console.log("+++++++++++++++++++++++++++++++++++++ INICIO +++++++++++++++++++++++++++++++++++++++");
// console.log("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++");
if (!isValidNumber(from)) { client.theMsg = msg;
console.log("HORA:"+new Date().toLocaleTimeString()+" FROM:"+from+", BODY:"+body+", HASMEDIA:"+hasMedia);
newBody = removeDiacritics(body) //MOD by CHV - Agregamos para quitar acentos
// newBody = remplazos(newBody);
vamosA = "";
if(!isValidNumber(from)){
return return
} }
// Este bug lo reporto Lucas Aldeco Brescia para evitar que se publiquen estados // Este bug lo reporto Lucas Aldeco Brescia para evitar que se publiquen estados
if (from === 'status@broadcast') { if (from === 'status@broadcast') {
return return
} }
message = body.toLowerCase(); message = newBody.toLowerCase();
console.log('BODY', message)
const number = cleanNumber(from) const number = cleanNumber(from)
await readChat(number, message)
var { key } = stepsInitial.find(k => k.keywords.includes(message)) || { key: null }//MOD by CHV - Se agrega para obtener KEY
await readChat(number, message, null , key) //MOD by CHV - Agregamos key/regla para guardarla en "chats/numero.json"
/** /**
* Guardamos el archivo multimedia que envia * Guardamos el archivo multimedia que envia
@@ -57,7 +79,6 @@ const listenMessage = () => client.on('message', async msg => {
*/ */
if (process.env.DATABASE === 'dialogflow') { if (process.env.DATABASE === 'dialogflow') {
if (process.env.DIALOGFLOW_MEDIA_FOR_SLOT_FILLING === 'true' && dialogflowFilter) { if (process.env.DIALOGFLOW_MEDIA_FOR_SLOT_FILLING === 'true' && dialogflowFilter) {
waitFor(_ => hasMedia, 30000) waitFor(_ => hasMedia, 30000)
.then(async _ => { .then(async _ => {
@@ -71,7 +92,6 @@ const listenMessage = () => client.on('message', async msg => {
}); });
dialogflowFilter = false; dialogflowFilter = false;
} }
if (!message.length) return; if (!message.length) return;
const response = await bothResponse(message.substring(256, -1), number); const response = await bothResponse(message.substring(256, -1), number);
await sendMessage(client, from, response.replyMessage); await sendMessage(client, from, response.replyMessage);
@@ -85,6 +105,47 @@ const listenMessage = () => client.on('message', async msg => {
return return
} }
if(body=='/listas'){
const productList = new List(
"Here's our list of products at 50% off",
"View all products",
[
{
title: "Products list",
rows: [
{ id: "apple", title: "Apple" },
{ id: "mango", title: "Mango" },
{ id: "banana", title: "Banana" },
],
},
],
"Please select a product"
);
console.log('##################################################################################################')
// console.log(from, lista)
// let sections = [{title:'sectionTitle',rows:[{id:'ListItem1', title: 'title1'},{id:'ListItem2', title:'title2'}]}];
// let lista = new List('List body','btnText',sections,'Title','footer');
console.log("****************** productList ******************")
console.log(productList)
client.sendMessage(from, productList); //cliente.sendMessage recibe el arreglo SIN nombres (solo las secciones los necesitan)
// client.sendMessage('5215527049036@c.us', productList);
// client.sendMessage('5215554192439@c.us', productList);
// await sendMessageList(client, '5215545815654@c.us', null, lista); //sendMessageList recibe el arreglo CON nombres, como viene del response.json
// await sendMessageList(client, '5215527049036@c.us', null, lista);
// await sendMessageList(client, '5215554192439@c.us', null, lista);
// client.sendMessage(from, lista);
}
/**
* PRUEBA BOTONES NUEVOS
*/
// if(body=="579"){
// const buttons_reply = new Buttons("Por favor vuelve a intentar, mandando *SOLO* la palabra *gallina* con una diagonal al principio 👇🏽 \n\n */gallina*\n ", [{body: "/gallina", id: 'errorGallina'}], 'Error', 'O haz clic en el siguiente botón') // Reply button
// for (const component of [buttons_reply]) await client.sendMessage(from, component);
// }
/** /**
* Ver si viene de un paso anterior * Ver si viene de un paso anterior
* Aqui podemos ir agregando más pasos * Aqui podemos ir agregando más pasos
@@ -92,57 +153,148 @@ const listenMessage = () => client.on('message', async msg => {
*/ */
const lastStep = await lastTrigger(from) || null; const lastStep = await lastTrigger(from) || null;
// console.log("LAST STEP="+lastStep+", FROM:"+from);
if (lastStep) { if (lastStep) {
const response = await responseMessages(lastStep) const response = await responseMessages(lastStep)
await sendMessage(client, from, response.replyMessage); console.log("CLIENT="+client+", FROM:"+from+", REPLYMESSAGE:"+response.replyMessage);
await sendMessage(client, from, response.replyMessage, lastStep);
} }
/** /**
* Respondemos al primero paso si encuentra palabras clave * Respondemos al primero paso si encuentra palabras clave
*/ */
const step = await getMessages(message); // const step = await getMessages(message, );
// console.log("STEP - "+step+"|"+message);
// console.log("****** STEP="+step);
// console.log("****** MESSAGE:"+message);
const step = await getMessages(message, from);
if (step) { if (step) {
const response = await responseMessages(step); const response = await responseMessages(step);
// console.log("URL:"+nuevaRespuesta);
// console.log("HAY URL?? : "+nuevaRespuesta.search("/URL"));
var resps = require('./flow/response.json');
nuevaRespuesta = remplazos(resps[step].replyMessage.join(''), client);
var pasoRequerido = resps[step].pasoRequerido;
// var hayRequest = false;
// if(hayRequest==false && nuevaRespuesta.search("/URL")>-1){console.log("Paramos flujo para que no mande el mensaje '/URL'."); return;}//Si el trigger es desbloqueo ya no hace nada mas.
/** /**
* Si quieres enviar botones * 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) { if (!response.delay && response.media) {
sendMedia(client, from, response.media); // console.log("++++++++++++++++++++++++++++ SEND MEDIA NO DELAY +++++++++++++++++++++++++++++++++++");
sendMedia(client, from, response.media, response.trigger);
} }
if (response.delay && response.media) { if (response.delay && response.media) {
setTimeout(() => { setTimeout(() => {
sendMedia(client, from, response.media); // console.log("++++++++++++++++++++++++++++ SEND MEDIA AND DELAY +++++++++++++++++++++++++++++++++++");
sendMedia(client, from, response.media, response.trigger);
}, response.delay) }, response.delay)
} }
if (response.delay){
// console.log("+++++++++++++++++++ SENDING MSG WITH DELAY ("+response.delay+") +++++++++++++++++");
setTimeout(() => {
sendMessage(client, from, nuevaRespuesta, response.trigger, step);
// console.log(" ************* Msg with delay SENT ****************")
}, response.delay)
}
else{
await sendMessage(client, from, nuevaRespuesta, response.trigger, step);
}
if(response.hasOwnProperty('actions')){
const { actions } = response;
// console.log("++++++++++++++++++++++++++++ SEND MESG BUTTON/LIST +++++++++++++++++++++++++++++++++++");
if(actions['sections'] === undefined){ //Botones
console.log("Botones")
await sendMessageButton(client, from, null, actions);
}
else{ //Listas
console.log("Listas")
// console.log(actions)
await sendMessageList(client, from, null, actions);
}
}
return return
} }
/**
* Regresa el mensaje enviado, con los remplazos procesados.
*/
if(message.search('/rpt') > -1){
newBody = remplazos(newBody, client);
newBody = newBody.replace("/rpt ", "");
client.sendMessage(from, newBody);
return
}
/*
============================================================================
========================== ENVIO MASIVO TEST ===========================
============================================================================
*/
if(message=='/spam'){
const masivo = require('./spam.json')
var saludo;
var caritas;
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
async function retardo() {
for (i=0;i<masivo.length;i++) {
console.log(masivo[i].numero+"@c.us");
var rnd = getRandomInt(1,7); // Random entre 1 y 6 segundos.
if(rnd==1||rnd==4){saludo = "Hola ";}
else if(rnd==2||rnd==5){saludo = "Saludos ";}
else {saludo = "%saludo% ";}
if(rnd==1){caritas = "👨🏻‍🦰👩🏻‍🦰";}
else if(rnd==2){caritas = "👩🏻‍🦰👨🏻‍🦰";}
else if(rnd==3){caritas = "🧔🏽👧🏽";}
else if(rnd==4){caritas = "👧🏽🧔🏽";}
else if(rnd==5){caritas = "👩🏻‍🦰🧔🏽";}
else if(rnd==6){caritas = "🧔🏽👩🏻‍🦰";}
if(i % 15 === 0){console.log("******** VAN 15, HACEMOS PAUSA DE 10 SEGUNDOS ********"); await sleep(10000);} //
console.log(`============= Mandamos el mensaje ${i} ==============`);
var elTextoDelMensaje = caritas + " *" + saludo + "amigo tendero* ❗❗👋🏻\n🕊 *GUNA* trae para ti dinámicas digitales, con las que podrás participar para ganar increíbles premios. 🏆💸💰\nSigue los siguientes pasos: 😃\n*1.* 📲Sigue la página de Yo Soy Guna en Facebook en la siguiente liga ➡️ https://www.facebook.com/yosoyguna\n*2.* 👉🏻Es importante des click en el botón Me Gusta 👍\n*3.* 🧐Sigue la dinámica que publicaremos , subiendo tu foto 📸 con los siguientes #yosoyguna #gunatenderos #gunachampions\n*4.* 🥳🎉En esta misma página , podrás ver publicados los ganadores🏅 y el tiempo en que serán elegidos. 💲 Además de tener acceso a increíbles promociones 🤑";
sendMedia(client, masivo[i].numero+"@c.us", "envioMasivoGuna.jpg");
await sleep(500);
client.sendMessage(masivo[i].numero+"@c.us", remplazos(elTextoDelMensaje, client));
// client.sendMessage(masivo[i].numero+"@c.us", "Este es un mensaje de prueba para *"+masivo[i].numero+"*, HORA:*"+new Date().toLocaleTimeString()+"*");
console.log(`Esperamos ${rnd} segundos...`);
await sleep(rnd*1000);
}
console.log('Done');
}
retardo();
}
function getRandomInt(min, max) {
min = Math.ceil(min);
max = Math.floor(max);
return Math.floor(Math.random() * (max - min) + min); // The maximum is exclusive and the minimum is inclusive
}
//Si quieres tener un mensaje por defecto //Si quieres tener un mensaje por defecto
if (process.env.DEFAULT_MESSAGE === 'true') { if (process.env.DEFAULT_MESSAGE === 'true') {
const response = await responseMessages('DEFAULT') const response = await responseMessages('DEFAULT')
await sendMessage(client, from, response.replyMessage, response.trigger); await sendMessage(client, from, response.replyMessage, response.trigger);
/** /**
* Si quieres enviar botones * Si quieres enviar botones
*/ */
if (response.hasOwnProperty('actions')) { if(response.hasOwnProperty('actions')){
const { actions } = response; const { actions } = response;
if(actions['sections'] === undefined){ //Botones
console.log("Botones")
await sendMessageButton(client, from, null, actions); await sendMessageButton(client, from, null, actions);
}
else{ //Listas
console.log("Listas")
await sendMessageList(client, from, null, actions);
}
} }
return return
} }
}); });
/** /**
* Este evento es necesario para el filtro de Dialogflow * Este evento es necesario para el filtro de Dialogflow
@@ -165,45 +317,194 @@ const listenMessageFromBot = () => client.on('message_create', async botMsg => {
} }
}); });
client = new Client({ client = new Client({
authStrategy: new LocalAuth(), authStrategy: new LocalAuth(),
puppeteer: { headless: true } puppeteer: { headless: true, args: ['--no-sandbox','--disable-setuid-sandbox'] }
}); });
client.on('qr', qr => generateImage(qr, () => { client.on('qr', qr => generateImage(qr, () => {
qrcode.generate(qr, { small: true }); qrcode.generate(qr, { small: true });
console.log(`Ver QR http://localhost:${port}/qr`) console.log(`Ver QR http://localhost:${port}/qr`)
socketEvents.sendQR(qr) socketEvents.sendQR(qr)
})) }))
client.on('ready', (a) => { client.on('ready', (a) => {
connectionReady() connectionReady()
listenMessage() listenMessage()
listenMessageFromBot() listenMessageFromBot()
// socketEvents.sendStatus(client) // socketEvents.sendStatus(client)
}); });
client.on('auth_failure', (e) => { client.on('auth_failure', (e) => {
// console.log(e) // console.log(e)
// connectionLost() // connectionLost()
}); });
client.on('authenticated', () => { client.on('authenticated', () => {
console.log('AUTHENTICATED'); console.log('AUTHENTICATED');
}); });
client.initialize(); client.initialize();
/** /**
* Verificamos si tienes un gesto de db * Verificamos si tienes un gesto de db
*/ */
if (process.env.DATABASE === 'mysql') { if (process.env.DATABASE === 'mysql') {
mysqlConnection.connect() mysqlConnection.connect()
}
server.listen(port, () => {
console.log(`El server esta listo en el puerto ${port}`);
})
checkEnvFile();
function chkFile(theFile){ //MOD by CHV - Agregamos para revisar que exista el archivo "chats/numero.json"
if (fs.existsSync(theFile)) {
// console.log("Si existe el archivo "+ theFile);
var h = true;
}
else{
// console.log("No existe el archivo "+ theFile);
var h = false;
}
return h;
} }
server.listen(port, () => { /**
console.log(`El server esta listo por el puerto ${port}`); * Regresa el historial de mensajes del número especificado del directorio "chats".
}) */
checkEnvFile(); function traeMensajes(from){ //MOD by CHV - Agregamos para traer el historial de mensajes
var histlMsjs = {};
var hayHistorial = (chkFile(`${__dirname}/chats/`+from+".json"));
if(hayHistorial){
let rawdata = fs.readFileSync(`./chats/${from}.json`);
let elHistorial = JSON.parse(rawdata);
histlMsjs = elHistorial["messages"];
// totalMsjs = histlMsjs.length-1;
ultimoMensaje = histlMsjs[histlMsjs.length-1];
// let mensajeAnterior = elHistorial["messages"][totalMsjs-1];
// console.log("Mensajes:"+totalMsjs+", Ultimo:"+JSON.stringify(ultimoMensaje));
// console.log("Anterior:"+JSON.stringify(mensajeAnterior));
}
return histlMsjs;
}
/*
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
var defaultDiacriticsRemovalMap = [ //MOD by CHV - Agregamos para eliminar acentos
{'base':'A', 'letters':'\u0041\u24B6\uFF21\u00C0\u00C1\u00C2\u1EA6\u1EA4\u1EAA\u1EA8\u00C3\u0100\u0102\u1EB0\u1EAE\u1EB4\u1EB2\u0226\u01E0\u00C4\u01DE\u1EA2\u00C5\u01FA\u01CD\u0200\u0202\u1EA0\u1EAC\u1EB6\u1E00\u0104\u023A\u2C6F'},
{'base':'AA','letters':'\uA732'},
{'base':'AE','letters':'\u00C6\u01FC\u01E2'},
{'base':'AO','letters':'\uA734'},
{'base':'AU','letters':'\uA736'},
{'base':'AV','letters':'\uA738\uA73A'},
{'base':'AY','letters':'\uA73C'},
{'base':'B', 'letters':'\u0042\u24B7\uFF22\u1E02\u1E04\u1E06\u0243\u0182\u0181'},
{'base':'C', 'letters':'\u0043\u24B8\uFF23\u0106\u0108\u010A\u010C\u00C7\u1E08\u0187\u023B\uA73E'},
{'base':'D', 'letters':'\u0044\u24B9\uFF24\u1E0A\u010E\u1E0C\u1E10\u1E12\u1E0E\u0110\u018B\u018A\u0189\uA779\u00D0'},
{'base':'DZ','letters':'\u01F1\u01C4'},
{'base':'Dz','letters':'\u01F2\u01C5'},
{'base':'E', 'letters':'\u0045\u24BA\uFF25\u00C8\u00C9\u00CA\u1EC0\u1EBE\u1EC4\u1EC2\u1EBC\u0112\u1E14\u1E16\u0114\u0116\u00CB\u1EBA\u011A\u0204\u0206\u1EB8\u1EC6\u0228\u1E1C\u0118\u1E18\u1E1A\u0190\u018E'},
{'base':'F', 'letters':'\u0046\u24BB\uFF26\u1E1E\u0191\uA77B'},
{'base':'G', 'letters':'\u0047\u24BC\uFF27\u01F4\u011C\u1E20\u011E\u0120\u01E6\u0122\u01E4\u0193\uA7A0\uA77D\uA77E'},
{'base':'H', 'letters':'\u0048\u24BD\uFF28\u0124\u1E22\u1E26\u021E\u1E24\u1E28\u1E2A\u0126\u2C67\u2C75\uA78D'},
{'base':'I', 'letters':'\u0049\u24BE\uFF29\u00CC\u00CD\u00CE\u0128\u012A\u012C\u0130\u00CF\u1E2E\u1EC8\u01CF\u0208\u020A\u1ECA\u012E\u1E2C\u0197'},
{'base':'J', 'letters':'\u004A\u24BF\uFF2A\u0134\u0248'},
{'base':'K', 'letters':'\u004B\u24C0\uFF2B\u1E30\u01E8\u1E32\u0136\u1E34\u0198\u2C69\uA740\uA742\uA744\uA7A2'},
{'base':'L', 'letters':'\u004C\u24C1\uFF2C\u013F\u0139\u013D\u1E36\u1E38\u013B\u1E3C\u1E3A\u0141\u023D\u2C62\u2C60\uA748\uA746\uA780'},
{'base':'LJ','letters':'\u01C7'},
{'base':'Lj','letters':'\u01C8'},
{'base':'M', 'letters':'\u004D\u24C2\uFF2D\u1E3E\u1E40\u1E42\u2C6E\u019C'},
{'base':'N', 'letters':'\u004E\u24C3\uFF2E\u01F8\u0143\u00D1\u1E44\u0147\u1E46\u0145\u1E4A\u1E48\u0220\u019D\uA790\uA7A4'},
{'base':'NJ','letters':'\u01CA'},
{'base':'Nj','letters':'\u01CB'},
{'base':'O', 'letters':'\u004F\u24C4\uFF2F\u00D2\u00D3\u00D4\u1ED2\u1ED0\u1ED6\u1ED4\u00D5\u1E4C\u022C\u1E4E\u014C\u1E50\u1E52\u014E\u022E\u0230\u00D6\u022A\u1ECE\u0150\u01D1\u020C\u020E\u01A0\u1EDC\u1EDA\u1EE0\u1EDE\u1EE2\u1ECC\u1ED8\u01EA\u01EC\u00D8\u01FE\u0186\u019F\uA74A\uA74C'},
{'base':'OI','letters':'\u01A2'},
{'base':'OO','letters':'\uA74E'},
{'base':'OU','letters':'\u0222'},
{'base':'OE','letters':'\u008C\u0152'},
{'base':'oe','letters':'\u009C\u0153'},
{'base':'P', 'letters':'\u0050\u24C5\uFF30\u1E54\u1E56\u01A4\u2C63\uA750\uA752\uA754'},
{'base':'Q', 'letters':'\u0051\u24C6\uFF31\uA756\uA758\u024A'},
{'base':'R', 'letters':'\u0052\u24C7\uFF32\u0154\u1E58\u0158\u0210\u0212\u1E5A\u1E5C\u0156\u1E5E\u024C\u2C64\uA75A\uA7A6\uA782'},
{'base':'S', 'letters':'\u0053\u24C8\uFF33\u1E9E\u015A\u1E64\u015C\u1E60\u0160\u1E66\u1E62\u1E68\u0218\u015E\u2C7E\uA7A8\uA784'},
{'base':'T', 'letters':'\u0054\u24C9\uFF34\u1E6A\u0164\u1E6C\u021A\u0162\u1E70\u1E6E\u0166\u01AC\u01AE\u023E\uA786'},
{'base':'TZ','letters':'\uA728'},
{'base':'U', 'letters':'\u0055\u24CA\uFF35\u00D9\u00DA\u00DB\u0168\u1E78\u016A\u1E7A\u016C\u00DC\u01DB\u01D7\u01D5\u01D9\u1EE6\u016E\u0170\u01D3\u0214\u0216\u01AF\u1EEA\u1EE8\u1EEE\u1EEC\u1EF0\u1EE4\u1E72\u0172\u1E76\u1E74\u0244'},
{'base':'V', 'letters':'\u0056\u24CB\uFF36\u1E7C\u1E7E\u01B2\uA75E\u0245'},
{'base':'VY','letters':'\uA760'},
{'base':'W', 'letters':'\u0057\u24CC\uFF37\u1E80\u1E82\u0174\u1E86\u1E84\u1E88\u2C72'},
{'base':'X', 'letters':'\u0058\u24CD\uFF38\u1E8A\u1E8C'},
{'base':'Y', 'letters':'\u0059\u24CE\uFF39\u1EF2\u00DD\u0176\u1EF8\u0232\u1E8E\u0178\u1EF6\u1EF4\u01B3\u024E\u1EFE'},
{'base':'Z', 'letters':'\u005A\u24CF\uFF3A\u0179\u1E90\u017B\u017D\u1E92\u1E94\u01B5\u0224\u2C7F\u2C6B\uA762'},
{'base':'a', 'letters':'\u0061\u24D0\uFF41\u1E9A\u00E0\u00E1\u00E2\u1EA7\u1EA5\u1EAB\u1EA9\u00E3\u0101\u0103\u1EB1\u1EAF\u1EB5\u1EB3\u0227\u01E1\u00E4\u01DF\u1EA3\u00E5\u01FB\u01CE\u0201\u0203\u1EA1\u1EAD\u1EB7\u1E01\u0105\u2C65\u0250'},
{'base':'aa','letters':'\uA733'},
{'base':'ae','letters':'\u00E6\u01FD\u01E3'},
{'base':'ao','letters':'\uA735'},
{'base':'au','letters':'\uA737'},
{'base':'av','letters':'\uA739\uA73B'},
{'base':'ay','letters':'\uA73D'},
{'base':'b', 'letters':'\u0062\u24D1\uFF42\u1E03\u1E05\u1E07\u0180\u0183\u0253'},
{'base':'c', 'letters':'\u0063\u24D2\uFF43\u0107\u0109\u010B\u010D\u00E7\u1E09\u0188\u023C\uA73F\u2184'},
{'base':'d', 'letters':'\u0064\u24D3\uFF44\u1E0B\u010F\u1E0D\u1E11\u1E13\u1E0F\u0111\u018C\u0256\u0257\uA77A'},
{'base':'dz','letters':'\u01F3\u01C6'},
{'base':'e', 'letters':'\u0065\u24D4\uFF45\u00E8\u00E9\u00EA\u1EC1\u1EBF\u1EC5\u1EC3\u1EBD\u0113\u1E15\u1E17\u0115\u0117\u00EB\u1EBB\u011B\u0205\u0207\u1EB9\u1EC7\u0229\u1E1D\u0119\u1E19\u1E1B\u0247\u025B\u01DD'},
{'base':'f', 'letters':'\u0066\u24D5\uFF46\u1E1F\u0192\uA77C'},
{'base':'g', 'letters':'\u0067\u24D6\uFF47\u01F5\u011D\u1E21\u011F\u0121\u01E7\u0123\u01E5\u0260\uA7A1\u1D79\uA77F'},
{'base':'h', 'letters':'\u0068\u24D7\uFF48\u0125\u1E23\u1E27\u021F\u1E25\u1E29\u1E2B\u1E96\u0127\u2C68\u2C76\u0265'},
{'base':'hv','letters':'\u0195'},
{'base':'i', 'letters':'\u0069\u24D8\uFF49\u00EC\u00ED\u00EE\u0129\u012B\u012D\u00EF\u1E2F\u1EC9\u01D0\u0209\u020B\u1ECB\u012F\u1E2D\u0268\u0131'},
{'base':'j', 'letters':'\u006A\u24D9\uFF4A\u0135\u01F0\u0249'},
{'base':'k', 'letters':'\u006B\u24DA\uFF4B\u1E31\u01E9\u1E33\u0137\u1E35\u0199\u2C6A\uA741\uA743\uA745\uA7A3'},
{'base':'l', 'letters':'\u006C\u24DB\uFF4C\u0140\u013A\u013E\u1E37\u1E39\u013C\u1E3D\u1E3B\u017F\u0142\u019A\u026B\u2C61\uA749\uA781\uA747'},
{'base':'lj','letters':'\u01C9'},
{'base':'m', 'letters':'\u006D\u24DC\uFF4D\u1E3F\u1E41\u1E43\u0271\u026F'},
{'base':'n', 'letters':'\u006E\u24DD\uFF4E\u01F9\u0144\u00F1\u1E45\u0148\u1E47\u0146\u1E4B\u1E49\u019E\u0272\u0149\uA791\uA7A5'},
{'base':'nj','letters':'\u01CC'},
{'base':'o', 'letters':'\u006F\u24DE\uFF4F\u00F2\u00F3\u00F4\u1ED3\u1ED1\u1ED7\u1ED5\u00F5\u1E4D\u022D\u1E4F\u014D\u1E51\u1E53\u014F\u022F\u0231\u00F6\u022B\u1ECF\u0151\u01D2\u020D\u020F\u01A1\u1EDD\u1EDB\u1EE1\u1EDF\u1EE3\u1ECD\u1ED9\u01EB\u01ED\u00F8\u01FF\u0254\uA74B\uA74D\u0275'},
{'base':'oi','letters':'\u01A3'},
{'base':'ou','letters':'\u0223'},
{'base':'oo','letters':'\uA74F'},
{'base':'p','letters':'\u0070\u24DF\uFF50\u1E55\u1E57\u01A5\u1D7D\uA751\uA753\uA755'},
{'base':'q','letters':'\u0071\u24E0\uFF51\u024B\uA757\uA759'},
{'base':'r','letters':'\u0072\u24E1\uFF52\u0155\u1E59\u0159\u0211\u0213\u1E5B\u1E5D\u0157\u1E5F\u024D\u027D\uA75B\uA7A7\uA783'},
{'base':'s','letters':'\u0073\u24E2\uFF53\u00DF\u015B\u1E65\u015D\u1E61\u0161\u1E67\u1E63\u1E69\u0219\u015F\u023F\uA7A9\uA785\u1E9B'},
{'base':'t','letters':'\u0074\u24E3\uFF54\u1E6B\u1E97\u0165\u1E6D\u021B\u0163\u1E71\u1E6F\u0167\u01AD\u0288\u2C66\uA787'},
{'base':'tz','letters':'\uA729'},
{'base':'u','letters': '\u0075\u24E4\uFF55\u00F9\u00FA\u00FB\u0169\u1E79\u016B\u1E7B\u016D\u00FC\u01DC\u01D8\u01D6\u01DA\u1EE7\u016F\u0171\u01D4\u0215\u0217\u01B0\u1EEB\u1EE9\u1EEF\u1EED\u1EF1\u1EE5\u1E73\u0173\u1E77\u1E75\u0289'},
{'base':'v','letters':'\u0076\u24E5\uFF56\u1E7D\u1E7F\u028B\uA75F\u028C'},
{'base':'vy','letters':'\uA761'},
{'base':'w','letters':'\u0077\u24E6\uFF57\u1E81\u1E83\u0175\u1E87\u1E85\u1E98\u1E89\u2C73'},
{'base':'x','letters':'\u0078\u24E7\uFF58\u1E8B\u1E8D'},
{'base':'y','letters':'\u0079\u24E8\uFF59\u1EF3\u00FD\u0177\u1EF9\u0233\u1E8F\u00FF\u1EF7\u1E99\u1EF5\u01B4\u024F\u1EFF'},
{'base':'z','letters':'\u007A\u24E9\uFF5A\u017A\u1E91\u017C\u017E\u1E93\u1E95\u01B6\u0225\u0240\u2C6C\uA763'}
];
var diacriticsMap = {};
for (var i=0; i < defaultDiacriticsRemovalMap .length; i++){
var letters = defaultDiacriticsRemovalMap [i].letters;
for (var j=0; j < letters.length ; j++){
diacriticsMap[letters[j]] = defaultDiacriticsRemovalMap [i].base;
}
}
// "what?" version ... http://jsperf.com/diacritics/12
function removeDiacritics (str) {
return str.replace(/[^\u0000-\u007E]/g, function(a){
return diacriticsMap[a] || a;
});
}
// var paragraph = "L'avantage d'utiliser le lorem ipsum est bien évidemment de pouvoir créer des maquettes ou de remplir un site internet de contenus qui présentent un rendu s'approchant un maximum du rendu final. \n Par défaut lorem ipsum ne contient pas d'accent ni de caractères spéciaux contrairement à la langue française qui en contient beaucoup. C'est sur ce critère que nous proposons une solution avec cet outil qui générant du faux-texte lorem ipsum mais avec en plus, des caractères spéciaux tel que les accents ou certains symboles utiles pour la langue française. \n L'utilisation du lorem standard est facile dutilisation mais lorsque le futur client utilisera votre logiciel il se peut que certains caractères spéciaux ou qu'un accent ne soient pas codés correctement. \n Cette page a pour but donc de pouvoir perdre le moins de temps possible et donc de tester directement si tous les encodages de base de donnée ou des sites sont les bons de plus il permet de récuperer un code css avec le texte formaté !";
// alert(removeDiacritics(paragraph));

0
base-baileys-memory/node_modules/psl/.env generated vendored Normal file
View File

View File

@@ -1,7 +1,7 @@
const connectionReady = (cb = () =>{}) => { const connectionReady = (cb = () =>{}) => {
console.log('Listo para escuchas mensajes') console.log('Listo para escuchar mensajes')
console.log('Client is ready!'); console.log('El cliente esta listo!');
console.log('🔴 escribe: hola'); console.log('🔴 Escribe: /menu');
cb() cb()
} }

View File

@@ -1,25 +1,26 @@
const { get, reply, getIA } = require('../adapter') const { get, reply, getIA } = require('../adapter')
const { saveExternalFile, checkIsUrl } = require('./handle') const { saveExternalFile, checkIsUrl } = require('./handle')
const getMessages = async (message) => { const getMessages = async (message, num) => { //MOD by CHV - Agregamos el parametro "num" para recibir el numero desde "app.js"
const data = await get(message) // console.log("GETMESSAGES (flow.js)")
const data = await get(message, num) //MOD by CHV - Agregamos "num"
return data return data
} }
const responseMessages = async (step) => { const responseMessages = async (step) => {
const data = await reply(step) const data = await reply(step)
if (data && data.media) { if( data && data.media ){
const file = checkIsUrl(data.media) ? await saveExternalFile(data.media) : data.media; const file = checkIsUrl(data.media) ? await saveExternalFile(data.media) : data.media;
return { ...data, ...{ media: file } } return { ...data, ...{media:file}}
} }
return data return data
} }
const bothResponse = async (message, sessionId) => { const bothResponse = async (message) => {
const data = await getIA(message, sessionId) const data = await getIA(message)
if (data && data.media) { if(data && data.media){
const file = await saveExternalFile(data.media) const file = await saveExternalFile(data.media)
return { ...data, ...{ media: file } } return {...data,...{media:file}}
} }
return data return data
} }

View File

@@ -47,7 +47,7 @@ const generateImage = (base64, cb = () => {}) => {
let qr_svg = qr.image(base64, { type: 'svg', margin: 4 }); let qr_svg = qr.image(base64, { type: 'svg', margin: 4 });
qr_svg.pipe(require('fs').createWriteStream('./mediaSend/qr-code.svg')); qr_svg.pipe(require('fs').createWriteStream('./mediaSend/qr-code.svg'));
console.log(`⚡ Recuerda que el QR se actualiza cada minuto ⚡'`); console.log(`⚡ Recuerda que el QR se actualiza cada minuto ⚡'`);
console.log(`⚡ Actualiza F5 el navegador para mantener el mejor QR`); console.log(`⚡ Actualiza F5 el navegador para ver el QR mas reciente`);
cb() cb()
} }
@@ -69,7 +69,7 @@ const createClient = () => {
authStrategy: new LocalAuth( authStrategy: new LocalAuth(
{dataPath: './sessions/', {dataPath: './sessions/',
clientId: 'bot'}), clientId: 'bot'}),
puppeteer: { headless: false } puppeteer: { headless: false, args: ['--no-sandbox','--disable-setuid-sandbox'] }
}); });
} }

View File

@@ -2,8 +2,9 @@
const ExcelJS = require('exceljs'); const ExcelJS = require('exceljs');
const moment = require('moment'); const moment = require('moment');
const fs = require('fs'); const fs = require('fs');
const { MessageMedia, Buttons } = require('whatsapp-web.js'); const { MessageMedia, Buttons, List } = require('whatsapp-web.js');
const { cleanNumber } = require('./handle') const { cleanNumber } = require('./handle')
const { remplazos } = require('../adapter/index'); //MOD by CHV - Agregamos remplazos
const DELAY_TIME = 170; //ms const DELAY_TIME = 170; //ms
const DIR_MEDIA = `${__dirname}/../mediaSend`; const DIR_MEDIA = `${__dirname}/../mediaSend`;
// import { Low, JSONFile } from 'lowdb' // import { Low, JSONFile } from 'lowdb'
@@ -15,12 +16,15 @@ const { saveMessage } = require('../adapter')
* @param {*} fileName * @param {*} fileName
*/ */
const sendMedia = (client, number = null, fileName = null) => { const sendMedia = (client, number = null, fileName = null, trigger = null) => {
if(!client) return console.error("El objeto cliente no está definido."); if(!client) return console.error("El objeto cliente no está definido.");
console.log("MEDIA:"+fileName);
try { try {
number = cleanNumber(number || 0) number = cleanNumber(number || 0)
const file = `${DIR_MEDIA}/${fileName}`; const file = `${DIR_MEDIA}/${fileName}`;
console.log("FILE="+file);
if (fs.existsSync(file)) { if (fs.existsSync(file)) {
console.log("ARCHIVO EXISTE");
const media = MessageMedia.fromFilePath(file); const media = MessageMedia.fromFilePath(file);
client.sendMessage(number, media, { sendAudioAsVoice: true }); client.sendMessage(number, media, { sendAudioAsVoice: true });
} }
@@ -54,13 +58,14 @@ const sendMedia = (client, number = null, fileName = null) => {
* Enviamos un mensaje simple (texto) a nuestro cliente * Enviamos un mensaje simple (texto) a nuestro cliente
* @param {*} number * @param {*} number
*/ */
const sendMessage = async (client, number = null, text = null, trigger = null) => { const sendMessage = async (client, number = null, text = null, trigger = null, regla) => { //MOD by CHV - Agregamos el parametro "regla" para guardarlo en "chats/numero.json"
setTimeout(async () => { setTimeout(async () => {
number = cleanNumber(number) number = cleanNumber(number)
const message = text const message = text
client.sendMessage(number, message); client.sendMessage(number, message);
await readChat(number, message, trigger) await readChat(number, message, trigger, regla) //MOD by CHV - Agregamos el parametro "regla"
console.log(`⚡⚡⚡ Enviando mensajes....`); console.log(`⚡⚡⚡ Enviando mensajes....`);
// console.log("********************* SEND MESSAGE **************************************");
},DELAY_TIME) },DELAY_TIME)
} }
@@ -72,13 +77,31 @@ const sendMessageButton = async (client, number = null, text = null, actionButto
setTimeout(async () => { setTimeout(async () => {
number = cleanNumber(number) number = cleanNumber(number)
const { title = null, message = null, footer = null, buttons = [] } = actionButtons; const { title = null, message = null, footer = null, buttons = [] } = actionButtons;
let button = new Buttons(message,[...buttons], title, footer); let button = new Buttons(remplazos(message, client),[...buttons], remplazos(title, client), remplazos(footer, client));
client.sendMessage(number, button);
await readChat(number, message, actionButtons) await readChat(number, message, actionButtons)
console.log(`⚡⚡⚡ Enviando mensajes....`); client.sendMessage(number, button);
console.log(`⚡⚡⚡ Enviando mensajes (botones)....`);
// console.log("sendMessageButton.");
}, DELAY_TIME) }, DELAY_TIME)
// console.log("************************ SEND MESSAGE BUTTON ***********************************");
} }
/**
* Enviamos listas (con el formato de response.json)
* @param {*} number
*/
const sendMessageList = async (client, number = null, text = null, actionList) => {
setTimeout(async () => {
// console.log("********************** client **************************")
// console.log(client)
number = cleanNumber(number)
const { body = null, buttonText = null, sections = [], title = null, footer = null } = actionList;
let aList = new List( remplazos(body, client),remplazos(buttonText, client),[...sections],remplazos(title, client),remplazos(footer, client));
client.sendMessage(number, aList);
await readChat(number, message, actionList)
console.log('⚡⚡⚡ Enviando lista a '+number+' ....');
}, DELAY_TIME)
}
/** /**
* Opte * Opte
@@ -106,10 +129,10 @@ const lastTrigger = (number) => new Promise((resolve, reject) => {
* @param {*} number * @param {*} number
* @param {*} message * @param {*} message
*/ */
const readChat = async (number, message, trigger = null) => { const readChat = async (number, message, trigger = null, regla) => { //MOD by CHV - Agregamos el parametro "regla" para guardarlo en "chats/numero.json"
number = cleanNumber(number) number = cleanNumber(number)
await saveMessage( message, trigger, number ) await saveMessage( message, trigger, number, regla ) //MOD by CHV - Agregamos "regla"
console.log('Saved') // console.log('Saved')
} }
module.exports = { sendMessage, sendMedia, lastTrigger, sendMessageButton, readChat, sendMediaVoiceNote } module.exports = { sendMessage, sendMedia, lastTrigger, sendMessageButton, sendMessageList, readChat, sendMediaVoiceNote }

View File

@@ -1,93 +1,86 @@
[ [
{ {
"keywords": [ "keywords": [
"hola", "xmuchas gracias",
"hola!", "xgracias",
"ola", "xvale gracias"
"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" "key": "Gracias"
}, },
{ {
"keywords": [ "keywords": ["/menu"],
"cursos", "key": "menu"
"info",
"curso" ],
"key": "STEP_2"
}, },
{ {
"keywords": [ "keywords": ["1"],
"angular" "key": "opcion1"
],
"key": "STEP_2_1"
}, },
{ {
"keywords": [ "keywords": ["2"],
"node" "key": "opcion2"
],
"key": "STEP_2_2"
}, },
{ {
"keywords": [ "keywords": ["3"],
"ngrx" "key": "opcion3"
],
"key": "STEP_2_3"
}, },
{ {
"keywords": [ "keywords": "*",
"aws" "key": "recibenombre"
],
"key": "STEP_2_4"
}, },
{ {
"keywords": [ "keywords": "*",
"asesor", "key": "gRevisaCliente"
"asesores",
"Vendedor",
"cobrador"
],
"key": "STEP_3"
}, },
{ {
"keywords": [ "keywords": "*",
"muchas gracias", "key": "gGuardainfo"
"ok",
"gracias",
"vale gracias"
],
"key": "STEP_4"
}, },
{ {
"keywords": [ "keywords": ["rnd"],
"youtube" "key": "rnd"
],
"key": "STEP_5"
}, },
{ {
"keywords": [ "keywords": ["rnd2"],
"VER_CURSOS" "key": "rnd2"
],
"key": "STEP_6"
}, },
{ {
"keywords": [ "keywords": ["4"],
"telegram" "key": "lista"
],
"key": "STEP_7"
}, },
{ {
"keywords": [ "keywords": ["5"],
"audio" "key": "botones"
], },
"key": "STEP_8" {
"keywords": ["6"],
"key": "botonespaq3"
},
{
"keywords": ["cursos"],
"key": "cursos"
},
{
"keywords": ["youtube"],
"key": "youtube"
},
{
"keywords": ["telegram"],
"key": "telegram"
},
{
"keywords": ["manzana"],
"key": "manzana"
},
{
"keywords": ["mango"],
"key": "mango"
},
{
"keywords": ["platano"],
"key": "platano"
},
{
"keywords": "*pak*3*|*pak*angular*|*paquete*3*|*paquete*angular*",
"key": "paq3"
} }
] ]

View File

@@ -11,27 +11,99 @@
"media":null, "media":null,
"trigger":null "trigger":null
}, },
"STEP_0":{ "menu":{
"replyMessage":[ "replyMessage":[
"El flujo ha finalizado \n", "%saludo% %primer_nombre%, este es el menú, selecciona una opción: \n",
"pero puedes ver todo el codigo de este \n", "Pon *1* para mensajes anteriores.\n",
"repositorio en https://github.com/leifermendez/bot-whatsapp.git" "Pon *2* para ver remplazos.\n",
"Pon *3* para pedir nombre (RegExp).\n",
"Pon *4* para un ejemplo de listas y expresiones regulares.\n",
"Pon *5* para un ejemplo de botones.\n",
"Pon *6* para un ejemplo de botones y regExp.\n"
], ],
"media":null, "media":null,
"trigger":null "trigger":null
}, },
"STEP_1":{ "opcion1":{
"replyMessage":[ "replyMessage":[
"Hola! y✌ Bienvenido a este 🤖 CHATBOT de Whatsapp, lo primero \n", "Seleccionaste la opción 1\n",
"decirte que mi nombre es *Leifer Mendez*😎 \n", "*Ultimo mensaje:*\n",
"\n Si necesitas ver más info sobre las capacitacion tecnicas ", "%msjant_0%\n",
"escribe *cursos* o *info* o escribe *audio*" "*Penultimo mensaje:*\n",
"%msjant_1%\n",
"*Antepenultimo mensaje:*\n",
"%msjant_2% \n\n",
"Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 3 sin necesidad de volver a iniciar con */menu*."
], ],
"media":"https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif", "media":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"opcion2":{
"replyMessage":[
"Seleccionaste la opción 2\n",
"Remplazamos %saludo.% con *\"%saludo%\"*\n",
"Remplazamos %dia_semana.% con *\"%dia_semana%\"*\n",
"Remplazamos %hora24.%:%minutos.% con *\"%hora24%:%minutos%\"*\n",
"Remplazamos %.rnd_👍🏽,🤞🏼,🤪,🤔% con '%rnd_👍🏽,🤞🏼,🤪,🤔%'\n\n",
"Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 3 sin necesidad de volver a iniciar con */menu*."
],
"media":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"opcion3":{
"replyMessage":[
"Seleccionaste la opción 3\n\n",
"Por favor dame tu nombre.\n\n",
"Aquí vamos a aceptar *cualquier* texto, porque en el *initial.json* tenemos keywords : \"***\" (un asterisco en expresiones regulares quiere decir *\"cualquier cosa\"*)\n",
"Y en *response.json* en la opción correspondiente tenemos \"pasoRequerido\" : \"menu\", que quiere decir que SOLO se va a disparar cuando el paso anterior sea \"menu\"."
],
"media":null,
"pasoRequerido":"menu"
},
"recibenombre":{
"replyMessage":[
"Gracias por tu nombre *%msjant_0%*.\n\n",
"Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 3 sin necesidad de volver a iniciar con */menu*."
],
"media":null,
"trigger":null,
"pasoRequerido":"opcion3",
"goto":"menu"
},
"gRevisaCliente":{
"replyMessage":[
"/URL=http://localhost:8888/dbrquery?j={\"query\":\"select_revisacliente_GUNA\",\"exec\":\"ExecuteQuery\",\"params\":{\"par1\":\"XXPARAM1XX\"}}"
],
"media":null,
"trigger":null,
"pasoRequerido":"gallina"
},
"gGuardainfo":{
"replyMessage":[
"/URL=http://localhost:8888/dbrquery?j={\"query\":\"insert_registroGallina_GUNA\",\"exec\":\"ExecuteCommand\",\"params\":{\"par1\":\"XXPARAM1XX\", \"par2\":\"XXPARAM2XX\", \"par3\":\"XXPARAM3XX\", \"par4\":\"XXPARAM4XX\"}}"
],
"media":null,
"trigger":null,
"pasoRequerido":"gRevisaCliente"
},
"rnd":{
"replyMessage":[
"%saludo%\nHoy es %dia_semana%.\nSon las %hora24%:%minutos% hrs.\nSon las %hora12%:%minutos% %ampm%\n*Palabra random:* %rnd_arbol,burro,cabra,dinosaurio,elefante,fuego,gorila%\n*Emoji random:* %rnd_👍🏽,😁,🤣,🤔,🤦🏽‍♂️,🙄,😎%\n*Número random:* %rnd_1,2,3,4,5,6,7%\n"
],
"media":null,
"trigger":null
},
"rnd2":{
"replyMessage":[
""
],
"media":null,
"trigger":null, "trigger":null,
"actions":{ "actions":{
"title":"¿Que te interesa ver?", "title":"¿Que te interesa ver?",
"message":"Recuerda todo este contenido es gratis y estaria genial que me siguas!", "message":"%saludo%\nHoy es %dia_semana%.\nSon las %hora24%:%minutos% hrs.\nSon las %hora12%:%minutos% %ampm%\n*Palabra random:* %rnd_arbol,burro,cabra,dinosaurio,elefante,fuego,gorila%\n*Emoji random:* %rnd_👍🏽,😁,🤣,🤔,🤦🏽‍♂️,🙄,😎%\n*Número random:* %rnd_1,2,3,4,5,6,7%\n",
"footer":"Gracias", "footer":"Gracias",
"buttons":[ "buttons":[
{"body":"Cursos"}, {"body":"Cursos"},
@@ -40,109 +112,108 @@
] ]
} }
}, },
"STEP_2":{ "lista":{
"replyMessage":[ "replyMessage":[
"Perfecto, te voy a pasar la lista ", "*%saludo%*, este es un ejemplo de listas"
"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, "media":null,
"trigger":null "trigger":null,
"actions":{
"body":"Hola *%primer_nombre%*, estos son ejemplos del uso de expresiones regulares, todas las opciones de la lista disparan la misma regla:\n\n'*pak*3*|*pak*angular*|*paquete*3*|*paquete*angular*'\n\nAutomáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*.",
"buttonText":"Ver los ejemplos de RegEx",
"sections": [
{"title":"Selecciona un mensaje:",
"rows":[
{"id": "paq3", "title": "Me gusta el paquete 3"},
{"id": "paqA", "title": "Por favor mas info del paquete de Angular"},
{"id": "pakA", "title": "Me gustó el pak de Angular"},
{"id": "pak3", "title": "Estoy interesado en el pak 3"}
]
}
],
"title":"Por favor selecciona un producto"
}, },
"STEP_2_3":{ "pasoRequerido":"menu",
"goto":"menu"
},
"botones":{
"replyMessage":[ "replyMessage":[
"NGRX para manejar estados en Angular \n", "*%saludo%*, este es un ejemplo de botones"
"*(Gratis)* https://bit.ly/ngrx-desde-cero \n", ],
"A darle! 😮" "media":"https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif",
"trigger":null,
"actions":{
"title":"¿Que te interesa ver %primer_nombre%?",
"message":"Recuerda todo este contenido es gratis y estaria genial que me sigas!",
"footer":"Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*.",
"buttons":[
{"body":"Cursos"},
{"body":"Youtube"},
{"body":"Telegram"}
]
},
"pasoRequerido":"menu",
"goto":"menu"
},
"cursos":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste *Cursos*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"youtube":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste *YouTube*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"telegram":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste *Telegram*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"manzana":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste *manzana*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"mango":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste *mango*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"platano":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste *platano*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"paq3":{
"replyMessage":["*%saludo% %primer_nombre%*, seleccionaste el *paquete 3 de Angular*\n\n","Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*."],
"trigger":null,
"pasoRequerido":"menu",
"goto":"menu"
},
"botonespaq3":{
"replyMessage":[
"*%saludo%*, este es un ejemplo de botones y regExp"
], ],
"media":null, "media":null,
"trigger":null "trigger":null,
"actions":{
"title":"Hola %primer_nombre%, escoge un mensaje:",
"message":"Estos son ejemplos del uso de expresiones regulares, todos los botones disparan la misma regla:\n\n'*pak*3*|*pak*angular*|*paquete*3*|*paquete*angular*'\n\n",
"footer":"Automáticamente el flujo se regresa al *menú*, asi que puedes poner nuevamente un número del 1 al 5 sin necesidad de volver a iniciar con */menu*.",
"buttons":[
{"body":"Me gusta el paquete 3"},
{"body":"Mas info del paquete de Angular"},
{"body":"Quiero mas información del pak 3"}
]
}, },
"STEP_2_4":{ "pasoRequerido":"menu",
"replyMessage":[ "goto":"menu"
"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
} }
} }

Binary file not shown.

4350
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,36 +1,18 @@
{ {
"name": "bot-whatsapp", "name": "test-ws-bot",
"version": "1.0.0", "version": "1.0.0",
"description": "Bot de wahtsapp open source para MVP o pequeños negocios", "description": "",
"main": "app.js", "main": "app.js",
"scripts": { "scripts": {
"start": "node ./app.js", "start": "node ./app.js",
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },
"keywords": [ "keywords": [],
"whatsapp", "author": "",
"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": {
"type": "git",
"url": "https://github.com/leifermendez/bot-whatsapp"
},
"license": "ISC", "license": "ISC",
"dependencies": { "dependencies": {
"@google-cloud/dialogflow": "^5.2.0", "@google-cloud/dialogflow": "^5.2.0",
"axios": "^0.27.2",
"cors": "^2.8.5", "cors": "^2.8.5",
"dotenv": "^16.0.1", "dotenv": "^16.0.1",
"exceljs": "^4.3.0", "exceljs": "^4.3.0",
@@ -45,7 +27,7 @@
"qrcode-terminal": "^0.12.0", "qrcode-terminal": "^0.12.0",
"socket.io": "^4.5.1", "socket.io": "^4.5.1",
"stormdb": "^0.6.0", "stormdb": "^0.6.0",
"whatsapp-web.js": "^1.18.4", "whatsapp-web.js": "github:pedroslopez/whatsapp-web.js#fix-buttons-list",
"xlsx": "^0.18.5" "xlsx": "^0.18.5"
}, },
"devDependencies": { "devDependencies": {

38
spam.json Normal file
View File

@@ -0,0 +1,38 @@
[
{
"numero":"52155XXXXXX39"
},
{
"numero":"52155XXXXXX28"
},
{
"numero":"52155XXXXXX39"
},
{
"numero":"52155XXXXXX28"
},
{
"numero":"52155XXXXXX39"
},
{
"numero":"52155XXXXXX28"
},
{
"numero":"52155XXXXXX39"
},
{
"numero":"52155XXXXXX28"
},
{
"numero":"52155XXXXXX39"
},
{
"numero":"52155XXXXXX28"
},
{
"numero":"52155XXXXXX28"
},
{
"numero":"52155XXXXXX39"
}
]