Compare commits

..

59 Commits

Author SHA1 Message Date
Leifer Mendez
ee87e9e875 Merge pull request #140 from jzvi12/main
Use WhatsApp Number as an Unique Dialogflow Session ID
2022-12-06 09:57:10 +01:00
jzvi12
1884832192 fixed dialogflow session id 2022-12-05 19:10:41 -05:00
jzvi12
f21a58b6ff Update README.md 2022-12-02 10:16:16 -05:00
Leifer Mendez
8a4f134327 Update latest 2022-11-19 11:08:51 +01:00
Leifer Mendez
9b6ce92612 Update diaglogflow.js 2022-10-31 09:51:18 +01:00
Leifer
663fcafc9c update 2022-10-24 09:39:38 +02:00
Leifer
f36ddd3014 update 2022-10-20 13:51:43 +02:00
Leifer Mendez
3fadaaaf13 Merge pull request #82 from Gonzalito87/patch-7
Update de librerias
2022-08-09 20:57:57 +02:00
Leifer Mendez
dfa569f29d Merge pull request #83 from aurik3/dev-pull
coreccion nanoid y send.js
2022-08-09 20:57:41 +02:00
Leifer Mendez
601508f379 Merge branch 'main' into dev-pull 2022-08-09 20:57:29 +02:00
aurik3
e7ad205268 coreccion nanoid y send.js
se corrigen errores en el codigo
2022-08-09 13:43:28 -05:00
Gonzalito87
f62ba0a076 Update de librerias 2022-08-08 15:19:13 -03:00
Leifer Mendez
a9efa0aa58 Merge pull request #71 from ulisesvina/main
Arreglado: comprobación de parámetros en funciones y problemas de inconsistencia.
2022-08-08 14:47:33 +02:00
Leifer Mendez
3276c21bc3 Merge pull request #75 from aurik3/dev-pull
Update Julio 2022
2022-07-22 12:57:14 +02:00
aurik3
1114f25a71 Update Julio 2022
Se añade localAuth para mantener la session despues de escanear el codigo y se hace un code fix al api rest

Co-Authored-By: Leifer Mendez <15802366+leifermendez@users.noreply.github.com>
2022-07-19 18:15:12 -05:00
Ulises Viña
f13a34ff75 Update send.js 2022-07-05 20:59:06 -05:00
Leifer Mendez
d45ea85e7d Merge pull request #57 from leifermendez/rev-global
rex
2022-04-27 21:33:51 +02:00
Leifer Mendez
10e2b138d3 rex 2022-04-27 21:32:29 +02:00
Leifer Mendez
a1bf5ba5c2 Merge pull request #55 from Gonzalito87/patch-3
Update package.json
2022-04-27 21:03:06 +02:00
Gonzalito87
19102b7b3a Update package.json
actualizacion de repositorio de whats app
2022-04-26 11:12:35 -03:00
Leifer Mendez
5efcc2a9a6 Merge pull request #54 from Gonzalito87/patch-1
Update diaglogflow.js
2022-04-25 19:33:28 +02:00
Gonzalito87
8279c07a88 Update diaglogflow.js 2022-04-25 13:42:12 -03:00
Leifer Mendez
02d7b3bd98 Merge pull request #50 from leifermendez/update
Update
2022-04-20 14:17:02 +02:00
Leifer Mendez
f8f6a3000d Merge branch 'update' of github.com:leifermendez/bot-whatsapp into update 2022-04-20 14:16:46 +02:00
Leifer Mendez
9a92b152a4 fix 2022-04-20 14:16:42 +02:00
Leifer Mendez
f86700deaf Update README.md 2022-04-15 12:03:22 +02:00
Leifer Mendez
4ba259b46c Update README.md 2022-04-15 12:02:59 +02:00
Leifer Mendez
cf459e94d2 Update README.md 2022-04-15 12:01:29 +02:00
Leifer Mendez
4f8ed1361c Merge pull request #47 from leifermendez/update
Update
2022-04-15 12:00:43 +02:00
Leifer Mendez
bad8802241 Merge branch 'main' into update 2022-04-15 12:00:37 +02:00
Leifer Mendez
f09ac862d5 clean credentials 2022-04-15 11:58:09 +02:00
Leifer Mendez
fe7567e1a9 update many stuff 2022-04-15 11:57:32 +02:00
Leifer Mendez
9b0b7f4d54 befor update 2022-04-15 11:02:12 +02:00
Leifer Mendez
3ddbf462a8 Update package.json 2022-03-29 17:02:09 +02:00
Leifer Mendez
e6043c99a7 Merge pull request #35 from leifermendez/dev
update
2022-03-23 09:41:38 +01:00
Leifer Mendez
b1daa0020e update 2022-03-23 09:41:18 +01:00
Leifer Mendez
190d35c9a5 Merge pull request #30 from leifermendez/dev
Dev
2022-03-16 10:05:54 +01:00
Leifer Mendez
e4378fe848 se agego multi-device .env 2022-03-16 10:05:13 +01:00
Leifer Mendez
981a6bd928 Merge pull request #25 from tonyvazgar/main
Pasado a DEV
2022-03-15 10:24:01 +01:00
Leifer Mendez
676e48021f Merge pull request #28 from leifermendez/dev
Dev
2022-03-15 10:22:38 +01:00
Leifer Mendez
1d4daf10db change csv to json 2022-03-15 10:20:25 +01:00
Leifer Mendez
3c9341d87d Merge pull request #24 from rrruuuyyy/main
into dev branch
2022-03-15 10:06:11 +01:00
unknown
04982941a7 Save messages in Mysql or Json 2022-03-14 14:17:28 -06:00
Tony
ba4f05ebb2 Sesion en multi-device funcionando y escuchando, se guarda en localauth 2022-03-14 13:49:42 -06:00
Leifer Mendez
5aaf761fce update core 2022-03-10 17:45:30 +01:00
Leifer Mendez
12539d00fa before beta multi 2022-03-09 20:24:13 +01:00
Leifer Mendez
ec8ad955ee readme 2022-02-28 21:16:46 +01:00
Leifer Mendez
d10504c40b npm update 2022-02-28 21:14:00 +01:00
Leifer Mendez
d200100caa update 2022-02-26 11:50:44 +01:00
Leifer Mendez
902431c533 fix buttons 2022-02-23 15:55:56 +01:00
Leifer Mendez
e23540593a add voice note 2022-02-23 09:29:16 +01:00
Leifer Mendez
9b548d9418 Update README.md 2022-02-16 11:51:25 +01:00
Leifer Mendez
c25de59a93 Update README.md 2022-02-16 11:10:26 +01:00
Leifer Mendez
cfe2c17165 Merge pull request #5 from leifermendez/update
npm update
2022-02-16 08:59:28 +01:00
Leifer Mendez
1309b7f806 npm update 2022-02-16 08:58:30 +01:00
Leifer Mendez
1071469e53 Merge pull request #4 from leifermendez/issueAsyncFunction
issue async function
2022-02-14 18:16:28 +01:00
Leifer Mendez
1795e8de20 issue async function 2022-02-14 18:15:33 +01:00
Leifer Mendez
7414d958ab Merge pull request #3 from leifermendez/buttons
Buttons
2022-02-11 15:55:42 +01:00
Leifer Mendez
a3ebebb19c Update README.md 2022-02-09 19:58:47 +01:00
23 changed files with 6975 additions and 1323 deletions

View File

@@ -9,3 +9,5 @@ SQL_HOST=
SQL_USER=
SQL_PASS=
SQL_DATABASE=
KEEP_DIALOG_FLOW=false
MULTI_DEVICE=true

2
.gitignore vendored
View File

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

View File

@@ -1,18 +1,4 @@
## 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)
#### 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-ventas)
> Comprarme un cafe!
[![Comprar](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/leifermendez)
#### Actualización
@@ -23,22 +9,55 @@ El siguiente proyecto se realizó con fines educativos para el canal de [Youtube
| JSON File | ✅ |
| QR Scan (route) | ✅ |
| Easy deploy heroku | ✅ |
| Buttons | ✅ |
| Buttons | ✅ (No funciona en multi-device)|
| Send Voice Note | ✅ |
| Add support ubuntu/linux | ✅ |
### (Nuevo) Botones
[![btn](https://i.imgur.com/W7oYlSu.png)](https://youtu.be/5lEMCeWEJ8o)
## 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
@@ -50,9 +69,9 @@ sudo apt install -y gconf-service libasound2 libatk1.0-0 libc6 libcairo2 libcups
__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)
```
npm i
```
__Configurar .env__
> Con el editor de texto crea un archivo `.env` el cual debes de guiarte del archivo `.env.example`
@@ -71,21 +90,14 @@ 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
<img src="https://i.imgur.com/RSbPtat.png" width="500" />
Visitar la pagina
`http://localhost:3000/qr`
![](https://i.imgur.com/Q3JEDlP.png)
Tambien puedes visitar la pagina http://127.0.0.1:3000/qr
__Listo 😎__
> Cuando sale este mensaje tu BOT está __listo__ para trabajar!
@@ -102,11 +114,3 @@ __Listo 😎__
> 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)

View File

@@ -1,9 +1,11 @@
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
*/
const KEEP_DIALOG_FLOW = (process.env.KEEP_DIALOG_FLOW === 'true')
let PROJECID;
let CONFIGURATION;
let sessionClient;
@@ -28,9 +30,9 @@ const checkFileCredentials = () => {
// Detect intent method
const detectIntent = async (queryText) => {
const detectIntent = async (queryText, waPhoneNumber) => {
let media = null;
const sessionId = nanoid.nanoid()
const sessionId = KEEP_DIALOG_FLOW ? 1 : waPhoneNumber;
const sessionPath = sessionClient.projectAgentSessionPath(PROJECID, sessionId);
const languageCode = process.env.LANGUAGE
const request = {
@@ -54,7 +56,7 @@ const detectIntent = async (queryText) => {
const { fields } = parsePayload.payload
media = fields.media.stringValue || null
}
// const customPayload = parsePayload['payload']
const customPayload = parsePayload ? parsePayload['payload'] : null
const parseData = {
replyMessage: queryResult.fulfillmentText,
@@ -64,8 +66,8 @@ const detectIntent = async (queryText) => {
return parseData
}
const getDataIa = (message = '', cb = () => { }) => {
detectIntent(message).then((res) => {
const getDataIa = (message = '', sessionId = '', cb = () => { }) => {
detectIntent(message, sessionId).then((res) => {
cb(res)
})
}

View File

@@ -1,4 +1,5 @@
const { getData, getReply } = require('./mysql')
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')
@@ -51,17 +52,39 @@ const reply = (step) => new Promise((resolve, reject) => {
}
})
const getIA = (message) => new Promise((resolve, reject) => {
const getIA = (message, sessionId) => new Promise((resolve, reject) => {
/**
* Si usas dialogflow
*/
if (process.env.DATABASE === 'dialogflow') {
let resData = { replyMessage: '', media: null, trigger: null }
getDataIa(message,(dt) => {
getDataIa(message, sessionId, (dt) => {
resData = { ...resData, ...dt }
resolve(resData)
})
}
})
module.exports = { get, reply, getIA }
/**
*
* @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:
resolve(true)
break;
}
})
module.exports = { get, reply, getIA, saveMessage }

20
adapter/jsonDb.js Normal file
View File

@@ -0,0 +1,20 @@
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 }

View File

@@ -21,8 +21,54 @@ getReply = (option_key = '', callback) => connection.query(
replyMessage:response?.replyMessage || '',
trigger:response?.trigger || '',
media:response?.media || ''
}
callback(value)
});
module.exports = {getData, getReply}
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) {
//TODO esta parte es mejor incluirla directamente en el archivo .sql template
console.log('DEBES DE CREAR LA TABLA DE MESSAGE')
// 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}

148
app.js
View File

@@ -6,10 +6,10 @@ const fs = require('fs');
const express = require('express');
const cors = require('cors')
const qrcode = require('qrcode-terminal');
const { Client } = require('whatsapp-web.js');
const { Client, LocalAuth } = require('whatsapp-web.js');
const mysqlConnection = require('./config/mysql')
const { middlewareClient } = require('./middleware/client')
const { generateImage, cleanNumber } = require('./controllers/handle')
const { generateImage, cleanNumber, checkEnvFile, createClient, isValidNumber } = require('./controllers/handle')
const { connectionReady, connectionLost } = require('./controllers/connection')
const { saveMedia } = require('./controllers/save')
const { getMessages, responseMessages, bothResponse } = require('./controllers/flows')
@@ -17,43 +17,32 @@ const { sendMedia, sendMessage, lastTrigger, sendMessageButton, readChat } = req
const app = express();
app.use(cors())
app.use(express.json())
const MULTI_DEVICE = process.env.MULTI_DEVICE || 'true';
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;
app.use('/', require('./routes/web'))
/**
* Escuchamos cuando entre un mensaje
*/
const listenMessage = () => client.on('message', async msg => {
const { from, body, hasMedia } = msg;
if (!isValidNumber(from)) {
return
}
// 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)
console.log('BODY', message)
const number = cleanNumber(from)
await readChat(number, message)
/**
* Guardamos el archivo multimedia que envia
*/
@@ -67,7 +56,8 @@ const listenMessage = () => client.on('message', async msg => {
*/
if (process.env.DATABASE === 'dialogflow') {
const response = await bothResponse(message);
if (!message.length) return;
const response = await bothResponse(message, number);
await sendMessage(client, from, response.replyMessage);
if (response.media) {
sendMedia(client, from, response.media);
@@ -82,7 +72,6 @@ const listenMessage = () => client.on('message', async msg => {
*/
const lastStep = await lastTrigger(from) || null;
console.log({ lastStep })
if (lastStep) {
const response = await responseMessages(lastStep)
await sendMessage(client, from, response.replyMessage);
@@ -92,12 +81,21 @@ const listenMessage = () => client.on('message', async msg => {
* Respondemos al primero paso si encuentra palabras clave
*/
const step = await getMessages(message);
console.log({ step })
if (step) {
const response = await responseMessages(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);
@@ -118,7 +116,7 @@ const listenMessage = () => client.on('message', async msg => {
/**
* Si quieres enviar botones
*/
if(response.hasOwnProperty('actions')){
if (response.hasOwnProperty('actions')) {
const { actions } = response;
await sendMessageButton(client, from, null, actions);
}
@@ -126,87 +124,38 @@ const listenMessage = () => client.on('message', async msg => {
}
});
/**
* Revisamos si tenemos credenciales guardadas para inciar sessio
* este paso evita volver a escanear el QRCODE
*/
const withSession = () => {
// Si exsite cargamos el archivo con las credenciales
console.log(`Validando session con Whatsapp...`)
sessionData = require(SESSION_FILE_PATH);
client = new Client({
session: sessionData,
puppeteer: {
args: [
'--no-sandbox'
],
}
});
client.on('ready', () => {
connectionReady()
listenMessage()
loadRoutes(client);
socketEvents.sendStatus()
});
client.on('auth_failure', () => connectionLost())
client = new Client({
authStrategy: new LocalAuth(),
puppeteer: { headless: true }
});
client.initialize();
}
client.on('qr', qr => generateImage(qr, () => {
qrcode.generate(qr, { small: true });
/**
* Generamos un QRCODE para iniciar sesion
*/
const withOutSession = () => {
console.log('No tenemos session guardada');
console.log(`Ver QR http://localhost:${port}/qr`)
socketEvents.sendQR(qr)
}))
client = new Client({
puppeteer: {
args: [
'--no-sandbox'
],
}
});
client.on('ready', (a) => {
connectionReady()
listenMessage()
// socketEvents.sendStatus(client)
});
client.on('qr', qr => generateImage(qr, () => {
qrcode.generate(qr, { small: true });
console.log(`Ver QR http://localhost:${port}/qr`)
socketEvents.sendQR(qr)
}))
client.on('auth_failure', (e) => {
// console.log(e)
// connectionLost()
});
client.on('ready', (a) => {
connectionReady()
listenMessage()
loadRoutes(client);
// socketEvents.sendStatus(client)
});
client.on('authenticated', () => {
console.log('AUTHENTICATED');
});
client.on('auth_failure', () => connectionLost());
client.initialize();
client.on('authenticated', (session) => {
sessionData = session;
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)) ? withSession() : withOutSession();
/**
* Verificamos si tienes un gesto de db
@@ -219,5 +168,4 @@ if (process.env.DATABASE === 'mysql') {
server.listen(port, () => {
console.log(`El server esta listo por el puerto ${port}`);
})
checkEnvFile();

Binary file not shown.

View File

@@ -1,9 +1,9 @@
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 || 'secret',
database : process.env.SQL_DATABASE || 'my_db'
user : process.env.SQL_USER || 'root',
password : process.env.SQL_PASS || '',
database : process.env.SQL_DATABASE || 'pruebas'
});
const connect = () => connection.connect(function(err) {

View File

@@ -1,6 +1,7 @@
const connectionReady = (cb = () =>{}) => {
console.log('Listo para escuchas mensajes')
console.log('Client is ready!');
console.log('🔴 escribe: hola');
cb()
}

View File

@@ -1,5 +1,5 @@
const {get, reply, getIA} = require('../adapter')
const {saveExternalFile} = require('./handle')
const {saveExternalFile, checkIsUrl} = require('./handle')
const getMessages = async (message) => {
const data = await get(message)
@@ -9,14 +9,14 @@ const getMessages = async (message) => {
const responseMessages = async (step) => {
const data = await reply(step)
if(data && data.media){
const file = await saveExternalFile(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)
const bothResponse = async (message, sessionId) => {
const data = await getIA(message, sessionId)
if(data && data.media){
const file = await saveExternalFile(data.media)
return {...data,...{media:file}}

View File

@@ -1,8 +1,10 @@
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 || 'true';
const cleanNumber = (number) => {
number = number.replace('@c.us', '');
@@ -31,6 +33,16 @@ const saveExternalFile = (url) => new Promise((resolve, reject) => {
});
})
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'));
@@ -39,4 +51,33 @@ const generateImage = (base64, cb = () => {}) => {
cb()
}
module.exports = {cleanNumber, saveExternalFile, generateImage}
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 = () => {
client = new Client({
authStrategy: new LocalAuth(
{dataPath: './sessions/',
clientId: 'bot'}),
puppeteer: { headless: false }
});
}
const isValidNumber = (rawNumber) => {
const regexGroup = /\@g.us\b/gm;
const exist = rawNumber.match(regexGroup);
return !exist
}
module.exports = {cleanNumber, saveExternalFile, generateImage, checkIsUrl, checkEnvFile, createClient, isValidNumber}

View File

@@ -4,32 +4,64 @@ const moment = require('moment');
const fs = require('fs');
const { MessageMedia, Buttons } = require('whatsapp-web.js');
const { cleanNumber } = require('./handle')
const { saveMedia } = require('../controllers/save')
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) => {
const dirMedia = `${__dirname}/../mediaSend/${fileName}`;
number = cleanNumber(number)
if (fs.existsSync(dirMedia)) {
const media = MessageMedia.fromFilePath(dirMedia);
client.sendMessage(number, media);
const sendMedia = (client, number = null, fileName = null) => {
if(!client) return cosnole.error("El objeto cliente no está definido.");
try {
number = cleanNumber(number || 0)
const file = `${DIR_MEDIA}/${fileName}`;
if (fs.existsSync(file)) {
const media = MessageMedia.fromFilePath(file);
client.sendMessage(number, media, { sendAudioAsVoice: true });
}
} catch(e) {
throw e;
}
}
/**
* Enviamos archivos como notas de voz
* @param {*} number
* @param {*} fileName
*/
const sendMediaVoiceNote = (client, number = null, fileName = null) => {
if(!client) return cosnole.error("El objeto cliente no está definido.");
try {
number = cleanNumber(number || 0)
const file = `${DIR_MEDIA}/${fileName}`;
if (fs.existsSync(file)) {
const media = MessageMedia.fromFilePath(file);
client.sendMessage(number, media ,{ sendAudioAsVoice: true });
}
}catch(e) {
throw e;
}
}
/**
* 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)
}
/**
@@ -73,49 +105,9 @@ const lastTrigger = (number) => new Promise((resolve, reject) => {
* @param {*} message
*/
const readChat = async (number, message, trigger = null) => {
setTimeout(() => {
const pathExcel = `${__dirname}/../chats/${number}.xlsx`;
const workbook = new ExcelJS.Workbook();
const today = moment().format('DD-MM-YYYY hh:mm')
if (fs.existsSync(pathExcel)) {
/**
* Si existe el archivo de conversacion lo actualizamos
*/
const workbook = new ExcelJS.Workbook();
workbook.xlsx.readFile(pathExcel)
.then(() => {
const worksheet = workbook.getWorksheet(1);
const lastRow = worksheet.lastRow;
var getRowInsert = worksheet.getRow(++(lastRow.number));
getRowInsert.getCell('A').value = today;
getRowInsert.getCell('B').value = message;
getRowInsert.getCell('C').value = trigger;
getRowInsert.commit();
workbook.xlsx.writeFile(pathExcel);
});
} else {
/**
* NO existe el archivo de conversacion lo creamos
*/
const worksheet = workbook.addWorksheet('Chats');
worksheet.columns = [
{ header: 'Fecha', key: 'number_customer' },
{ header: 'Mensajes', key: 'message' },
{ header: 'Disparador', key: 'trigger' }
];
worksheet.addRow([today, message, trigger]);
workbook.xlsx.writeFile(pathExcel)
.then(() => {
console.log("saved");
})
.catch((err) => {
console.log("err", err);
});
}
}, 900)
number = cleanNumber(number)
await saveMessage( message, trigger, number )
console.log('Saved')
}
module.exports = { sendMessage, sendMedia, lastTrigger, sendMessageButton, readChat }
module.exports = { sendMessage, sendMedia, lastTrigger, sendMessageButton, readChat, sendMediaVoiceNote }

View File

@@ -2,6 +2,7 @@ const fs = require('fs')
const { sendMessage } = require('../controllers/send')
const sendMessagePost = (req, res) => {
console.log('asdasdasdasdasd')
const { message, number } = req.body
const client = req.clientWs || null;
sendMessage(client, number, message)

View File

@@ -21,8 +21,7 @@
"keywords": [
"cursos",
"info",
"curso"
],
"curso" ],
"key": "STEP_2"
},
{
@@ -69,20 +68,26 @@
},
{
"keywords": [
"👉 Youtube"
"youtube"
],
"key": "STEP_5"
},
{
"keywords": [
"😎 Cursos"
"VER_CURSOS"
],
"key": "STEP_6"
},
{
"keywords": [
"😁 Telegram"
"telegram"
],
"key": "STEP_7"
},
{
"keywords": [
"audio"
],
"key": "STEP_8"
}
]

View File

@@ -1,21 +1,15 @@
{
"DEFAULT":{
"replyMessage":[
"🆗 Bienvenido a este 🤖 CHATBOT de Whatsapp, lo primero \n",
"decirte que mi nombre es *Leifer Mendez*😎 y te dejo opciones rapidas \n"
"*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,
"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"}
]
}
"trigger":null
},
"STEP_0":{
"replyMessage":[
@@ -28,13 +22,23 @@
},
"STEP_1":{
"replyMessage":[
"✌️ Bienvenido a este 🤖 CHATBOT de Whatsapp, lo primero \n",
"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*"
"escribe *cursos* o *info* o escribe *audio*"
],
"media":null,
"trigger":null
"media":"https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif",
"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":[
@@ -106,9 +110,10 @@
},
"STEP_5":{
"replyMessage":[
"Muy bien te comparto el canal de Youtube \n"
"Muy bien te comparto el canal de Youtube \n",
"https://youtube.com/leifermendez \n"
],
"media":"https://youtube.com/leifermendez",
"media":null,
"trigger":null
},
"STEP_6":{
@@ -127,9 +132,17 @@
},
"STEP_7":{
"replyMessage":[
"Vente al telegram \n"
"Vente al telegram \n",
"https://t.me/leifermendez \n"
],
"media":"https://t.me/leifermendez",
"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.

BIN
mediaSend/nota-de-voz.mp3 Normal file

Binary file not shown.

0
middleware/db.js Normal file
View File

7667
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,37 +1,56 @@
{
"name": "test-ws-bot",
"name": "bot-whatsapp",
"version": "1.0.0",
"description": "",
"description": "Bot de wahtsapp open source para MVP o pequeños negocios",
"main": "app.js",
"scripts": {
"start": "node ./app.js",
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [],
"author": "",
"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": {
"type": "git",
"url": "https://github.com/leifermendez/bot-whatsapp"
},
"license": "ISC",
"dependencies": {
"@google-cloud/dialogflow": "^4.6.0",
"@google-cloud/dialogflow": "^5.2.0",
"cors": "^2.8.5",
"dotenv": "^11.0.0",
"dotenv": "^16.0.1",
"exceljs": "^4.3.0",
"express": "^4.17.2",
"file-type": "^16.5.3",
"mime-db": "^1.51.0",
"moment": "^2.29.1",
"express": "^4.18.1",
"file-type": "^17.1.6",
"mime-db": "^1.52.0",
"moment": "^2.29.4",
"mysql": "^2.18.1",
"nanoid": "^3.1.32",
"qr-image": "^3.2.0",
"qrcode-terminal": "^0.12.0",
"socket.io": "^4.4.1",
"whatsapp-web.js": "^1.15.4",
"xlsx": "^0.16.9"
"socket.io": "^4.5.1",
"stormdb": "^0.6.0",
"whatsapp-web.js": "^1.18.4",
"xlsx": "^0.18.5"
},
"devDependencies": {
"pm2": "^5.1.2",
"prettier": "2.5.1"
"pm2": "^5.2.0",
"prettier": "2.7.1"
},
"engines": {
"node": "14.x"
"node": "16.x"
}
}

View File

@@ -1,6 +1,6 @@
const express = require('express')
const router = express.Router();
const { sendMessagePost } = require('../controllers/web')
const { sendMessagePost } = require('../controllers/web')|
router.post('/send', sendMessagePost)