merge changes, to merge the updated versions
Merge branch 'el_lojas_melhorias' of github.com:AdrianoRobson/projeto-hit into el_lojas_melhoriasfeat-scaling-ticket-remote-creation
						commit
						cc12cafb99
					
				|  | @ -0,0 +1,8 @@ | ||||||
|  | PORT=8019  | ||||||
|  | PORT_START=8020 | ||||||
|  | BASE_URL=http://localhost   | ||||||
|  | PASS="strongpassword, strongpassword32"   | ||||||
|  | DB_MONGO_URL=mongodb://localhost:27017   | ||||||
|  | DB_MONGO_NAME=session_out_omnihit_db | ||||||
|  | DB_MONGO_NAME_CAMPAIGN=broker_omnihit | ||||||
|  |   | ||||||
|  | @ -0,0 +1,520 @@ | ||||||
|  | const express = require('express') | ||||||
|  | const bodyparser = require('body-parser') | ||||||
|  | const dotenv = require('dotenv') | ||||||
|  | dotenv.config({ path: '.env' }) | ||||||
|  | const copyFolder = require('./helpers/copyFolder') | ||||||
|  | const createDir = require('./helpers/createDir') | ||||||
|  | const createFile = require('./helpers/createFile') | ||||||
|  | const path = require('path') | ||||||
|  | const db_info = require('./db_conn') | ||||||
|  | const fs = require('fs') | ||||||
|  | let mysql_conn = require('./helpers/mysql_conn.js') | ||||||
|  | const { exec, execSync, spawn } = require('child_process') | ||||||
|  | 
 | ||||||
|  | const startPm2Process = require('./helpers/startPm2Process') | ||||||
|  | const removeDir = require('./helpers/remove_dir') | ||||||
|  | const setSessionName = require('./helpers/setSessionNumber') | ||||||
|  | const getNumberFromName = require('./helpers/getNumberSequence') | ||||||
|  | const pm2 = require('pm2') | ||||||
|  | const bcrypt = require('bcrypt') | ||||||
|  | const OmnihitDBConn = require('./model/db_conn') | ||||||
|  | 
 | ||||||
|  | const app = express() | ||||||
|  | 
 | ||||||
|  | app.use(bodyparser.json()) | ||||||
|  | 
 | ||||||
|  | app.get('/', function (req, res) { | ||||||
|  |   return res.send('Express + TypeScript Server') | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | app.post('/api/session', async function (req, res) { | ||||||
|  |   let { app_name, whatsappId, client_url, number } = req.body | ||||||
|  | 
 | ||||||
|  |   let oldNumber = '' | ||||||
|  | 
 | ||||||
|  |   if (app_name) { | ||||||
|  |     app_name = app_name.trim() | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   console.log('__dirname: ', path.join(__dirname, '..', app_name)) | ||||||
|  | 
 | ||||||
|  |   console.log( | ||||||
|  |     'app_name: ', | ||||||
|  |     app_name, | ||||||
|  |     ' | whatsappId: ', | ||||||
|  |     whatsappId, | ||||||
|  |     ' | client_url: ', | ||||||
|  |     client_url | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const sessionsPath = path.join(__dirname, '..', 'sessions') | ||||||
|  | 
 | ||||||
|  |   const directoriesInDIrectory = fs | ||||||
|  |     .readdirSync(sessionsPath, { withFileTypes: true }) | ||||||
|  |     .filter((item) => item.isDirectory()) | ||||||
|  |     .map((item) => item.name) | ||||||
|  | 
 | ||||||
|  |   console.log('directoriesInDIrectory: ', directoriesInDIrectory) | ||||||
|  | 
 | ||||||
|  |   const dirExist = directoriesInDIrectory.filter((e) => e.trim() == app_name) | ||||||
|  | 
 | ||||||
|  |   let dirSessionsApp = path.join(sessionsPath, app_name) | ||||||
|  | 
 | ||||||
|  |   if (dirExist.length == 0) { | ||||||
|  |     let create = createDir(dirSessionsApp) | ||||||
|  | 
 | ||||||
|  |     if (!create) { | ||||||
|  |       res.status(500).json({ message: 'Cannot create the directory path!' }) | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   let appPort = [] | ||||||
|  | 
 | ||||||
|  |   let existSubDir = false | ||||||
|  | 
 | ||||||
|  |   for (let i = 0; i < directoriesInDIrectory.length; i++) { | ||||||
|  |     console.log('directoriesInDIrectory[i]', directoriesInDIrectory[i]) | ||||||
|  | 
 | ||||||
|  |     const subDir = fs | ||||||
|  |       .readdirSync(path.join(sessionsPath, directoriesInDIrectory[i]), { | ||||||
|  |         withFileTypes: true, | ||||||
|  |       }) | ||||||
|  |       .filter((item) => item.isDirectory()) | ||||||
|  |       .map((item) => item.name) | ||||||
|  | 
 | ||||||
|  |     for (let x = 0; x < subDir.length; x++) { | ||||||
|  |       console.log('subdir: ', subDir[x]) | ||||||
|  | 
 | ||||||
|  |       let whatsId = subDir[x].split('_')[0] | ||||||
|  | 
 | ||||||
|  |       if (whatsId == whatsappId && app_name == directoriesInDIrectory[i]) { | ||||||
|  |         let currPath = path.join( | ||||||
|  |           sessionsPath, | ||||||
|  |           directoriesInDIrectory[i], | ||||||
|  |           subDir[x] | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         console.log( | ||||||
|  |           'PATH: ', | ||||||
|  |           path.join(sessionsPath, directoriesInDIrectory[i], subDir[x]) | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         oldNumber = subDir[x].split('_')[1] | ||||||
|  | 
 | ||||||
|  |         if (oldNumber != number) { | ||||||
|  |           deletePm2Process(subDir[x], currPath) | ||||||
|  | 
 | ||||||
|  |           removeDir(currPath) | ||||||
|  |         } else { | ||||||
|  |           res.send('ok') | ||||||
|  |           return | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       let auxPort = subDir[x].split('_')[3] | ||||||
|  | 
 | ||||||
|  |       console.log('---------> auxPort: ' + auxPort) | ||||||
|  | 
 | ||||||
|  |       if (auxPort) { | ||||||
|  |         auxPort = +auxPort.trim() | ||||||
|  | 
 | ||||||
|  |         if (!isNaN(auxPort)) { | ||||||
|  |           appPort.push(auxPort) | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       existSubDir = true | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   appPort = existSubDir ? Math.max(...appPort) + 1 : process.env.PORT_START | ||||||
|  | 
 | ||||||
|  |   console.log('new port: ', appPort) | ||||||
|  | 
 | ||||||
|  |   let dirSessionAppName | ||||||
|  | 
 | ||||||
|  |   let numberSession = 1 | ||||||
|  | 
 | ||||||
|  |   let lstPass = process.env.PASS | ||||||
|  | 
 | ||||||
|  |   if (!lstPass) { | ||||||
|  |     console.log('PASS VARIABLE NOT FOUND INTO .ENV!') | ||||||
|  |     return res.send('OK') | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   let db_credentials | ||||||
|  |   try { | ||||||
|  |     db_credentials = await OmnihitDBConn.findOne({ client_url }) | ||||||
|  | 
 | ||||||
|  |     if (!db_credentials) { | ||||||
|  |       db_credentials = new OmnihitDBConn({ | ||||||
|  |         client_url: client_url, | ||||||
|  |         db_conf: { | ||||||
|  |           DB: '', | ||||||
|  |           DB_HOST: '', | ||||||
|  |           DB_USER: '', | ||||||
|  |           DB_PASS: '', | ||||||
|  |           DB_PORT: '', | ||||||
|  |         }, | ||||||
|  |       }) | ||||||
|  |       await db_credentials.save() | ||||||
|  |       return res.send('ok') | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     console.log(error) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   if (db_credentials && db_credentials.db_conf.DB.trim().length > 0) { | ||||||
|  |     lstPass = lstPass.split(',') | ||||||
|  |     let password = null | ||||||
|  | 
 | ||||||
|  |     // password = await lstPass.find(
 | ||||||
|  |     //   async (pass) =>
 | ||||||
|  |     //     await bcrypt.compare(pass.trim(), db_credentials.db_conf.DB_PASS)
 | ||||||
|  |     // )
 | ||||||
|  | 
 | ||||||
|  |     for (let i = 0; i < lstPass.length; i++) { | ||||||
|  |       const hasPass = await bcrypt.compare( | ||||||
|  |         lstPass[i].trim(), | ||||||
|  |         db_credentials.db_conf.DB_PASS | ||||||
|  |       ) | ||||||
|  | 
 | ||||||
|  |       if (hasPass) { | ||||||
|  |         password = lstPass[i].trim() | ||||||
|  |         break | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (password) { | ||||||
|  |       db_credentials.db_conf.DB_PASS = password | ||||||
|  |     } else { | ||||||
|  |       return res.send('ok') | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   if ( | ||||||
|  |     db_credentials.db_conf && | ||||||
|  |     Object.keys(db_credentials.db_conf).length > 0 | ||||||
|  |   ) { | ||||||
|  |     const whatsapp_numbers = await new Promise((resolve, reject) => { | ||||||
|  |       mysql_conn(db_credentials.db_conf).query( | ||||||
|  |         'SELECT name, number FROM Whatsapps WHERE name LIKE ?', | ||||||
|  |         [`%${number}%`], | ||||||
|  |         (err, result) => { | ||||||
|  |           if (err) { | ||||||
|  |             reject(err) | ||||||
|  |           } else { | ||||||
|  |             resolve(result) | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     let session_num = [] | ||||||
|  | 
 | ||||||
|  |     if (whatsapp_numbers && whatsapp_numbers.length > 0) { | ||||||
|  |       console.log('whatsapp_numbers.length: ', whatsapp_numbers.length) | ||||||
|  | 
 | ||||||
|  |       if (whatsapp_numbers.length == 5) { | ||||||
|  |         res.status(400).json({ | ||||||
|  |           message: 'Cannot create more than 4 sessions from the same number', | ||||||
|  |         }) | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       let regex = /-> [a-zA-Z]\d$/ | ||||||
|  | 
 | ||||||
|  |       let numbered_sessions = whatsapp_numbers.filter((e) => regex.test(e.name)) | ||||||
|  | 
 | ||||||
|  |       console.log('numbered_sessions: ', numbered_sessions) | ||||||
|  | 
 | ||||||
|  |       session_num = numbered_sessions.map((e) => | ||||||
|  |         parseInt( | ||||||
|  |           e.name | ||||||
|  |             .split('->') | ||||||
|  |           [e.name.split('->').length - 1].trim() | ||||||
|  |             .match(/\d+/)[0] | ||||||
|  |         ) | ||||||
|  |       ) | ||||||
|  | 
 | ||||||
|  |       console.log('session_num', session_num) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let index = 1 | ||||||
|  | 
 | ||||||
|  |     while (index <= 4) { | ||||||
|  |       if (!session_num.includes(index)) { | ||||||
|  |         console.log(index) | ||||||
|  |         numberSession = index | ||||||
|  |         break | ||||||
|  |       } | ||||||
|  |       index++ | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // numberSession = Math.max(...session_number) + 1
 | ||||||
|  | 
 | ||||||
|  |   console.log('Number session: ', numberSession) | ||||||
|  | 
 | ||||||
|  |   // }
 | ||||||
|  | 
 | ||||||
|  |   dirSessionAppName = `${whatsappId}_${number}_${numberSession}_${appPort}` | ||||||
|  | 
 | ||||||
|  |   destDir = path.join(dirSessionsApp, dirSessionAppName) | ||||||
|  | 
 | ||||||
|  |   originDir = path.join(__dirname, '..', 'whats') | ||||||
|  | 
 | ||||||
|  |   copyFolder(originDir, destDir) | ||||||
|  | 
 | ||||||
|  |   if ( | ||||||
|  |     db_credentials.db_conf && | ||||||
|  |     Object.keys(db_credentials.db_conf).length > 0 | ||||||
|  |   ) { | ||||||
|  |     console.log('***SUCCESS SEED DIRECTORY CREATED***') | ||||||
|  | 
 | ||||||
|  |     let whatsName | ||||||
|  | 
 | ||||||
|  |     const whatsapp = await new Promise((resolve, reject) => { | ||||||
|  |       mysql_conn(db_credentials.db_conf).query( | ||||||
|  |         'SELECT name from Whatsapps where id = ?', | ||||||
|  |         [whatsappId], | ||||||
|  |         (err, result) => { | ||||||
|  |           if (err) { | ||||||
|  |             reject(err) | ||||||
|  |           } else { | ||||||
|  |             resolve(result) | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     if (whatsapp[0]['name']?.split('->')?.length > 0) { | ||||||
|  |       whatsName = `${whatsapp[0]['name'].split('->')[0]} -> S${numberSession}` | ||||||
|  |     } else { | ||||||
|  |       whatsName = `${whatsapp[0]['name']} -> S${numberSession}` | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     console.log('whatsName: ', whatsName) | ||||||
|  | 
 | ||||||
|  |     console.log( | ||||||
|  |       `url: ${process.env.BASE_URL}:${appPort}\n whatsname: ${whatsName}\n whatsappId: ${whatsappId}` | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     await new Promise((resolve, reject) => { | ||||||
|  |       mysql_conn(db_credentials.db_conf).query( | ||||||
|  |         'UPDATE Whatsapps SET url = ?, name = ? where id = ?', | ||||||
|  |         [`${process.env.BASE_URL}:${appPort}`, `${whatsName}`, whatsappId], | ||||||
|  | 
 | ||||||
|  |         function (err, result) { | ||||||
|  |           if (err) { | ||||||
|  |             reject(err) | ||||||
|  |             console.log('===> ERROR: ' + err) | ||||||
|  |           } else { | ||||||
|  |             resolve(result) | ||||||
|  |             // console.log('RESULT: ', result)
 | ||||||
|  |           } | ||||||
|  |           // else
 | ||||||
|  |           //     console.log('myslq result: ', result);
 | ||||||
|  |         } | ||||||
|  |       ) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     let whatsappName = `${number} - s${numberSession}` | ||||||
|  | 
 | ||||||
|  |     console.log('-------------- numberSession', numberSession) | ||||||
|  | 
 | ||||||
|  |     if (whatsapp.length > 0) { | ||||||
|  |       if (whatsapp[0]['name'].split(' ').length > 0) { | ||||||
|  |         whatsappName = `${whatsapp[0]['name'].split(' ')[0] | ||||||
|  |           } - S${numberSession}` | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     console.log('whatsapp: ', whatsapp) | ||||||
|  |     console.log("whatsapp[0]['name']: ", whatsapp[0]['name']) | ||||||
|  | 
 | ||||||
|  |     const keys = Object.keys(db_credentials.db_conf) | ||||||
|  | 
 | ||||||
|  |     var stream = fs.createWriteStream(path.join(destDir, '.env')) | ||||||
|  |     stream.once('open', function (fd) { | ||||||
|  |       stream.write('# NUMBER AND NAME THAT WILL BE DISPLAYED ON CONSOLE\n') | ||||||
|  |       stream.write(`MOBILEUID=${number}\n`) | ||||||
|  |       stream.write(`MOBILENAME=${whatsappName}\n`) | ||||||
|  |       stream.write(`OLD_MOBILEUID=${oldNumber}\n`) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.write('# PORT NUMBER FOR THIS API\n') | ||||||
|  |       stream.write(`PORT=${appPort}\n`) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.write('# URL FROM THE OMNIHIT BACKEND API\n') | ||||||
|  |       stream.write(`CLIENT_URL=${client_url}\n`) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.write('# OMNIHIT DATABASE\n') | ||||||
|  |       keys.forEach((key, index) => { | ||||||
|  |         stream.write(`${key}=${db_credentials.db_conf[key]}\n`) | ||||||
|  |       }) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.write( | ||||||
|  |         `# WHATSAPP ID OF THE TABLE Whatsapps FROM THE OMNIHIT DATABASE\n` | ||||||
|  |       ) | ||||||
|  |       stream.write(`WHATSAPP_ID=${whatsappId}\n`) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.write('# MONGO CONNECTION\n') | ||||||
|  |       stream.write(`DB_MONGO_URL=${process.env.DB_MONGO_URL}\n`) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.write('# MONGO COLLECTION\n') | ||||||
|  |       stream.write(`DB_MONGO_NAME=${process.env.DB_MONGO_NAME_CAMPAIGN}\n`) | ||||||
|  |       stream.write('\n') | ||||||
|  | 
 | ||||||
|  |       stream.end() | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     console.log('----------------destDir: ', destDir) | ||||||
|  | 
 | ||||||
|  |     execSync(`npm install`, { cwd: destDir }, (error, stdout, stderr) => { | ||||||
|  |       if (error) { | ||||||
|  |         console.log(`error: ${error.message}`) | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       if (stderr) { | ||||||
|  |         console.log(`stderr: ${stderr}`) | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       console.log(`stdout: ${stdout}`) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     const env = { | ||||||
|  |       PORT: appPort, | ||||||
|  |       DB_MONGO_URL: process.env.DB_MONGO_URL, | ||||||
|  |       DB_MONGO_NAME: process.env.DB_MONGO_NAME_CAMPAIGN | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     startPm2Process(dirSessionAppName, 'app.js', destDir, env) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   res.send('OK') | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | app.post('/api/session/edit', async function (req, res) { | ||||||
|  |   const { app_name, whatsappId, client_url, number } = req.body | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | app.post('/api/session/del', async function (req, res) { | ||||||
|  |   let { whatsappId, app_name } = req.body | ||||||
|  | 
 | ||||||
|  |   if (app_name) { | ||||||
|  |     app_name = app_name.trim() | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const sessionsPath = path.join(__dirname, '..', 'sessions') | ||||||
|  | 
 | ||||||
|  |   const directoriesInDIrectory = fs | ||||||
|  |     .readdirSync(sessionsPath, { withFileTypes: true }) | ||||||
|  |     .filter((item) => item.isDirectory()) | ||||||
|  |     .map((item) => item.name) | ||||||
|  | 
 | ||||||
|  |   console.log('directoriesInDIrectory: ', directoriesInDIrectory) | ||||||
|  | 
 | ||||||
|  |   const dirExist = directoriesInDIrectory.filter((e) => e.trim() == app_name) | ||||||
|  | 
 | ||||||
|  |   console.log('dirExist: ', dirExist) | ||||||
|  | 
 | ||||||
|  |   if (dirExist.length == 0) res.send('ok') | ||||||
|  | 
 | ||||||
|  |   for (let i = 0; i < directoriesInDIrectory.length; i++) { | ||||||
|  |     console.log('directoriesInDIrectory[i]', directoriesInDIrectory[i]) | ||||||
|  | 
 | ||||||
|  |     const subDir = fs | ||||||
|  |       .readdirSync(path.join(sessionsPath, directoriesInDIrectory[i]), { | ||||||
|  |         withFileTypes: true, | ||||||
|  |       }) | ||||||
|  |       .filter((item) => item.isDirectory()) | ||||||
|  |       .map((item) => item.name) | ||||||
|  | 
 | ||||||
|  |     for (let x = 0; x < subDir.length; x++) { | ||||||
|  |       console.log('subdir: ', subDir[x]) | ||||||
|  | 
 | ||||||
|  |       let whatsId = subDir[x].split('_')[0] | ||||||
|  | 
 | ||||||
|  |       if (whatsId == whatsappId && app_name == directoriesInDIrectory[i]) { | ||||||
|  |         let currPath = path.join( | ||||||
|  |           sessionsPath, | ||||||
|  |           directoriesInDIrectory[i], | ||||||
|  |           subDir[x] | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         deletePm2Process(subDir[x], currPath) | ||||||
|  | 
 | ||||||
|  |         console.log('currPath: ', currPath) | ||||||
|  | 
 | ||||||
|  |         removeDir(currPath) | ||||||
|  | 
 | ||||||
|  |         return res.send('ok') | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   res.send('ok') | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | app.listen(process.env.PORT || 8003, function () { | ||||||
|  |   console.log( | ||||||
|  |     '\u26A1[server]: Server is running at Port ::: ' + process.env.PORT || 8003 | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | process.on('uncaughtException', function (err) { | ||||||
|  |   console.error(' ') | ||||||
|  |   console.error( | ||||||
|  |     '----- ' + new Date().toUTCString() + ' ----------------------------------' | ||||||
|  |   ) | ||||||
|  |   console.error('Erro uncaughtException: ', err.message) | ||||||
|  |   console.error(err.stack) | ||||||
|  |   console.error(' ') | ||||||
|  |   return | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | function deletePm2Process(process_name, currPath) { | ||||||
|  |   pm2.connect(function (err) { | ||||||
|  |     if (err) { | ||||||
|  |       console.error(err) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pm2.list(function (err, processes) { | ||||||
|  |       if (err) { | ||||||
|  |         console.error(err) | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       processes.forEach(function (process) { | ||||||
|  |         console.log('.........process.name: ', process.name) | ||||||
|  | 
 | ||||||
|  |         if (process.name === process_name) { | ||||||
|  |           execSync( | ||||||
|  |             `pm2 delete ${process_name} && pm2 save --force`, | ||||||
|  |             { cwd: currPath }, | ||||||
|  |             (error, stdout, stderr) => { | ||||||
|  |               if (error) { | ||||||
|  |                 console.log(`error: ${error.message}`) | ||||||
|  |                 return | ||||||
|  |               } | ||||||
|  |               if (stderr) { | ||||||
|  |                 console.log(`stderr: ${stderr}`) | ||||||
|  |                 return | ||||||
|  |               } | ||||||
|  |               console.log(`stdout: ${stdout}`) | ||||||
|  |             } | ||||||
|  |           ) | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  | 
 | ||||||
|  |       pm2.disconnect() | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | @ -0,0 +1,11 @@ | ||||||
|  | const mongoose = require('mongoose') | ||||||
|  | require('dotenv').config({ path: `${process.cwd()}/.env` }) | ||||||
|  | 
 | ||||||
|  | async function main(){  | ||||||
|  |     await mongoose.connect(process.env.DB_MONGO_URL,  { dbName: process.env.DB_MONGO_NAME }) | ||||||
|  |     console.log('Conectou ao Mongoose!') | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | main().catch((err)=>console.log(err)) | ||||||
|  | 
 | ||||||
|  | module.exports = mongoose | ||||||
|  | @ -0,0 +1,27 @@ | ||||||
|  | const db = [ | ||||||
|  | 
 | ||||||
|  |     { | ||||||
|  |         client_url: "http://localhost:8080", | ||||||
|  |         db_conf: { | ||||||
|  |             DB: "whaticket", | ||||||
|  |             DB_HOST: "localhost", | ||||||
|  |             DB_USER: "whaticket", | ||||||
|  |             DB_PASS: "strongpassword", | ||||||
|  |             DB_PORT: "3306" | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     { | ||||||
|  |         client_url: "http://localhost:8081", | ||||||
|  |         db_conf: { | ||||||
|  |             DB: "whaticket", | ||||||
|  |             DB_HOST: "localhost", | ||||||
|  |             DB_USER: "whaticket", | ||||||
|  |             DB_PASS: "strongpassword", | ||||||
|  |             DB_PORT: "3306" | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | ] | ||||||
|  |   | ||||||
|  | module.exports = db; | ||||||
|  | @ -0,0 +1,20 @@ | ||||||
|  | const bcrypt = require('bcrypt') | ||||||
|  | 
 | ||||||
|  | // const pass = async () => {
 | ||||||
|  | //   //    create a password
 | ||||||
|  | //   const salt = await bcrypt.genSalt(12)
 | ||||||
|  | //   const passwordHash = await bcrypt.hash('7901228899', salt)
 | ||||||
|  | //   console.log(passwordHash)
 | ||||||
|  | // }
 | ||||||
|  | // pass()
 | ||||||
|  | 
 | ||||||
|  | const passDec = async () => { | ||||||
|  |   const _pass = await bcrypt.compare( | ||||||
|  |     'strongpassword', | ||||||
|  |     '$2b$12$PZ8N1jU77nnNUCCGyKTMNOi2QI7X/SgPsISVQfr.cQ/jgdx5Z7AqC' | ||||||
|  |   ) | ||||||
|  |   console.log('_pass: ', _pass) | ||||||
|  | } | ||||||
|  | passDec() | ||||||
|  | 
 | ||||||
|  | console.log('process.cwd(): ', process.cwd()) | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | const fsPromises = require("fs/promises"); | ||||||
|  | const fs = require('fs-extra') | ||||||
|  | 
 | ||||||
|  | // Delete a directory and its children 
 | ||||||
|  | function copyFolder(sourcePath, destPath) { | ||||||
|  | 
 | ||||||
|  |     fs.copySync(sourcePath, destPath, { overwrite: true }, (err) => { | ||||||
|  |         if (err) { | ||||||
|  |             console.error(err); | ||||||
|  |         } else { | ||||||
|  |             console.log("Copy dir success!"); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = copyFolder; | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | const fs = require('fs'); | ||||||
|  | 
 | ||||||
|  | function createDir(dir) { | ||||||
|  | 
 | ||||||
|  |     // create new directory
 | ||||||
|  |     try { | ||||||
|  |         // check if directory already exists
 | ||||||
|  |         if (!fs.existsSync(dir)) { | ||||||
|  |             fs.mkdirSync(dir); | ||||||
|  |             console.log("Directory is created."); | ||||||
|  |         } else { | ||||||
|  |             console.log("Directory already exists."); | ||||||
|  |         } | ||||||
|  |     } catch (err) { | ||||||
|  |         console.log(err); | ||||||
|  |         return false | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return true | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = createDir; | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | const fs = require('fs'); | ||||||
|  | 
 | ||||||
|  | function createFile(saveDir, rows, fileName) { | ||||||
|  | 
 | ||||||
|  |     var stream = fs.createWriteStream(path.join(saveDir, fileName)); | ||||||
|  | 
 | ||||||
|  |     stream.once('open', function (fd) { | ||||||
|  | 
 | ||||||
|  |         rows.forEach(element => { | ||||||
|  |             stream.write(element); | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         stream.end(); | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = createFile | ||||||
|  | @ -0,0 +1,46 @@ | ||||||
|  | const pm2 = require('pm2'); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | function findAndDeletePm2Process(pm2_process_name) { | ||||||
|  | 
 | ||||||
|  |     pm2.connect(function (err) { | ||||||
|  |         if (err) { | ||||||
|  |             console.error(err); | ||||||
|  |             // process.exit(2); 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         pm2.list(function (err, processes) { | ||||||
|  |             if (err) { | ||||||
|  |                 console.error(err); | ||||||
|  |                 // process.exit(2); 
 | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const processToDelete = processes.find(process => process.name === pm2_process_name); | ||||||
|  | 
 | ||||||
|  |             if (!processToDelete) { | ||||||
|  |                 console.error('Process not found'); | ||||||
|  |                 // process.exit(2); 
 | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  | 
 | ||||||
|  |                 pm2.delete(processToDelete.pm_id, function (err) { | ||||||
|  |                     if (err) { | ||||||
|  |                         console.error(err); | ||||||
|  |                         // process.exit(2);
 | ||||||
|  |                     } | ||||||
|  |                     else{ | ||||||
|  |                         console.log(`Process deleted: ${pm2_process_name}`) | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     pm2.disconnect(); | ||||||
|  |                 }); | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = findAndDeletePm2Process | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | 
 | ||||||
|  | function getNumberFromName(name) { | ||||||
|  | 
 | ||||||
|  |     name = name + ' ' | ||||||
|  | 
 | ||||||
|  |     let number = name.split('') | ||||||
|  | 
 | ||||||
|  |     let newNumber = '' | ||||||
|  |     let list = [] | ||||||
|  | 
 | ||||||
|  |     for (let i = 0; i < number.length; i++) { | ||||||
|  | 
 | ||||||
|  |         if (!isNaN(Number(number[i])) && number[i].trim().length > 0) { | ||||||
|  |             newNumber += number[i] | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             if (!isNaN(Number(newNumber)) && newNumber.trim().length > 0) { | ||||||
|  |                 list.push(newNumber) | ||||||
|  |             } | ||||||
|  |             newNumber = '' | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // longestString = list.filter(str => str.length === Math.max(...list.map(s => s.length)))[0];
 | ||||||
|  | 
 | ||||||
|  |     // console.log('list: ', list)
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     let longestString = ""; // variable to store the longest string
 | ||||||
|  |     for (let i = 0; i < list.length; i++) { | ||||||
|  |         if (list[i].length > longestString.length) { | ||||||
|  |             longestString = list[i]; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return longestString | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = getNumberFromName; | ||||||
|  | @ -0,0 +1,109 @@ | ||||||
|  | const dotenv = require('dotenv'); | ||||||
|  | dotenv.config({ path: `${process.cwd()}/.env` }); | ||||||
|  | const path = require('path') | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | function mysql_conn(config) { | ||||||
|  |     // Ubicua Plataform - MYSQL Module
 | ||||||
|  |     try { | ||||||
|  |         var mysql_npm = require('mysql'); | ||||||
|  |     } catch (err) { | ||||||
|  |         console.log("Cannot find `mysql` module. Is it installed ? Try `npm install mysql` or `npm install`."); | ||||||
|  |     } | ||||||
|  |   | ||||||
|  |     var db_config = { | ||||||
|  |         host: config.DB_HOST, | ||||||
|  |         user: config.DB_USER, | ||||||
|  |         password: config.DB_PASS, | ||||||
|  |         database: config.DB, | ||||||
|  |         charset: 'utf8mb4_general_ci', | ||||||
|  |         port: config.DB_PORT | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     //-
 | ||||||
|  |     //- Create the connection variable
 | ||||||
|  |     //-
 | ||||||
|  |     var connection = mysql_npm.createPool(db_config);  | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     //-
 | ||||||
|  |     //- Establish a new connection
 | ||||||
|  |     //-
 | ||||||
|  |     connection.getConnection(function (err) { | ||||||
|  |         if (err) { | ||||||
|  |             // mysqlErrorHandling(connection, err);
 | ||||||
|  |             console.log("\n\t *** Cannot establish a connection with the database. ***"); | ||||||
|  | 
 | ||||||
|  |             connection = reconnect(connection); | ||||||
|  |         } else { | ||||||
|  |             console.log("\n\t *** New connection established with the database. ***") | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     //-
 | ||||||
|  |     //- Reconnection function
 | ||||||
|  |     //-
 | ||||||
|  |     function reconnect(connection) { | ||||||
|  |         console.log("\n New connection tentative..."); | ||||||
|  | 
 | ||||||
|  |         //- Create a new one
 | ||||||
|  |         connection = mysql_npm.createPool(db_config); | ||||||
|  | 
 | ||||||
|  |         //- Try to reconnect
 | ||||||
|  |         connection.getConnection(function (err) { | ||||||
|  |             if (err) { | ||||||
|  |                 //- Try to connect every 2 seconds.
 | ||||||
|  |                 setTimeout(reconnect(connection), 2000); | ||||||
|  |             } else { | ||||||
|  |                 console.log("\n\t *** New connection established with the database. ***") | ||||||
|  |                 return connection; | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     //-
 | ||||||
|  |     //- Error listener
 | ||||||
|  |     //-
 | ||||||
|  |     connection.on('error', function (err) { | ||||||
|  | 
 | ||||||
|  |         //-
 | ||||||
|  |         //- The server close the connection.
 | ||||||
|  |         //-
 | ||||||
|  |         if (err.code === "PROTOCOL_CONNECTION_LOST") { | ||||||
|  |             console.log("/!\\ Cannot establish a connection with the database. /!\\ (" + err.code + ")"); | ||||||
|  |             return reconnect(connection); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         else if (err.code === "PROTOCOL_ENQUEUE_AFTER_QUIT") { | ||||||
|  |             console.log("/!\\ Cannot establish a connection with the database. /!\\ (" + err.code + ")"); | ||||||
|  |             return reconnect(connection); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         else if (err.code === "PROTOCOL_ENQUEUE_AFTER_FATAL_ERROR") { | ||||||
|  |             console.log("/!\\ Cannot establish a connection with the database. /!\\ (" + err.code + ")"); | ||||||
|  |             return reconnect(connection); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         else if (err.code === "PROTOCOL_ENQUEUE_HANDSHAKE_TWICE") { | ||||||
|  |             console.log("/!\\ Cannot establish a connection with the database. /!\\ (" + err.code + ")"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         else { | ||||||
|  |             console.log("/!\\ Cannot establish a connection with the database. /!\\ (" + err.code + ")"); | ||||||
|  |             return reconnect(connection); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     return connection | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Export
 | ||||||
|  | //-
 | ||||||
|  | module.exports = mysql_conn; | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | import axios from "axios";  | ||||||
|  | 
 | ||||||
|  | async function postData(url, body = {}) { | ||||||
|  | 
 | ||||||
|  |     let response; | ||||||
|  | 
 | ||||||
|  |     try { | ||||||
|  |         response = await axios.post(url, body); | ||||||
|  |         console.log(response.data); // handle successful response
 | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error(error); // handle error
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return response | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = postData | ||||||
|  | @ -0,0 +1,28 @@ | ||||||
|  | const fsPromises = require("fs/promises"); | ||||||
|  | const fs = require('fs') | ||||||
|  | 
 | ||||||
|  | // Delete a directory and its children 
 | ||||||
|  | const removeDir = async (dirPath) => { | ||||||
|  | 
 | ||||||
|  |     if (fs.existsSync(dirPath)) { | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             await fsPromises.rm(dirPath, { recursive: true, force: true }); | ||||||
|  |             console.log("Directory removed!"); | ||||||
|  | 
 | ||||||
|  |             return true | ||||||
|  |         } | ||||||
|  |         catch (err) { | ||||||
|  |             console.log('An error occurred while removing the directory: ', err);    | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |     else {  | ||||||
|  |         console.log('Directory not found to remove: ', dirPath)  | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return false | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = removeDir ; | ||||||
|  | @ -0,0 +1,152 @@ | ||||||
|  | 
 | ||||||
|  | import os from 'os'; | ||||||
|  | import dir from 'path'; | ||||||
|  | import fs from 'fs'; | ||||||
|  | 
 | ||||||
|  | export const setJSON = (obj) => { | ||||||
|  | 
 | ||||||
|  |     const sessionControlFile = dir.join(process.cwd(), `sessionsDir.json`); | ||||||
|  | 
 | ||||||
|  |     try { | ||||||
|  | 
 | ||||||
|  |         if (fs.existsSync(sessionControlFile)) { | ||||||
|  | 
 | ||||||
|  |             const sessionsDir = fs.readFileSync(sessionControlFile, { encoding: 'utf8', flag: 'r' }); | ||||||
|  | 
 | ||||||
|  |             let lstRestore = JSON.parse(sessionsDir) | ||||||
|  | 
 | ||||||
|  |             lstRestore.push(obj) | ||||||
|  | 
 | ||||||
|  |             fs.writeFileSync(sessionControlFile, JSON.stringify(lstRestore), "utf8");  | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  | 
 | ||||||
|  |             console.log('sessionsDir.json file not found! It will be created.'); | ||||||
|  | 
 | ||||||
|  |             if (Array.isArray(obj)) { | ||||||
|  | 
 | ||||||
|  |                 fs.writeFileSync(sessionControlFile, JSON.stringify(obj), "utf8"); | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  | 
 | ||||||
|  |                 fs.writeFileSync(sessionControlFile, JSON.stringify([obj]), "utf8"); | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } catch (error) { | ||||||
|  |         console.log('There was an error on try to read the sessionsDir.json file: ', error) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const shifRestoreControll = () => { | ||||||
|  | 
 | ||||||
|  |     const sessionControlFile = dir.join(os.tmpdir(), `sessionsDir.json`); | ||||||
|  | 
 | ||||||
|  |     try { | ||||||
|  | 
 | ||||||
|  |         if (fs.existsSync(sessionControlFile)) { | ||||||
|  | 
 | ||||||
|  |             const sessionsDir = fs.readFileSync(sessionControlFile, { encoding: 'utf8', flag: 'r' }); | ||||||
|  | 
 | ||||||
|  |             let lstRestore: any = JSON.parse(sessionsDir) | ||||||
|  | 
 | ||||||
|  |             let whatsapp: any = lstRestore.shift() | ||||||
|  | 
 | ||||||
|  |             fs.writeFileSync(sessionControlFile, JSON.stringify(lstRestore), "utf8"); | ||||||
|  | 
 | ||||||
|  |             return whatsapp | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } catch (error) { | ||||||
|  |         console.log('There was an error on try to read the sessionsDir.json file: ', error) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return {} | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const delRestoreControllFile = () => { | ||||||
|  | 
 | ||||||
|  |     const sessionControlFile = dir.join(os.tmpdir(), `sessionsDir.json`); | ||||||
|  | 
 | ||||||
|  |     try { | ||||||
|  | 
 | ||||||
|  |         if (fs.existsSync(sessionControlFile)) { | ||||||
|  | 
 | ||||||
|  |             fs.unlinkSync(sessionControlFile) | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  | 
 | ||||||
|  |             console.log('sessionsDir.json file not found!'); | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } catch (error) { | ||||||
|  |         console.log('There was an error on try delete the sessionsDir.json file: ', error) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export const getRestoreControll = () => { | ||||||
|  | 
 | ||||||
|  |     const sessionControlFile = dir.join(os.tmpdir(), `sessionsDir.json`); | ||||||
|  | 
 | ||||||
|  |     try { | ||||||
|  | 
 | ||||||
|  |         if (fs.existsSync(sessionControlFile)) { | ||||||
|  | 
 | ||||||
|  |             const sessionsDir = fs.readFileSync(sessionControlFile, { encoding: 'utf8', flag: 'r' }); | ||||||
|  | 
 | ||||||
|  |             let lstRestore: any = JSON.parse(sessionsDir) | ||||||
|  | 
 | ||||||
|  |             return lstRestore | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  | 
 | ||||||
|  |             console.log('sessionsDir.json file not found!'); | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } catch (error) { | ||||||
|  |         console.log('There was an error on try to read the sessionsDir.json file: ', error) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return [] | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export const _restore = async (whatsapp: Whatsapp, msg_file_title: string) => { | ||||||
|  | 
 | ||||||
|  |     return | ||||||
|  | 
 | ||||||
|  |     if (whatsapp.status != 'RESTORING' && whatsapp.status != 'qrcode') { | ||||||
|  | 
 | ||||||
|  |         console.log('THE WHATSAAP ID: ', whatsapp.id, ' WILL BE RESTORED SOON!') | ||||||
|  | 
 | ||||||
|  |         await whatsapp.update({ status: "RESTORING", }); | ||||||
|  | 
 | ||||||
|  |         const io = getIO(); | ||||||
|  | 
 | ||||||
|  |         io.emit("whatsappSession", { | ||||||
|  |             action: "update", | ||||||
|  |             session: whatsapp | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // await insertOrUpeateWhatsCache(`whatsapp:${whatsapp.id}`, { status: "RESTORING", })
 | ||||||
|  | 
 | ||||||
|  |         setTimeout(async () => await autoRestore(whatsapp.id, msg_file_title), 95000); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | }  | ||||||
|  | @ -0,0 +1,105 @@ | ||||||
|  | const fs = require('fs'); | ||||||
|  | let mysql_conn = require('./mysql_conn.js'); | ||||||
|  | 
 | ||||||
|  | async function sessionNumber(db_info, whatsappId, number, dirSessionsApp) { | ||||||
|  | 
 | ||||||
|  |     let numberSession = 1 | ||||||
|  | 
 | ||||||
|  |     let whatsappName | ||||||
|  | 
 | ||||||
|  |     const dirSessionsNumberAppDirectories = fs.readdirSync(dirSessionsApp, { withFileTypes: true }) | ||||||
|  |         .filter((item) => item.isDirectory() && item.name.includes(`${number}`)) | ||||||
|  |         .map((item) => item.name); | ||||||
|  | 
 | ||||||
|  |     if (dirSessionsNumberAppDirectories.length > 0) { | ||||||
|  | 
 | ||||||
|  |         let session_number = dirSessionsNumberAppDirectories.map((e) => +e.split('_')[2]) | ||||||
|  | 
 | ||||||
|  |         numberSession = Math.max(...session_number) + 1 | ||||||
|  | 
 | ||||||
|  |         console.log('Number session: ', numberSession) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         if (numberSession > 4) { | ||||||
|  |             res.status(400).json({ message: 'Cannot create more than 4 sessions from the same number' }) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     let db = db_info.filter((e) => e.client_url == client_url) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     if (db && db.length > 0) { | ||||||
|  | 
 | ||||||
|  |         db = db[0].db_conf | ||||||
|  | 
 | ||||||
|  |         let whatsName | ||||||
|  | 
 | ||||||
|  |         const whatsapp = await new Promise((resolve, reject) => { | ||||||
|  | 
 | ||||||
|  |             mysql_conn(db).query("SELECT name from Whatsapps where id = ?", [whatsappId], (err, result) => { | ||||||
|  | 
 | ||||||
|  |                 if (err) { | ||||||
|  |                     reject(err) | ||||||
|  |                 } | ||||||
|  |                 else { | ||||||
|  |                     resolve(result) | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |         }) | ||||||
|  | 
 | ||||||
|  |         if (whatsapp[0]["name"].split('->').length > 0) { | ||||||
|  |             whatsName = `${whatsapp[0]["name"].split('->')[0]} -> S${numberSession}` | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             whatsName = `${whatsapp[0]["name"]} -> S${numberSession}` | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         console.log('whatsName: ', whatsName) | ||||||
|  | 
 | ||||||
|  |         console.log(`url: ${process.env.BASE_URL}:${appPort}\n whatsname: ${whatsName}\n whatsappId: ${whatsappId}`) | ||||||
|  | 
 | ||||||
|  |         // await new Promise((resolve, reject) => {
 | ||||||
|  |         //     mysql_conn(db).query("UPDATE Whatsapps SET name = ? where id = ?", [ `${whatsName}`, whatsappId],
 | ||||||
|  | 
 | ||||||
|  |         //         function (err, result) {
 | ||||||
|  |         //             if (err) {
 | ||||||
|  |         //                 reject(err)
 | ||||||
|  |         //                 console.log("===> ERROR: " + err);
 | ||||||
|  |         //             }
 | ||||||
|  |         //             else {
 | ||||||
|  |         //                 resolve(result)
 | ||||||
|  |         //                 console.log('RESULT: ', result)
 | ||||||
|  |         //             }
 | ||||||
|  |         //             // else
 | ||||||
|  |         //             //     console.log('myslq result: ', result);
 | ||||||
|  |         //         });
 | ||||||
|  |         // })
 | ||||||
|  | 
 | ||||||
|  |         whatsappName = `${number} - s${numberSession}` | ||||||
|  | 
 | ||||||
|  |         console.log('-------------- numberSession', numberSession) | ||||||
|  | 
 | ||||||
|  |         if (whatsapp.length > 0) { | ||||||
|  | 
 | ||||||
|  |             if (whatsapp[0]['name'].split(' ').length > 0) { | ||||||
|  | 
 | ||||||
|  |                 whatsappName = `${whatsapp[0]['name'].split(' ')[0]} - S${numberSession}` | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     console.log('---------> whatsappName', whatsappName) | ||||||
|  | 
 | ||||||
|  |     return whatsappName | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = sessionNumber | ||||||
|  | @ -0,0 +1,50 @@ | ||||||
|  | const pm2 = require('pm2') | ||||||
|  | const { execSync } = require("child_process") | ||||||
|  | 
 | ||||||
|  | function startPm2Process(process_name, file, path, env) { | ||||||
|  | 
 | ||||||
|  |     pm2.connect(function (err) { | ||||||
|  |         if (err) { | ||||||
|  |             console.error(err) | ||||||
|  |             // process.exit(2);
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         console.log('ENV PM2: ', env) | ||||||
|  | 
 | ||||||
|  |         pm2.start({ | ||||||
|  |             name: process_name, | ||||||
|  |             script: file, | ||||||
|  |             cwd: path, | ||||||
|  |             env | ||||||
|  |             // env: {
 | ||||||
|  |             //     NODE_ENV: 'production',
 | ||||||
|  | 
 | ||||||
|  |             //     PORT: port, 
 | ||||||
|  |             // }
 | ||||||
|  |             // additional options here if needed
 | ||||||
|  |         }, function (err, apps) { | ||||||
|  |             if (err) { | ||||||
|  |                 console.error(err) | ||||||
|  |                 // process.exit(2);
 | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 execSync(`pm2 save --force`, { cwd: path }, (error, stdout, stderr) => { | ||||||
|  |                     if (error) { | ||||||
|  |                         console.log(`error: ${error.message}`) | ||||||
|  |                         return | ||||||
|  |                     } | ||||||
|  |                     if (stderr) { | ||||||
|  |                         console.log(`stderr: ${stderr}`) | ||||||
|  |                         return | ||||||
|  |                     } | ||||||
|  |                     console.log(`stdout: ${stdout}`) | ||||||
|  |                 }) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             pm2.disconnect() | ||||||
|  |         }) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = startPm2Process | ||||||
|  | @ -0,0 +1,33 @@ | ||||||
|  | const mongoose = require('../db/connMongo') | ||||||
|  | const { Schema } = mongoose | ||||||
|  | 
 | ||||||
|  | const OmnihitDBConn = mongoose.model( | ||||||
|  |   'Omnihit_db_conn', | ||||||
|  |   new Schema( | ||||||
|  |     { | ||||||
|  |       client_url: { | ||||||
|  |         type: String, | ||||||
|  |       }, | ||||||
|  |       db_conf: { | ||||||
|  |         DB: { | ||||||
|  |           type: String, | ||||||
|  |         }, | ||||||
|  |         DB_HOST: { | ||||||
|  |           type: String, | ||||||
|  |         }, | ||||||
|  |         DB_USER: { | ||||||
|  |           type: String, | ||||||
|  |         }, | ||||||
|  |         DB_PASS: { | ||||||
|  |           type: String, | ||||||
|  |         }, | ||||||
|  |         DB_PORT: { | ||||||
|  |           type: String, | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { timestamps: true } | ||||||
|  |   ) | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | module.exports = OmnihitDBConn | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,24 @@ | ||||||
|  | { | ||||||
|  |   "name": "api-sessions-controller", | ||||||
|  |   "version": "1.0.0", | ||||||
|  |   "description": "", | ||||||
|  |   "main": "index.js", | ||||||
|  |   "scripts": { | ||||||
|  |     "start": "nodemon app.js", | ||||||
|  |     "test": "echo \"Error: no test specified\" && exit 1" | ||||||
|  |   }, | ||||||
|  |   "keywords": [], | ||||||
|  |   "author": "Adriano <adriano08andrade@hotmail.com>", | ||||||
|  |   "license": "MIT", | ||||||
|  |   "dependencies": { | ||||||
|  |     "bcrypt": "^5.1.0", | ||||||
|  |     "body-parser": "^1.20.1", | ||||||
|  |     "dotenv": "^16.0.3", | ||||||
|  |     "express": "^4.18.2", | ||||||
|  |     "fs-extra": "^11.1.0", | ||||||
|  |     "mongoose": "^7.4.0", | ||||||
|  |     "mysql": "^2.18.1", | ||||||
|  |     "nodemon": "^2.0.20", | ||||||
|  |     "socket.io": "^4.5.4" | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,25 @@ | ||||||
|  | # NUMBER AND NAME THAT WILL BE DISPLAYED ON CONSOLE | ||||||
|  | MOBILEUID=5517988310949 | ||||||
|  | MOBILENAME=test - S1 | ||||||
|  | 
 | ||||||
|  | # PORT NUMBER FOR THIS API | ||||||
|  | PORT=8029 | ||||||
|  | 
 | ||||||
|  | # URL FROM THE OMNIHIT BACKEND API | ||||||
|  | CLIENT_URL=http://localhost:8080 | ||||||
|  | 
 | ||||||
|  | # OMNIHIT DATABASE | ||||||
|  | DB=whaticket | ||||||
|  | DB_HOST=localhost | ||||||
|  | DB_USER=whaticket | ||||||
|  | DB_PASS=strongpassword | ||||||
|  | DB_PORT=3306 | ||||||
|  | 
 | ||||||
|  | # WHATSAPP ID OF THE TABLE Whatsapps FROM THE OMNIHIT DATABASE | ||||||
|  | WHATSAPP_ID=223 | ||||||
|  | 
 | ||||||
|  | # MONGO CONNECTION | ||||||
|  | DB_MONGO_URL=mongodb://localhost:27017   | ||||||
|  | 
 | ||||||
|  | # MONGO COLLECTION | ||||||
|  | DB_MONGO_NAME=broker_omnihit | ||||||
|  | @ -0,0 +1,31 @@ | ||||||
|  |   | ||||||
|  | 
 | ||||||
|  | # dependencies | ||||||
|  | node_modules | ||||||
|  | /node_modules  | ||||||
|  | 
 | ||||||
|  | /medias/*.* | ||||||
|  | /medias/in/*.* | ||||||
|  |   | ||||||
|  |   | ||||||
|  | /WWebJS/session-OmniHIT/Default/**  | ||||||
|  |   | ||||||
|  | 
 | ||||||
|  | # testing | ||||||
|  | /coverage | ||||||
|  | 
 | ||||||
|  | # production | ||||||
|  | /build | ||||||
|  | 
 | ||||||
|  | # misc | ||||||
|  | .DS_Store | ||||||
|  | .env.local | ||||||
|  | .env.development.local | ||||||
|  | .env.test.local | ||||||
|  | .env.production.local | ||||||
|  | 
 | ||||||
|  | DevToolsActivePort* | ||||||
|  | 
 | ||||||
|  | npm-debug.log* | ||||||
|  | yarn-debug.log* | ||||||
|  | yarn-error.log* | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,141 @@ | ||||||
|  | const dotenv = require('dotenv'); | ||||||
|  | dotenv.config({ path: `${process.cwd()}/.env` }); | ||||||
|  | const path = require('path')    | ||||||
|  | 
 | ||||||
|  | const MongoClient = require( 'mongodb' ).MongoClient; | ||||||
|  | 
 | ||||||
|  | const url = process.env.DB_URL; | ||||||
|  | 
 | ||||||
|  | var _db; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  | 
 | ||||||
|  |   connectToServer: function( callback ) { | ||||||
|  | 
 | ||||||
|  |     MongoClient.connect( url,  { useNewUrlParser: true }, function( err, client ) { | ||||||
|  | 
 | ||||||
|  |       _db  = client.db(process.env.DB_NAME); | ||||||
|  | 
 | ||||||
|  |       return callback( err ); | ||||||
|  | 
 | ||||||
|  |     } ); | ||||||
|  | 
 | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   getDb: function() { | ||||||
|  | 
 | ||||||
|  |     return _db; | ||||||
|  | 
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // // PRODUCTION CONNECTION
 | ||||||
|  | // const MongoClient = require( 'mongodb' ).MongoClient;
 | ||||||
|  | 
 | ||||||
|  | // const url = "mongodb://admin:d1nf54012022prod*@172.31.187.8:27017";
 | ||||||
|  | 
 | ||||||
|  | // var _db;
 | ||||||
|  | 
 | ||||||
|  | // module.exports = {
 | ||||||
|  | 
 | ||||||
|  | //   connectToServer: function( callback ) {
 | ||||||
|  | 
 | ||||||
|  | //     MongoClient.connect( url,  { useNewUrlParser: true }, function( err, client ) {
 | ||||||
|  | 
 | ||||||
|  | //       _db  = client.db('db_omnihit');
 | ||||||
|  | 
 | ||||||
|  | //       return callback( err );
 | ||||||
|  | 
 | ||||||
|  | //     } );
 | ||||||
|  | 
 | ||||||
|  | //   },
 | ||||||
|  | 
 | ||||||
|  | //   getDb: function() {
 | ||||||
|  | 
 | ||||||
|  | //     return _db;
 | ||||||
|  | 
 | ||||||
|  | //   }
 | ||||||
|  | 
 | ||||||
|  | // };
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // LOCA CONNECTION
 | ||||||
|  | // const MongoClient = require( 'mongodb' ).MongoClient;
 | ||||||
|  | 
 | ||||||
|  | // const url = 'mongodb://localhost:27017';
 | ||||||
|  | 
 | ||||||
|  | // var _db;
 | ||||||
|  | 
 | ||||||
|  | // module.exports = {
 | ||||||
|  | 
 | ||||||
|  | //   connectToServer: function( callback ) {
 | ||||||
|  | //     MongoClient.connect( url,  { useNewUrlParser: true }, function( err, client ) {
 | ||||||
|  | //       _db  = client.db('db_omnihit');
 | ||||||
|  | //       return callback( err );
 | ||||||
|  | //     } );
 | ||||||
|  | //   },
 | ||||||
|  | 
 | ||||||
|  | //   getDb: function() {
 | ||||||
|  | //     return _db;
 | ||||||
|  | //   }
 | ||||||
|  | // }   
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  | 
 | ||||||
|  | const MongoClient = require( 'mongodb' ).MongoClient; | ||||||
|  | 
 | ||||||
|  | const url = "mongodb://admin:d1nf54012022*@172.31.187.2:27017"; | ||||||
|  | 
 | ||||||
|  | var _db;   | ||||||
|  | 
 | ||||||
|  | module.exports = {  | ||||||
|  | 
 | ||||||
|  |   connectToServer: function( callback ) { | ||||||
|  | 
 | ||||||
|  |     MongoClient.connect( url,  { useNewUrlParser: true }, function( err, client ) { | ||||||
|  | 
 | ||||||
|  |       _db  = client.db('db_omnihit_todoo'); | ||||||
|  | 
 | ||||||
|  |       return callback( err ); | ||||||
|  | 
 | ||||||
|  |     } ); | ||||||
|  | 
 | ||||||
|  |   },  | ||||||
|  | 
 | ||||||
|  |   getDb: function() { | ||||||
|  | 
 | ||||||
|  |     return _db; | ||||||
|  | 
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | */ | ||||||
|  |    | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // const MongoClient = require( 'mongodb' ).MongoClient;
 | ||||||
|  | 
 | ||||||
|  | // const url = "mongodb://admin:d1nf5401@192.168.15.13/admin?retryWrites=true&w=majority";
 | ||||||
|  | 
 | ||||||
|  | // var _db;
 | ||||||
|  | 
 | ||||||
|  | // module.exports = {
 | ||||||
|  | 
 | ||||||
|  | //   connectToServer: function( callback ) {
 | ||||||
|  | //     MongoClient.connect( url,  { useNewUrlParser: true }, function( err, client ) {
 | ||||||
|  | //       _db  = client.db('db_omnihit');
 | ||||||
|  | //       return callback( err );
 | ||||||
|  | //     } );
 | ||||||
|  | //   },
 | ||||||
|  | 
 | ||||||
|  | //   getDb: function() {
 | ||||||
|  | //     return _db;
 | ||||||
|  | //   }
 | ||||||
|  | // };
 | ||||||
|  | @ -0,0 +1,18 @@ | ||||||
|  | const { MongoClient } = require('mongodb') | ||||||
|  | 
 | ||||||
|  | const uri = process.env.DB_MONGO_URL | ||||||
|  | 
 | ||||||
|  | const mongo = new MongoClient(uri)  | ||||||
|  | 
 | ||||||
|  | async function run() { | ||||||
|  |     try { | ||||||
|  |         await mongo.connect()  | ||||||
|  |         console.log('Connectado ao mongo db')   | ||||||
|  |     } catch (err) { | ||||||
|  |         console.log(err) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | run() | ||||||
|  | 
 | ||||||
|  | module.exports = mongo | ||||||
|  | @ -0,0 +1,85 @@ | ||||||
|  | const os = require('os'); | ||||||
|  | const dbcloud = require('./dbcloud.js'); | ||||||
|  | const dotenv = require('dotenv'); | ||||||
|  | const axios = require('axios').default; | ||||||
|  | dotenv.config({path: '../.env'});  | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  | 
 | ||||||
|  |   // qrcode: async function(mobileuid, mobilename, qr ) {
 | ||||||
|  |   //   payload = {}
 | ||||||
|  |   //   payload['mobileuid'] = mobileuid;
 | ||||||
|  |   //   payload['mobilename'] = mobilename;
 | ||||||
|  |   //   payload['qrcode'] = qr;
 | ||||||
|  |   //   await axios.post(process.env.URL_QRCODE, payload, { timeout: 2000, headers: { 'content-type': 'application/json' } });
 | ||||||
|  |   //   console.log(new Date().toISOString() + " >>> SEND QR CODE ---------------------------------------");
 | ||||||
|  |   //   return 200;
 | ||||||
|  |   // },
 | ||||||
|  | 
 | ||||||
|  |   monitor: async function(mobileuid, mobilename, stat) { | ||||||
|  | 
 | ||||||
|  |       | ||||||
|  |     let _totalmem = parseInt(os.totalmem()); | ||||||
|  |     let _freemem  = parseInt(os.freemem()); | ||||||
|  |     let _memory = 100 - (_freemem / _totalmem * 100);  | ||||||
|  |      | ||||||
|  |      | ||||||
|  |     payload = {} | ||||||
|  |     payload['mobileuid']   = mobileuid; | ||||||
|  |     payload['mobilename']  = mobilename; | ||||||
|  |     payload['memory']      = _memory; | ||||||
|  | 
 | ||||||
|  |     let db = dbcloud.getDb(); | ||||||
|  | 
 | ||||||
|  |     let mco = await db.collection('tab_counts').find({ "_id": mobileuid }).limit(1).toArray();  | ||||||
|  | 
 | ||||||
|  |     if ( mco.length == 0 ) {  | ||||||
|  |         payload['_id']         = mobileuid; | ||||||
|  |         payload['hour']        = 0; | ||||||
|  |         payload['in']          = 0; | ||||||
|  |         payload['out']         = 0; | ||||||
|  |         payload['lastmessage'] = new Date(new Date() + 'UTC'); | ||||||
|  | 
 | ||||||
|  |         payload['in_today']    = 0; | ||||||
|  |         payload['out_today']   = 0; | ||||||
|  | 
 | ||||||
|  |         await db.collection('tab_counts').insertOne(payload); | ||||||
|  | 
 | ||||||
|  |     }else{   | ||||||
|  |       payload['hour']        = mco[0]['hour']; | ||||||
|  |       payload['in']          = mco[0]['in']; | ||||||
|  |       payload['out']         = mco[0]['out']; | ||||||
|  |       payload['lastmessage'] = mco[0]['lastmessage']       | ||||||
|  |       payload['in_today']    = mco[0]['in_today']; | ||||||
|  |       payload['out_today']   = mco[0]['out_today']; | ||||||
|  |     }       | ||||||
|  | 
 | ||||||
|  |     payload['dt']          = new Date(new Date() + 'UTC'); | ||||||
|  |     payload['status']      = stat; | ||||||
|  |      | ||||||
|  | 
 | ||||||
|  |     console.log(new Date().toISOString() + " >>> SEND MONITOR ALARM ---------------------------------------");  | ||||||
|  |     console.log(new Date().toISOString() + " >>> payload: ",payload);  | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     let monitor = await db.collection('tab_monitor').find({ "_id": mobileuid }).limit(1).toArray(); | ||||||
|  | 
 | ||||||
|  |     if ( monitor.length == 0 ) {  | ||||||
|  | 
 | ||||||
|  |       payload['_id'] = mobileuid | ||||||
|  |        | ||||||
|  |       await db.collection('tab_monitor').insertOne(payload);  | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |     else{  | ||||||
|  | 
 | ||||||
|  |       await  db.collection('tab_monitor').updateOne({ "_id": mobileuid }, { $set: payload }, true); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     return 200; | ||||||
|  |   }   | ||||||
|  | } | ||||||
|  | @ -0,0 +1,43 @@ | ||||||
|  | module.exports = function() { | ||||||
|  | 
 | ||||||
|  |     this.getTimestamp = function() { | ||||||
|  |             var date = new Date(); | ||||||
|  |             var year = date.getFullYear(); | ||||||
|  |             var month = ("0"+(date.getMonth()+1)).substr(-2); | ||||||
|  |             var day = ("0"+date.getDate()).substr(-2); | ||||||
|  |             var hour = ("0"+date.getHours()).substr(-2); | ||||||
|  |             var minutes = ("0"+date.getMinutes()).substr(-2); | ||||||
|  |             var seconds = ("0"+date.getSeconds()).substr(-2); | ||||||
|  |             return year+"-"+month+"-"+day+" "+hour+":"+minutes+":"+seconds; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     this.log = function(desc, message) { | ||||||
|  |       console.log(getTimestamp() + ' >> ' + desc + " :: "); | ||||||
|  |       console.log(message); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     this.getTime = function() { | ||||||
|  |             var date = new Date(); | ||||||
|  |             var hour = ("0"+date.getHours()).substr(-2); | ||||||
|  |             var minutes = ("0"+date.getMinutes()).substr(-2); | ||||||
|  |             var seconds = ("0"+date.getSeconds()).substr(-2); | ||||||
|  |             return hour+":"+minutes+":"+seconds; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     this.forceGC = function() { | ||||||
|  |        if (global.gc) { | ||||||
|  |            console.log(getTimestamp() + " >> Starting Garbage Collector..."); | ||||||
|  |            global.gc(); | ||||||
|  |        } else { | ||||||
|  |            console.warn("Garbage Collector não habilitado! Execute seu programa com node --expose-gc app.js."); | ||||||
|  |        } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     this.isJSON = function(str) { | ||||||
|  |        try { | ||||||
|  |            return (JSON.parse(str) && !!str); | ||||||
|  |        } catch (e) { | ||||||
|  |            return false; | ||||||
|  |        } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | const removeDir = require('./remove_dir'); | ||||||
|  | const copyFolder = require('./copyFolder'); | ||||||
|  | const path = require('path'); | ||||||
|  | const fs = require('fs'); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | async function backup_session(destroy, save_session_after, save_first_read_only=false) { | ||||||
|  | 
 | ||||||
|  |     console.log('process.cwd(): ', process.cwd()) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     const sessionBackupPath = path.join(process.cwd(), `session_backup`, `session-omnihit_sesssion`) | ||||||
|  | 
 | ||||||
|  |     if (fs.existsSync(sessionBackupPath) && save_first_read_only) return | ||||||
|  | 
 | ||||||
|  |     destroy = setTimeout(async () => {  | ||||||
|  | 
 | ||||||
|  |         const sessionPath = path.join(process.cwd(), '.wwebjs_auth', 'session-omnihit_sesssion') | ||||||
|  | 
 | ||||||
|  |         if (fs.existsSync(path.join(process.cwd(), '.wwebjs_auth'))) { | ||||||
|  | 
 | ||||||
|  |             await removeDir(sessionBackupPath) | ||||||
|  | 
 | ||||||
|  |             // copy the good session for backup dir 
 | ||||||
|  | 
 | ||||||
|  |             copyFolder(sessionPath, sessionBackupPath) | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             console.log('Directory not found to copy backup_session: ', sessionPath) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     }, save_session_after); | ||||||
|  | 
 | ||||||
|  |     return destroy | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = backup_session; | ||||||
|  | @ -0,0 +1,41 @@ | ||||||
|  | const http = require('http') | ||||||
|  | 
 | ||||||
|  | const checkInternetConnection = async () => { | ||||||
|  |     const options = { | ||||||
|  |         hostname: 'www.google.com', | ||||||
|  |         port: 80, | ||||||
|  |         method: 'HEAD' | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return new Promise((resolve, reject) => { | ||||||
|  |         const req = http.request(options, (res) => { | ||||||
|  |             if (res.statusCode === 200) { | ||||||
|  |                 resolve(true) | ||||||
|  |             } else { | ||||||
|  |                 resolve(false) | ||||||
|  |             } | ||||||
|  |             req.abort() | ||||||
|  |         }) | ||||||
|  | 
 | ||||||
|  |         req.on('error', (err) => { | ||||||
|  |             resolve(false) | ||||||
|  |         }) | ||||||
|  | 
 | ||||||
|  |         req.end() | ||||||
|  |     }) | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // (async () => {
 | ||||||
|  | //     try {
 | ||||||
|  | //         const isConnected = await checkInternetConnection()
 | ||||||
|  | //         if (isConnected) {
 | ||||||
|  | //             console.log('Internet connection is available.')
 | ||||||
|  | //         } else {
 | ||||||
|  | //             console.log('Internet connection is not available.')
 | ||||||
|  | //         }
 | ||||||
|  | //     } catch (error) {
 | ||||||
|  | //         console.error('Error checking internet connection:', error)
 | ||||||
|  | //     }
 | ||||||
|  | // })()
 | ||||||
|  | 
 | ||||||
|  | module.exports = checkInternetConnection | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | const fsPromises = require("fs/promises"); | ||||||
|  | const fs = require('fs-extra') | ||||||
|  | 
 | ||||||
|  | // Delete a directory and its children 
 | ||||||
|  | function copyFolder(sourcePath, destPath) { | ||||||
|  | 
 | ||||||
|  |     fs.copySync(sourcePath, destPath, { overwrite: true }, (err) => { | ||||||
|  |         if (err) { | ||||||
|  |             console.error(err); | ||||||
|  |         } else { | ||||||
|  |             console.log("Copy dir success!"); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = copyFolder; | ||||||
|  | @ -0,0 +1,36 @@ | ||||||
|  | const multer = require('multer') | ||||||
|  | const path = require('path') | ||||||
|  |   | ||||||
|  |   | ||||||
|  | 
 | ||||||
|  | //Destination to store the images
 | ||||||
|  | const imageStorage = multer.diskStorage({ | ||||||
|  |     destination: function(req, file, cb){ | ||||||
|  |        | ||||||
|  |         // let folder = "" 
 | ||||||
|  |         // if(req.baseUrl.includes("users")){
 | ||||||
|  |         //     folder = "users"
 | ||||||
|  |         // }else if(req.baseUrl.includes("pets")){
 | ||||||
|  |         //     folder = "pets"
 | ||||||
|  |         // }
 | ||||||
|  |           | ||||||
|  |         | ||||||
|  |         cb(null,  path.join(process.cwd(),'medias', 'out')) | ||||||
|  |     }, | ||||||
|  |     filename: function(req, file, cb) { | ||||||
|  |         cb(null, Date.now() + path.extname(file.originalname))  | ||||||
|  |     } | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | const imageUpload = multer({ | ||||||
|  |     storage: imageStorage, | ||||||
|  |     // fileFilter(req, file, cb){
 | ||||||
|  |     //     if (!file.originalname.match(/\.(jpg|jpeg|png)$/)){
 | ||||||
|  |     //         return cb(new Error('Por favor, envie apenas jpg ou png!'))
 | ||||||
|  |     //     }
 | ||||||
|  |     //     cb(undefined, true)
 | ||||||
|  |     // }
 | ||||||
|  |      | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | module.exports = { imageUpload } | ||||||
|  | @ -0,0 +1,129 @@ | ||||||
|  | const dotenv = require('dotenv'); | ||||||
|  | dotenv.config({ path: `${process.cwd()}/.env` }); | ||||||
|  | const path = require('path')    | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // Ubicua Plataform - MYSQL Module
 | ||||||
|  | try{ | ||||||
|  |     var mysql_npm = require('mysql'); | ||||||
|  | }catch(err){ | ||||||
|  |     console.log("Cannot find `mysql` module. Is it installed ? Try `npm install mysql` or `npm install`."); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | var db_config = { | ||||||
|  |         host         : process.env.DB_HOST, | ||||||
|  |         user         : process.env.DB_USER, | ||||||
|  |         password     : process.env.DB_PASS, | ||||||
|  |         database     : process.env.DB, | ||||||
|  |         charset      : 'utf8mb4_general_ci',         | ||||||
|  |         port         : process.env.DB_PORT | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Connection configuration
 | ||||||
|  | //-
 | ||||||
|  | // var db_config = {
 | ||||||
|  | //     host         : 'localhost',
 | ||||||
|  | //     user         : 'whaticket',
 | ||||||
|  | //     password     : '9147teste',
 | ||||||
|  | //     database     : 'db_cdnwork',
 | ||||||
|  | //     charset      : 'utf8mb4_general_ci',
 | ||||||
|  | //     port         : '6603'
 | ||||||
|  | // };
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // var db_config = {
 | ||||||
|  | //     host         : '172.31.187.7',
 | ||||||
|  | //     user         : 'todoo',
 | ||||||
|  | //     password     : '7901228899',
 | ||||||
|  | //     database     : 'db_cdnwork',
 | ||||||
|  | //     charset      : 'utf8mb4_general_ci'
 | ||||||
|  | // };
 | ||||||
|  |   | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Create the connection variable
 | ||||||
|  | //-
 | ||||||
|  | var connection = mysql_npm.createPool(db_config); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Establish a new connection
 | ||||||
|  | //-
 | ||||||
|  | connection.getConnection(function(err){ | ||||||
|  |     if(err) { | ||||||
|  |         // mysqlErrorHandling(connection, err);
 | ||||||
|  |         console.log("\n\t *** Cannot establish a connection with the database. ***"); | ||||||
|  | 
 | ||||||
|  |         connection = reconnect(connection); | ||||||
|  |     }else { | ||||||
|  |         console.log("\n\t *** New connection established with the database. ***") | ||||||
|  |     } | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Reconnection function
 | ||||||
|  | //-
 | ||||||
|  | function reconnect(connection){ | ||||||
|  |     console.log("\n New connection tentative..."); | ||||||
|  | 
 | ||||||
|  |     //- Create a new one
 | ||||||
|  |     connection = mysql_npm.createPool(db_config); | ||||||
|  | 
 | ||||||
|  |     //- Try to reconnect
 | ||||||
|  |     connection.getConnection(function(err){ | ||||||
|  |         if(err) { | ||||||
|  |             //- Try to connect every 2 seconds.
 | ||||||
|  |             setTimeout(reconnect(connection), 2000); | ||||||
|  |         }else { | ||||||
|  |             console.log("\n\t *** New connection established with the database. ***") | ||||||
|  |             return connection; | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Error listener
 | ||||||
|  | //-
 | ||||||
|  | connection.on('error', function(err) { | ||||||
|  | 
 | ||||||
|  |     //-
 | ||||||
|  |     //- The server close the connection.
 | ||||||
|  |     //-
 | ||||||
|  |     if(err.code === "PROTOCOL_CONNECTION_LOST"){     | ||||||
|  |         console.log("/!\\ Cannot establish a connection with the database. /!\\ ("+err.code+")"); | ||||||
|  |         return reconnect(connection); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     else if(err.code === "PROTOCOL_ENQUEUE_AFTER_QUIT"){ | ||||||
|  |         console.log("/!\\ Cannot establish a connection with the database. /!\\ ("+err.code+")"); | ||||||
|  |         return reconnect(connection); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     else if(err.code === "PROTOCOL_ENQUEUE_AFTER_FATAL_ERROR"){ | ||||||
|  |         console.log("/!\\ Cannot establish a connection with the database. /!\\ ("+err.code+")"); | ||||||
|  |         return reconnect(connection); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     else if(err.code === "PROTOCOL_ENQUEUE_HANDSHAKE_TWICE"){ | ||||||
|  |         console.log("/!\\ Cannot establish a connection with the database. /!\\ ("+err.code+")"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     else{ | ||||||
|  |         console.log("/!\\ Cannot establish a connection with the database. /!\\ ("+err.code+")"); | ||||||
|  |         return reconnect(connection); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //-
 | ||||||
|  | //- Export
 | ||||||
|  | //-
 | ||||||
|  | module.exports = connection; | ||||||
|  | @ -0,0 +1,28 @@ | ||||||
|  | const fsPromises = require("fs/promises"); | ||||||
|  | const fs = require('fs') | ||||||
|  | 
 | ||||||
|  | // Delete a directory and its children 
 | ||||||
|  | const removeDir = async (dirPath) => { | ||||||
|  | 
 | ||||||
|  |     if (fs.existsSync(dirPath)) { | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             await fsPromises.rm(dirPath, { recursive: true, force: true }); | ||||||
|  |             console.log("Directory removed!"); | ||||||
|  | 
 | ||||||
|  |             return true | ||||||
|  |         } | ||||||
|  |         catch (err) { | ||||||
|  |             console.log('An error occurred while removing the directory: ', err);    | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  |     else {  | ||||||
|  |         console.log('Directory not found to remove: ', dirPath)  | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return false | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = removeDir ; | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | const removeDir = require('./remove_dir'); | ||||||
|  | const copyFolder = require('./copyFolder'); | ||||||
|  | const path = require('path'); | ||||||
|  | const fs = require('fs'); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | async function restore(client) {  | ||||||
|  |   | ||||||
|  |     try { | ||||||
|  |         await client.destroy() | ||||||
|  |     } catch (error) { | ||||||
|  |         console.error(`Error on try destroy client: ${error}`) | ||||||
|  |     }  | ||||||
|  | 
 | ||||||
|  |     const sessionBackupPath = path.join(process.cwd(), `session_backup`, `session-omnihit_sesssion`) | ||||||
|  | 
 | ||||||
|  |     const sessionPath = path.join(process.cwd(), '.wwebjs_auth', 'session-omnihit_sesssion') | ||||||
|  | 
 | ||||||
|  |     if (fs.existsSync(path.join(process.cwd(), `session_backup`, `session-omnihit_sesssion`))) { | ||||||
|  | 
 | ||||||
|  |         await removeDir(sessionPath) | ||||||
|  | 
 | ||||||
|  |         // copy the good session for backup dir
 | ||||||
|  |         copyFolder(sessionBackupPath, sessionPath)  | ||||||
|  |          | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |         console.log('Directory not found to copy: ', sessionPath) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     setTimeout(() => { | ||||||
|  |         console.log('process.exit: kkkkkkkkkkkkkkkkkkkkk') | ||||||
|  |         process.exit() | ||||||
|  |     }, 5000) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = restore; | ||||||
|  | @ -0,0 +1,67 @@ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | var io = require('socket.io-client'); | ||||||
|  | 
 | ||||||
|  | var lst = []  | ||||||
|  | 
 | ||||||
|  | const _clear_lst = () => { | ||||||
|  | 
 | ||||||
|  |   if (lst.length <= 199) return | ||||||
|  | 
 | ||||||
|  |   const chunk = Math.floor((lst.length / 2)) | ||||||
|  | 
 | ||||||
|  |   lst = lst.slice(chunk, chunk + lst.length); | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | const multisessionIdControll = (msgId) => { | ||||||
|  | 
 | ||||||
|  |   _clear_lst() | ||||||
|  | 
 | ||||||
|  |   let index = lst.findIndex((x) => x.id == msgId) | ||||||
|  | 
 | ||||||
|  |   console.log('INDEX: ', index) | ||||||
|  | 
 | ||||||
|  |   if (index == -1) { | ||||||
|  | 
 | ||||||
|  |       lst.push({ id: msgId }) | ||||||
|  | 
 | ||||||
|  |   } | ||||||
|  |   else { | ||||||
|  |       console.log('IGNORED ID: ', msgId) | ||||||
|  | 
 | ||||||
|  |       return | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // console.log('LIST OF ID MESSAGE lst: ', lst)
 | ||||||
|  | 
 | ||||||
|  |   console.log('PASSOU.................................ID: ', msgId) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | const initIO = (url) => { | ||||||
|  | 
 | ||||||
|  |   const socket = io(url, { reconnect: true }); | ||||||
|  | 
 | ||||||
|  |   socket.on('connect', async () => { | ||||||
|  |     console.log('Made socket connection2'); | ||||||
|  |   });   | ||||||
|  | 
 | ||||||
|  |   socket.on('messageId', messageId => { | ||||||
|  | 
 | ||||||
|  |     console.log('-------> messageId: ', messageId); | ||||||
|  | 
 | ||||||
|  |     multisessionIdControll(messageId) | ||||||
|  | 
 | ||||||
|  |     console.log('socket lst: ', lst) | ||||||
|  | 
 | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return socket; | ||||||
|  | };  | ||||||
|  | 
 | ||||||
|  | module.exports = { initIO, lst } | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,34 @@ | ||||||
|  | { | ||||||
|  |   "name": "omnihit", | ||||||
|  |   "version": "1.0.0", | ||||||
|  |   "description": "", | ||||||
|  |   "main": "app.js", | ||||||
|  |   "scripts": { | ||||||
|  |     "test": "echo \"Error: no test specified\" && exit 1", | ||||||
|  |     "start": "nodemon ./app.js" | ||||||
|  |   }, | ||||||
|  |   "author": "Edson da Silva", | ||||||
|  |   "license": "ISC", | ||||||
|  |   "dependencies": { | ||||||
|  |     "axios": "^0.21.4", | ||||||
|  |     "body-parser": "^1.19.0", | ||||||
|  |     "dotenv": "^16.0.0", | ||||||
|  |     "express": "^4.17.1", | ||||||
|  |     "form-data": "^4.0.0", | ||||||
|  |     "fs-extra": "^11.1.0", | ||||||
|  |     "logger": "^0.0.1", | ||||||
|  |     "mime": "^2.4.5", | ||||||
|  |     "mongodb": "^4.1.1", | ||||||
|  |     "mongoose": "^7.4.3", | ||||||
|  |     "multer": "^1.4.4", | ||||||
|  |     "mysql": "^2.18.1", | ||||||
|  |     "node-os-utils": "^1.3.5", | ||||||
|  |     "qr-encode": "^0.3.0", | ||||||
|  |     "qrcode-terminal": "^0.12.0", | ||||||
|  |     "socket.io": "^4.5.4", | ||||||
|  |     "socket.io-client": "^4.5.4" | ||||||
|  |   }, | ||||||
|  |   "devDependencies": { | ||||||
|  |     "nodemon": "^2.0.20" | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | @ -183,11 +183,17 @@ socketIo.on('connect_error', async function (err) { | ||||||
| }) | }) | ||||||
| //
 | //
 | ||||||
| 
 | 
 | ||||||
|  | const wwebVersion = '2.2402.5'; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| //NOVA OPÇÃO MD
 | //NOVA OPÇÃO MD
 | ||||||
| client = new Client({ | client = new Client({ | ||||||
|     authStrategy: new LocalAuth({ clientId: 'omnihit_sesssion' }), |     authStrategy: new LocalAuth({ clientId: 'omnihit_sesssion' }), | ||||||
|     puppeteer: { args: ['--no-sandbox', '--disable-setuid-sandbox'], executablePath: process.env.CHROME_BIN || '/usr/bin/google-chrome-stable' }, |     puppeteer: { args: ['--no-sandbox', '--disable-setuid-sandbox'], executablePath: process.env.CHROME_BIN || '/usr/bin/google-chrome-stable' }, | ||||||
|  |     webVersionCache: { | ||||||
|  |         type: 'remote', | ||||||
|  |         remotePath: `https://raw.githubusercontent.com/wppconnect-team/wa-version/main/html/${wwebVersion}.html`, | ||||||
|  |     }, | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| client.initialize() | client.initialize() | ||||||
|  |  | ||||||
|  | @ -20,11 +20,13 @@ import { | ||||||
| } from "../helpers/ContactsCache"; | } from "../helpers/ContactsCache"; | ||||||
| 
 | 
 | ||||||
| import { off } from "process"; | import { off } from "process"; | ||||||
| import GetContactService from "../services/ContactServices/GetContactService" | import GetContactService from "../services/ContactServices/GetContactService"; | ||||||
|  | import ContactQueue from "../models/ContactQueues"; | ||||||
| 
 | 
 | ||||||
| type IndexQuery = { | type IndexQuery = { | ||||||
|   searchParam: string; |   searchParam: string; | ||||||
|   pageNumber: string; |   pageNumber: string; | ||||||
|  |   userId?:string; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| type IndexGetContactQuery = { | type IndexGetContactQuery = { | ||||||
|  | @ -41,10 +43,11 @@ interface ContactData { | ||||||
|   number: string; |   number: string; | ||||||
|   email?: string; |   email?: string; | ||||||
|   extraInfo?: ExtraInfo[]; |   extraInfo?: ExtraInfo[]; | ||||||
|  |   queueIds?: number[]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export const index = async (req: Request, res: Response): Promise<Response> => { | export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   let { searchParam, pageNumber } = req.query as IndexQuery; |   let { searchParam, pageNumber, userId } = req.query as IndexQuery; | ||||||
| 
 | 
 | ||||||
|   console.log("PAGE NUMBER CONTACT: ", pageNumber); |   console.log("PAGE NUMBER CONTACT: ", pageNumber); | ||||||
| 
 | 
 | ||||||
|  | @ -78,13 +81,12 @@ export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|         error |         error | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|   } |   }  | ||||||
| 
 |  | ||||||
|   console.log("QUERY CONTACTS FROM DATABASE SEARCH PARAM: ", searchParam); |  | ||||||
| 
 | 
 | ||||||
|   const { contacts, count, hasMore } = await ListContactsService({ |   const { contacts, count, hasMore } = await ListContactsService({ | ||||||
|     searchParam, |     searchParam, | ||||||
|     pageNumber |     pageNumber, | ||||||
|  |     userId | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   return res.json({ contacts, count, hasMore }); |   return res.json({ contacts, count, hasMore }); | ||||||
|  | @ -124,13 +126,13 @@ export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
| 
 | 
 | ||||||
|   newContact.number = addStartPhoneNumber(newContact.number); |   newContact.number = addStartPhoneNumber(newContact.number); | ||||||
| 
 | 
 | ||||||
|   const validNumber = await CheckIsValidContact(newContact.number);  |   const validNumber = await CheckIsValidContact(newContact.number); | ||||||
| 
 | 
 | ||||||
|   if (!validNumber) { |   if (!validNumber) { | ||||||
|     throw new AppError("ERR_WAPP_CHECK_CONTACT"); |     throw new AppError("ERR_WAPP_CHECK_CONTACT"); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const profilePicUrl = await GetProfilePicUrl(validNumber);  |   const profilePicUrl = await GetProfilePicUrl(validNumber); | ||||||
| 
 | 
 | ||||||
|   let name = newContact.name; |   let name = newContact.name; | ||||||
|   let number = validNumber; |   let number = validNumber; | ||||||
|  | @ -142,7 +144,8 @@ export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|     number, |     number, | ||||||
|     email, |     email, | ||||||
|     profilePicUrl: profilePicUrl, |     profilePicUrl: profilePicUrl, | ||||||
|     extraInfo |     extraInfo, | ||||||
|  |     queueIds: newContact?.queueIds | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   const io = getIO(); |   const io = getIO(); | ||||||
|  | @ -208,6 +211,8 @@ export const remove = async ( | ||||||
| 
 | 
 | ||||||
|   await DeleteContactService(contactId); |   await DeleteContactService(contactId); | ||||||
| 
 | 
 | ||||||
|  |   await ContactQueue.destroy({ where: { contactId } }); | ||||||
|  | 
 | ||||||
|   const io = getIO(); |   const io = getIO(); | ||||||
|   io.emit("contact", { |   io.emit("contact", { | ||||||
|     action: "delete", |     action: "delete", | ||||||
|  |  | ||||||
|  | @ -0,0 +1,113 @@ | ||||||
|  | import * as Yup from "yup"; | ||||||
|  | import { Request, Response } from "express"; | ||||||
|  | import { getIO } from "../libs/socket"; | ||||||
|  | 
 | ||||||
|  | import AppError from "../errors/AppError"; | ||||||
|  | import ListPositionService from "../services/PositionService/ListPositionService"; | ||||||
|  | import ShowPositionService from "../services/PositionService/ShowPositionService"; | ||||||
|  | import CreatePositionService from "../services/PositionService/CreatePositionService"; | ||||||
|  | import UpdatePositionService from "../services/PositionService/UpdatePositionService"; | ||||||
|  | import DeletePositionService from "../services/PositionService/DeletePositionService"; | ||||||
|  | 
 | ||||||
|  | type IndexQuery = { | ||||||
|  |   searchParam: string; | ||||||
|  |   pageNumber: string; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | interface PositionData { | ||||||
|  |   name: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const { searchParam, pageNumber } = req.query as IndexQuery; | ||||||
|  | 
 | ||||||
|  |   const { positions, count, hasMore } = await ListPositionService({ | ||||||
|  |     searchParam, | ||||||
|  |     pageNumber | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.json({ positions, count, hasMore }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const newPosition: PositionData = req.body; | ||||||
|  | 
 | ||||||
|  |   const PositionSchema = Yup.object().shape({ | ||||||
|  |     name: Yup.string().required() | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   try { | ||||||
|  |     await PositionSchema.validate(newPosition); | ||||||
|  |   } catch (err: any) { | ||||||
|  |     throw new AppError(err.message); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const position = await CreatePositionService({ | ||||||
|  |     ...newPosition | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const io = getIO(); | ||||||
|  |   io.emit("position", { | ||||||
|  |     action: "create", | ||||||
|  |     position | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(position); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const show = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const { positionId } = req.params; | ||||||
|  | 
 | ||||||
|  |   const position = await ShowPositionService(positionId); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(position); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const update = async ( | ||||||
|  |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   const positionData: PositionData = req.body; | ||||||
|  | 
 | ||||||
|  |   const schema = Yup.object().shape({ | ||||||
|  |     name: Yup.string().required() | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   try { | ||||||
|  |     await schema.validate(positionData); | ||||||
|  |   } catch (err: any) { | ||||||
|  |     throw new AppError(err.message); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const { positionId } = req.params; | ||||||
|  | 
 | ||||||
|  |   const position = await UpdatePositionService({ | ||||||
|  |     positionData, | ||||||
|  |     positionId | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const io = getIO(); | ||||||
|  |   io.emit("position", { | ||||||
|  |     action: "update", | ||||||
|  |     position | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(position); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const remove = async ( | ||||||
|  |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   const { positionId } = req.params; | ||||||
|  | 
 | ||||||
|  |   await DeletePositionService(positionId); | ||||||
|  | 
 | ||||||
|  |   const io = getIO(); | ||||||
|  |   io.emit("position", { | ||||||
|  |     action: "delete", | ||||||
|  |     positionId | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json({ message: "Position deleted" }); | ||||||
|  | }; | ||||||
|  | @ -8,6 +8,11 @@ import UpdateQueueService from "../services/QueueService/UpdateQueueService"; | ||||||
| import Queue from "../models/Queue"; | import Queue from "../models/Queue"; | ||||||
| import AppError from "../errors/AppError"; | import AppError from "../errors/AppError"; | ||||||
| import { del, get, set } from "../helpers/RedisClient"; | import { del, get, set } from "../helpers/RedisClient"; | ||||||
|  | import { Op } from "sequelize"; | ||||||
|  | import ListWhatsAppsService from "../services/WhatsappService/ListWhatsAppsService"; | ||||||
|  | import Whatsapp from "../models/Whatsapp"; | ||||||
|  | import QuickAnswerQueue from "../models/QuickAnswerQueue"; | ||||||
|  | import ContactQueue from "../models/ContactQueues"; | ||||||
| 
 | 
 | ||||||
| export const index = async (req: Request, res: Response): Promise<Response> => { | export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   const queues = await ListQueuesService(); |   const queues = await ListQueuesService(); | ||||||
|  | @ -15,10 +20,68 @@ export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   return res.status(200).json(queues); |   return res.status(200).json(queues); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const store = async (req: Request, res: Response): Promise<Response> => { | export const listQueues = async ( | ||||||
|   const { name, color, greetingMessage } = req.body; |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   const whatsapps = await Whatsapp.findAll({ | ||||||
|  |     where: { | ||||||
|  |       name: { [Op.ne]: "botqueue" }, | ||||||
|  |       number: { [Op.ne]: "" }, | ||||||
|  |       phoneNumberId: false | ||||||
|  |     }, | ||||||
|  |     attributes: ["number"], | ||||||
|  |     include: [ | ||||||
|  |       { | ||||||
|  |         model: Queue, | ||||||
|  |         as: "queues", | ||||||
|  |         attributes: ["id", "name"] | ||||||
|  |       } | ||||||
|  |     ] | ||||||
|  |   }); | ||||||
| 
 | 
 | ||||||
|   const queue = await CreateQueueService({ name, color, greetingMessage }); |   const whats = whatsapps | ||||||
|  |     ?.filter((w: any) => w?.queues?.length > 0) | ||||||
|  |     ?.map((w: any) => { | ||||||
|  |       const { number, queues } = w; | ||||||
|  |       return { | ||||||
|  |         number, | ||||||
|  |         queues: queues?.map((q: any) => { | ||||||
|  |           const { id, name } = q; | ||||||
|  |           return { id, name }; | ||||||
|  |         }) | ||||||
|  |       }; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |   let _queues: any = []; | ||||||
|  | 
 | ||||||
|  |   for (const w of whats) { | ||||||
|  |     const { queues } = w; | ||||||
|  | 
 | ||||||
|  |     for (const q of queues) { | ||||||
|  |       const { id: queueId, name } = q; | ||||||
|  | 
 | ||||||
|  |       const auxQueue = _queues.findIndex((q: any) => q.queueId == queueId); | ||||||
|  | 
 | ||||||
|  |       if (auxQueue == -1) { | ||||||
|  |         _queues.push({ queueId, name }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(_queues); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const { name, color, greetingMessage, farewellMessage, cc } = req.body; | ||||||
|  | 
 | ||||||
|  |   const queue = await CreateQueueService({ | ||||||
|  |     name, | ||||||
|  |     color, | ||||||
|  |     greetingMessage, | ||||||
|  |     cc, | ||||||
|  |     farewellMessage | ||||||
|  |   }); | ||||||
| 
 | 
 | ||||||
|   const io = getIO(); |   const io = getIO(); | ||||||
|   io.emit("queue", { |   io.emit("queue", { | ||||||
|  | @ -60,43 +123,43 @@ export const customization = async ( | ||||||
|     if (new_queues && new_queues.length > 0) { |     if (new_queues && new_queues.length > 0) { | ||||||
|       const db_queues: any = await Queue.findAll(); |       const db_queues: any = await Queue.findAll(); | ||||||
| 
 | 
 | ||||||
|       for (const i in new_queues) { |       // for (const i in new_queues) {
 | ||||||
|         let { queueName: name, color, greetingMessage } = new_queues[i]; |       //   let { queueName: name, color, greetingMessage } = new_queues[i];
 | ||||||
| 
 | 
 | ||||||
|         name = name?.trim()?.replace(/\s+/g, " "); |       //   name = name?.trim()?.replace(/\s+/g, " ");
 | ||||||
| 
 | 
 | ||||||
|         const update = db_queues.find( |       //   const update = db_queues.find(
 | ||||||
|           (q: any) => q.name?.trim()?.replace(/\s+/g, " ") == name |       //     (q: any) => q.name?.trim()?.replace(/\s+/g, " ") == name
 | ||||||
|         ); |       //   );
 | ||||||
| 
 | 
 | ||||||
|         if (update) { |       //   if (update) {
 | ||||||
|           const { id } = update; |       //     const { id } = update;
 | ||||||
|           // UPDATE
 |       //     // UPDATE
 | ||||||
|           // const queue = await UpdateQueueService(id, {
 |       //     // const queue = await UpdateQueueService(id, {
 | ||||||
|           //   name,
 |       //     //   name,
 | ||||||
|           //   color,
 |       //     //   color,
 | ||||||
|           //   greetingMessage
 |       //     //   greetingMessage
 | ||||||
|           // });
 |       //     // });
 | ||||||
| 
 | 
 | ||||||
|           // const io = getIO();
 |       //     // const io = getIO();
 | ||||||
|           // io.emit("queue", {
 |       //     // io.emit("queue", {
 | ||||||
|           //   action: "update",
 |       //     //   action: "update",
 | ||||||
|           //   queue
 |       //     //   queue
 | ||||||
|           // });
 |       //     // });
 | ||||||
|         } else { |       //   } else {
 | ||||||
|           // CREATE
 |       //     // CREATE
 | ||||||
|           // const queue = await CreateQueueService({
 |       //     // const queue = await CreateQueueService({
 | ||||||
|           //   name,
 |       //     //   name,
 | ||||||
|           //   color,
 |       //     //   color,
 | ||||||
|           //   greetingMessage
 |       //     //   greetingMessage
 | ||||||
|           // });
 |       //     // });
 | ||||||
|           // const io = getIO();
 |       //     // const io = getIO();
 | ||||||
|           // io.emit("queue", {
 |       //     // io.emit("queue", {
 | ||||||
|           //   action: "update",
 |       //     //   action: "update",
 | ||||||
|           //   queue
 |       //     //   queue
 | ||||||
|           // });
 |       //     // });
 | ||||||
|         } |       //   }
 | ||||||
|       } |       // }
 | ||||||
| 
 | 
 | ||||||
|       let remove_queues = db_queues.filter( |       let remove_queues = db_queues.filter( | ||||||
|         (q: any) => |         (q: any) => | ||||||
|  | @ -162,6 +225,9 @@ export const remove = async ( | ||||||
| 
 | 
 | ||||||
|   await DeleteQueueService(queueId); |   await DeleteQueueService(queueId); | ||||||
| 
 | 
 | ||||||
|  |   await QuickAnswerQueue.destroy({ where: { queueId } }); | ||||||
|  |   await ContactQueue.destroy({ where: { queueId } }); | ||||||
|  | 
 | ||||||
|   await del(`queue:${queueId}`); |   await del(`queue:${queueId}`); | ||||||
| 
 | 
 | ||||||
|   const io = getIO(); |   const io = getIO(); | ||||||
|  |  | ||||||
|  | @ -9,23 +9,27 @@ import UpdateQuickAnswerService from "../services/QuickAnswerService/UpdateQuick | ||||||
| import DeleteQuickAnswerService from "../services/QuickAnswerService/DeleteQuickAnswerService"; | import DeleteQuickAnswerService from "../services/QuickAnswerService/DeleteQuickAnswerService"; | ||||||
| 
 | 
 | ||||||
| import AppError from "../errors/AppError"; | import AppError from "../errors/AppError"; | ||||||
|  | import QuickAnswerQueue from "../models/QuickAnswerQueue"; | ||||||
| 
 | 
 | ||||||
| type IndexQuery = { | type IndexQuery = { | ||||||
|   searchParam: string; |   searchParam: string; | ||||||
|   pageNumber: string; |   pageNumber: string; | ||||||
|  |   userId?: string; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| interface QuickAnswerData { | interface QuickAnswerData { | ||||||
|   shortcut: string; |   shortcut: string; | ||||||
|   message: string; |   message: string; | ||||||
|  |   queueIds?: number[]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export const index = async (req: Request, res: Response): Promise<Response> => { | export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   const { searchParam, pageNumber } = req.query as IndexQuery; |   const { searchParam, pageNumber, userId } = req.query as IndexQuery; | ||||||
| 
 | 
 | ||||||
|   const { quickAnswers, count, hasMore } = await ListQuickAnswerService({ |   const { quickAnswers, count, hasMore } = await ListQuickAnswerService({ | ||||||
|     searchParam, |     searchParam, | ||||||
|     pageNumber |     pageNumber, | ||||||
|  |     userId | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   return res.json({ quickAnswers, count, hasMore }); |   return res.json({ quickAnswers, count, hasMore }); | ||||||
|  | @ -41,7 +45,7 @@ export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
| 
 | 
 | ||||||
|   try { |   try { | ||||||
|     await QuickAnswerSchema.validate(newQuickAnswer); |     await QuickAnswerSchema.validate(newQuickAnswer); | ||||||
|   } catch (err:any) { |   } catch (err: any) { | ||||||
|     throw new AppError(err.message); |     throw new AppError(err.message); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -59,9 +63,9 @@ export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const show = async (req: Request, res: Response): Promise<Response> => { | export const show = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   const { quickAnswerId } = req.params; |   const { quickAnswerId, userId } = req.params; | ||||||
| 
 | 
 | ||||||
|   const quickAnswer = await ShowQuickAnswerService(quickAnswerId); |   const quickAnswer = await ShowQuickAnswerService(quickAnswerId, userId); | ||||||
| 
 | 
 | ||||||
|   return res.status(200).json(quickAnswer); |   return res.status(200).json(quickAnswer); | ||||||
| }; | }; | ||||||
|  | @ -103,10 +107,12 @@ export const remove = async ( | ||||||
|   req: Request, |   req: Request, | ||||||
|   res: Response |   res: Response | ||||||
| ): Promise<Response> => { | ): Promise<Response> => { | ||||||
|   const { quickAnswerId } = req.params; |   const { quickAnswerId, queueId } = req.params; | ||||||
| 
 | 
 | ||||||
|   await DeleteQuickAnswerService(quickAnswerId); |   await DeleteQuickAnswerService(quickAnswerId); | ||||||
| 
 | 
 | ||||||
|  |   await QuickAnswerQueue.destroy({ where: { quickAnswerId } }); | ||||||
|  | 
 | ||||||
|   const io = getIO(); |   const io = getIO(); | ||||||
|   io.emit("quickAnswer", { |   io.emit("quickAnswer", { | ||||||
|     action: "delete", |     action: "delete", | ||||||
|  |  | ||||||
|  | @ -20,6 +20,7 @@ import ShowQueuesByUser from "../services/UserServices/ShowQueuesByUser"; | ||||||
| import { getIO } from "../libs/socket"; | import { getIO } from "../libs/socket"; | ||||||
| import { Json } from "sequelize/types/lib/utils"; | import { Json } from "sequelize/types/lib/utils"; | ||||||
| import ReportByNumberQueueService from "../services/ReportServices/ReportByNumberQueueService"; | import ReportByNumberQueueService from "../services/ReportServices/ReportByNumberQueueService"; | ||||||
|  | import CountStatusChatEndService from "../services/StatusChatEndService/CountStatusChatEndService"; | ||||||
| 
 | 
 | ||||||
| type IndexQuery = { | type IndexQuery = { | ||||||
|   userId: string; |   userId: string; | ||||||
|  | @ -261,9 +262,7 @@ export const reportMessagesUserByDateStartDateEnd = async ( | ||||||
|       data_query_messages[i].fromMe = "Cliente"; |       data_query_messages[i].fromMe = "Cliente"; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     data_query_messages[i].id = i + 1; |     data_query_messages[i].id = i + 1;  | ||||||
| 
 |  | ||||||
|     console.log("data_query_messages: ", data_query_messages[i]); |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return res.status(200).json(data_query_messages); |   return res.status(200).json(data_query_messages); | ||||||
|  | @ -312,7 +311,7 @@ export const reportService = async ( | ||||||
|   const reportService = await ReportByNumberQueueService({ |   const reportService = await ReportByNumberQueueService({ | ||||||
|     startDate, |     startDate, | ||||||
|     endDate |     endDate | ||||||
|   });  |   }); | ||||||
| 
 | 
 | ||||||
|   return res.status(200).json({ reportService }); |   return res.status(200).json({ reportService }); | ||||||
| }; | }; | ||||||
|  | @ -331,15 +330,37 @@ export const reportServiceByQueue = async ( | ||||||
| 
 | 
 | ||||||
|   const { startDate, endDate, queueId } = req.query as IndexQuery; |   const { startDate, endDate, queueId } = req.query as IndexQuery; | ||||||
| 
 | 
 | ||||||
|   console.log( |  | ||||||
|     `startDate: ${startDate} | endDate: ${endDate} | queueId: ${queueId}` |  | ||||||
|   ); |  | ||||||
| 
 |  | ||||||
|   const reportService = await ReportByNumberQueueService({ |   const reportService = await ReportByNumberQueueService({ | ||||||
|     startDate, |     startDate, | ||||||
|     endDate, |     endDate, | ||||||
|     queue: true |     queue: true | ||||||
|   });  |   }); | ||||||
| 
 | 
 | ||||||
|   return res.status(200).json({ reportService }); |   return res.status(200).json({ reportService }); | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
|  | export const reportTicksCountByStatusChatEnds = async ( | ||||||
|  |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   if ( | ||||||
|  |     req.user.profile !== "master" && | ||||||
|  |     req.user.profile !== "admin" && | ||||||
|  |     req.user.profile !== "supervisor" | ||||||
|  |   ) { | ||||||
|  |     throw new AppError("ERR_NO_PERMISSION", 403); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const { startDate, endDate } = req.query as IndexQuery; | ||||||
|  | 
 | ||||||
|  |   const dateToday = splitDateTime( | ||||||
|  |     new Date(format(new Date(), "yyyy-MM-dd HH:mm:ss", { locale: ptBR })) | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   const reportStatusChatEnd = await CountStatusChatEndService( | ||||||
|  |     startDate || dateToday.fullDate, | ||||||
|  |     endDate || dateToday.fullDate | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json({ reportStatusChatEnd }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | @ -4,9 +4,9 @@ import { getIO } from "../libs/socket"; | ||||||
| import DeleteSchedulingNotifyService from "../services/SchedulingNotifyServices/DeleteSchedulingNotifyService"; | import DeleteSchedulingNotifyService from "../services/SchedulingNotifyServices/DeleteSchedulingNotifyService"; | ||||||
| import ListSchedulingNotifyContactService from "../services/SchedulingNotifyServices/ListSchedulingNotifyContactService"; | import ListSchedulingNotifyContactService from "../services/SchedulingNotifyServices/ListSchedulingNotifyContactService"; | ||||||
| import CreateSchedulingNotifyService from "../services/SchedulingNotifyServices/CreateSchedulingNotifyService"; | import CreateSchedulingNotifyService from "../services/SchedulingNotifyServices/CreateSchedulingNotifyService"; | ||||||
| import ShowSchedulingNotifyService from "../services/SchedulingNotifyServices/ShowSchedulingNotifyService";  | import ShowSchedulingNotifyService from "../services/SchedulingNotifyServices/ShowSchedulingNotifyService"; | ||||||
| import { deleteScheduleByTicketIdCache } from "../helpers/SchedulingNotifyCache"; | import { deleteScheduleByTicketIdCache } from "../helpers/SchedulingNotifyCache"; | ||||||
| 
 | import ShowStatusChatEndService from "../services/StatusChatEndService/ShowStatusChatEndService"; | ||||||
| 
 | 
 | ||||||
| type IndexQuery = { | type IndexQuery = { | ||||||
|   contactNumber: string; |   contactNumber: string; | ||||||
|  | @ -14,58 +14,58 @@ type IndexQuery = { | ||||||
|   endDate: string; |   endDate: string; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | export const reportScheduleNotifyByDateStartDateEnd = async ( | ||||||
|  |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   const { contactNumber, startDate, endDate } = req.query as IndexQuery; | ||||||
| 
 | 
 | ||||||
| 
 |   const data_query = await ListSchedulingNotifyContactService( | ||||||
| export const reportScheduleNotifyByDateStartDateEnd = async (req: Request, res: Response): Promise<Response> => { |     contactNumber, | ||||||
| 
 |     startDate, | ||||||
|   const { contactNumber, startDate, endDate } = req.query as IndexQuery |     endDate | ||||||
| 
 |   ); | ||||||
|   const data_query = await ListSchedulingNotifyContactService(contactNumber, startDate, endDate); |   | ||||||
| 
 |  | ||||||
|   // console.group('DATA QUERY SCHEDULE:\n',data_query) 
 |  | ||||||
| 
 |  | ||||||
|   return res.status(200).json(data_query); |   return res.status(200).json(data_query); | ||||||
| 
 |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| 
 | export const createOrUpdateScheduleNotify = async ( | ||||||
| export const createOrUpdateScheduleNotify = async (req: Request, res: Response): Promise<Response> => { |   req: Request, | ||||||
| 
 |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|   const scheduleData = req.body; |   const scheduleData = req.body; | ||||||
|  |    | ||||||
|  |   const statusChatEnd = await ShowStatusChatEndService({ | ||||||
|  |     name: scheduleData.statusChatEndName | ||||||
|  |   }); | ||||||
| 
 | 
 | ||||||
|  |   const schedulingNotifyCreate = await CreateSchedulingNotifyService({ | ||||||
|  |     schedulingNotifyId: scheduleData.schedulingNotifyId, | ||||||
|  |     ticketId: scheduleData.ticketId, | ||||||
|  |     statusChatEndId: statusChatEnd.id, | ||||||
|  |     schedulingDate: scheduleData.schedulingDate, | ||||||
|  |     schedulingTime: scheduleData.schedulingTime, | ||||||
|  |     message: scheduleData.message | ||||||
|  |   }); | ||||||
| 
 | 
 | ||||||
| 
 |   // console.group(':::::::::::::::::: DATA schedulingNotifyCreate:\n',schedulingNotifyCreate)
 | ||||||
|   const schedulingNotifyCreate = await CreateSchedulingNotifyService( |   //  const io = getIO();
 | ||||||
|     { |   //  io.emit("schedulingNotify", {action: "update", schedulingNotifyCreate });
 | ||||||
|       schedulingNotifyId: scheduleData.schedulingNotifyId, |  | ||||||
|       ticketId: scheduleData.ticketId, |  | ||||||
|       statusChatEndId: scheduleData.statusChatEndId, |  | ||||||
|       schedulingDate: scheduleData.schedulingDate, |  | ||||||
|       schedulingTime: scheduleData.schedulingTime, |  | ||||||
|       message: scheduleData.message |  | ||||||
|     } |  | ||||||
|   ) |  | ||||||
| 
 |  | ||||||
|   // console.group(':::::::::::::::::: DATA schedulingNotifyCreate:\n',schedulingNotifyCreate)  
 |  | ||||||
|   //  const io = getIO();   
 |  | ||||||
|   //  io.emit("schedulingNotify", {action: "update", schedulingNotifyCreate });   
 |  | ||||||
| 
 | 
 | ||||||
|   return res.status(200).json(schedulingNotifyCreate); |   return res.status(200).json(schedulingNotifyCreate); | ||||||
| 
 |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const remove = async (req: Request, res: Response): Promise<Response> => { | export const remove = async ( | ||||||
| 
 |   req: Request, | ||||||
| 
 |   res: Response | ||||||
| 
 | ): Promise<Response> => { | ||||||
|   const { scheduleId } = req.params; |   const { scheduleId } = req.params; | ||||||
| 
 | 
 | ||||||
|   let schedule: any = await ShowSchedulingNotifyService(scheduleId) |   let schedule: any = await ShowSchedulingNotifyService(scheduleId); | ||||||
| 
 | 
 | ||||||
|   await deleteScheduleByTicketIdCache(schedule.ticketId) |   await deleteScheduleByTicketIdCache(schedule.ticketId); | ||||||
| 
 | 
 | ||||||
|   await DeleteSchedulingNotifyService(scheduleId); |   await DeleteSchedulingNotifyService(scheduleId); | ||||||
| 
 | 
 | ||||||
|   return res.status(200).send(); |   return res.status(200).send(); | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -16,18 +16,16 @@ export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   //   throw new AppError("ERR_NO_PERMISSION", 403);
 |   //   throw new AppError("ERR_NO_PERMISSION", 403);
 | ||||||
|   // }
 |   // }
 | ||||||
| 
 | 
 | ||||||
|   const settings = await ListSettingsService(); |   const settings = await ListSettingsService();   | ||||||
| 
 | 
 | ||||||
|   // const config = await SettingTicket.findAll();
 |   return res.status(200).json({ settings }); | ||||||
| 
 |  | ||||||
|   return res.status(200).json({ settings, }); |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const ticketSettings = async ( | export const ticketSettings = async ( | ||||||
|   req: Request, |   req: Request, | ||||||
|   res: Response |   res: Response | ||||||
| ): Promise<Response> => { | ): Promise<Response> => { | ||||||
|   const { number } = req.params;  |   const { number } = req.params; | ||||||
| 
 | 
 | ||||||
|   const config = await SettingTicket.findAll({ where: { number } }); |   const config = await SettingTicket.findAll({ where: { number } }); | ||||||
| 
 | 
 | ||||||
|  | @ -40,6 +38,7 @@ export const updateTicketSettings = async ( | ||||||
| ): Promise<Response> => { | ): Promise<Response> => { | ||||||
|   const { |   const { | ||||||
|     number, |     number, | ||||||
|  |     saturdayBusinessTime, | ||||||
|     outBusinessHours, |     outBusinessHours, | ||||||
|     ticketExpiration, |     ticketExpiration, | ||||||
|     weekend, |     weekend, | ||||||
|  | @ -47,7 +46,7 @@ export const updateTicketSettings = async ( | ||||||
|     sunday, |     sunday, | ||||||
|     holiday |     holiday | ||||||
|   } = req.body; |   } = req.body; | ||||||
|   | 
 | ||||||
|   if (!number) throw new AppError("No number selected", 400); |   if (!number) throw new AppError("No number selected", 400); | ||||||
| 
 | 
 | ||||||
|   if (outBusinessHours && Object.keys(outBusinessHours).length > 0) { |   if (outBusinessHours && Object.keys(outBusinessHours).length > 0) { | ||||||
|  | @ -58,6 +57,14 @@ export const updateTicketSettings = async ( | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   if (saturdayBusinessTime && Object.keys(saturdayBusinessTime).length > 0) { | ||||||
|  |     await updateSettingTicket({ | ||||||
|  |       ...saturdayBusinessTime, | ||||||
|  |       key: "saturdayBusinessTime", | ||||||
|  |       number | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   if (ticketExpiration && Object.keys(ticketExpiration).length > 0) { |   if (ticketExpiration && Object.keys(ticketExpiration).length > 0) { | ||||||
|     await updateSettingTicket({ |     await updateSettingTicket({ | ||||||
|       ...ticketExpiration, |       ...ticketExpiration, | ||||||
|  |  | ||||||
|  | @ -1,12 +1,131 @@ | ||||||
| import { Request, Response } from "express"; | import { Request, Response } from "express"; | ||||||
| import AppError from "../errors/AppError"; | import AppError from "../errors/AppError"; | ||||||
|    | import * as Yup from "yup"; | ||||||
| import ListStatusChatEndService from  "../services/StatusChatEndService/ListStatusChatEndService";  |  | ||||||
| 
 | 
 | ||||||
| export const show = async (req: Request, res: Response): Promise<Response> => {   | type IndexQuery = { | ||||||
|    |   searchParam: string; | ||||||
|     const { statusChatEnd, count, hasMore } = await ListStatusChatEndService({ searchParam: "", pageNumber: "1" });    |   pageNumber: string; | ||||||
|    | }; | ||||||
|     return res.status(200).json(statusChatEnd); | 
 | ||||||
|  | interface StatusChatEndData { | ||||||
|  |   name: string; | ||||||
|  |   farewellMessage: string; | ||||||
|  |   isDefault: boolean; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | import ListStatusChatEndService from "../services/StatusChatEndService/ListStatusChatEndService"; | ||||||
|  | import ShowStatusChatEndService from "../services/StatusChatEndService/ShowStatusChatEndService"; | ||||||
|  | import UpdateStatusChatEndService from "../services/StatusChatEndService/UpdateStatusChatEndService"; | ||||||
|  | import { getIO } from "../libs/socket"; | ||||||
|  | import StatusChatEnd from "../models/StatusChatEnd"; | ||||||
|  | import CreateStatusChatEndService from "../services/StatusChatEndService/CreateStatusChatEndService"; | ||||||
|  | import { del } from "../helpers/RedisClient"; | ||||||
|  | 
 | ||||||
|  | // export const show = async (req: Request, res: Response): Promise<Response> => {
 | ||||||
|  | 
 | ||||||
|  | //     const { statusChatEnd, count, hasMore } = await ListStatusChatEndService({ searchParam: "", pageNumber: "1" });
 | ||||||
|  | 
 | ||||||
|  | //     return res.status(200).json(statusChatEnd);
 | ||||||
|  | // };
 | ||||||
|  | 
 | ||||||
|  | export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const { searchParam, pageNumber } = req.query as IndexQuery; | ||||||
|  | 
 | ||||||
|  |   const { statusChatEnd, count, hasMore } = await ListStatusChatEndService({ | ||||||
|  |     searchParam, | ||||||
|  |     pageNumber | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.json({ statusChatEnd, count, hasMore }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const show = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const { statusChatEndId } = req.params; | ||||||
|  | 
 | ||||||
|  |   let statushCatEnd; | ||||||
|  | 
 | ||||||
|  |   const isNumber = (str: any) => !isNaN(str); | ||||||
|  | 
 | ||||||
|  |   if (isNumber(statusChatEndId)) | ||||||
|  |     statushCatEnd = await ShowStatusChatEndService({ id: statusChatEndId }); | ||||||
|  |   else statushCatEnd = await ShowStatusChatEndService({ isDefault: true }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(statushCatEnd); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  |   const newStatusChatEnd: StatusChatEndData = req.body; | ||||||
|  | 
 | ||||||
|  |   const StatusChatEndSchema = Yup.object().shape({ | ||||||
|  |     name: Yup.string().required() | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   try { | ||||||
|  |     await StatusChatEndSchema.validate(newStatusChatEnd); | ||||||
|  |   } catch (err: any) { | ||||||
|  |     throw new AppError(err.message); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const statusChatEnd = await CreateStatusChatEndService({ | ||||||
|  |     ...newStatusChatEnd | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const io = getIO(); | ||||||
|  |   io.emit("statusChatEnd", { | ||||||
|  |     action: "create", | ||||||
|  |     statusChatEnd | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(statusChatEnd); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const update = async ( | ||||||
|  |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   const statusChatEndData: StatusChatEndData = req.body; | ||||||
|  | 
 | ||||||
|  |   const schema = Yup.object().shape({ | ||||||
|  |     name: Yup.string() | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   try { | ||||||
|  |     await schema.validate(statusChatEndData); | ||||||
|  |   } catch (err: any) { | ||||||
|  |     throw new AppError(err.message); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const { statusChatEndId } = req.params; | ||||||
|  | 
 | ||||||
|  |   const statusChatEnd = await UpdateStatusChatEndService({ | ||||||
|  |     statusChatEndData, | ||||||
|  |     statusChatEndId | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const io = getIO(); | ||||||
|  |   io.emit("statusChatEnd", { | ||||||
|  |     action: "update", | ||||||
|  |     statusChatEnd | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json(statusChatEnd); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const remove = async ( | ||||||
|  |   req: Request, | ||||||
|  |   res: Response | ||||||
|  | ): Promise<Response> => { | ||||||
|  |   const { statusChatEndId } = req.params; | ||||||
|  | 
 | ||||||
|  |   await StatusChatEnd.destroy({ where: { id: statusChatEndId } }); | ||||||
|  | 
 | ||||||
|  |   await del(`statusChatEnd:${statusChatEndId}`); | ||||||
|  | 
 | ||||||
|  |   const io = getIO(); | ||||||
|  |   io.emit("statusChatEnd", { | ||||||
|  |     action: "delete", | ||||||
|  |     statusChatEndId | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return res.status(200).json({ message: "Status chat deleted" }); | ||||||
| }; | }; | ||||||
|    |  | ||||||
|  | @ -22,7 +22,7 @@ import format from "date-fns/format"; | ||||||
| import ListTicketsServiceCache from "../services/TicketServices/ListTicketServiceCache"; | import ListTicketsServiceCache from "../services/TicketServices/ListTicketServiceCache"; | ||||||
| 
 | 
 | ||||||
| import { searchTicketCache, loadTicketsCache } from "../helpers/TicketCache"; | import { searchTicketCache, loadTicketsCache } from "../helpers/TicketCache"; | ||||||
| import { Op } from "sequelize"; | import { Op, where } from "sequelize"; | ||||||
| 
 | 
 | ||||||
| type IndexQuery = { | type IndexQuery = { | ||||||
|   searchParam: string; |   searchParam: string; | ||||||
|  | @ -75,9 +75,12 @@ import GetProfilePicUrl from "../services/WbotServices/GetProfilePicUrl"; | ||||||
| import CreateContactService from "../services/ContactServices/CreateContactService"; | import CreateContactService from "../services/ContactServices/CreateContactService"; | ||||||
| import { botSendMessage } from "../services/WbotServices/wbotMessageListener"; | import { botSendMessage } from "../services/WbotServices/wbotMessageListener"; | ||||||
| import WhatsappQueue from "../models/WhatsappQueue"; | import WhatsappQueue from "../models/WhatsappQueue"; | ||||||
| import { get } from "../helpers/RedisClient" | import { get } from "../helpers/RedisClient"; | ||||||
|  | import CountStatusChatEndService from "../services/StatusChatEndService/CountStatusChatEndService"; | ||||||
|  | import Queue from "../models/Queue"; | ||||||
|  | import StatusChatEnd from "../models/StatusChatEnd"; | ||||||
| 
 | 
 | ||||||
| export const index = async (req: Request, res: Response): Promise<Response> => {   | export const index = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   const { |   const { | ||||||
|     pageNumber, |     pageNumber, | ||||||
|     status, |     status, | ||||||
|  | @ -118,10 +121,19 @@ export const remoteTicketCreation = async ( | ||||||
|   req: Request, |   req: Request, | ||||||
|   res: Response |   res: Response | ||||||
| ): Promise<Response> => { | ): Promise<Response> => { | ||||||
|   const { contact_from, contact_to, msg, contact_name }: any = req.body; |   let { queueId, contact_from, cc, contact_to, msg, contact_name }: any = | ||||||
|  |     req.body; | ||||||
| 
 | 
 | ||||||
|   const validate = ["contact_from", "contact_to", "msg"]; |   let whatsappId: any; | ||||||
|   const validateOnlyNumber = ["contact_from", "contact_to"]; | 
 | ||||||
|  |   if (!queueId && !contact_from && !cc) { | ||||||
|  |     return res.status(400).json({ | ||||||
|  |       error: `Property 'queueId' or 'contact_from' or 'cc' is required.` | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const validate = ["contact_to", "msg"]; | ||||||
|  |   const validateOnlyNumber = ["queueId", "contact_to", "contact_from"]; | ||||||
| 
 | 
 | ||||||
|   for (let prop of validate) { |   for (let prop of validate) { | ||||||
|     if (!req.body[prop]) |     if (!req.body[prop]) | ||||||
|  | @ -138,109 +150,144 @@ export const remoteTicketCreation = async ( | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const whatsapp = await Whatsapp.findOne({ |   if (queueId) { | ||||||
|     where: { number: contact_from, status: "CONNECTED" } |     const whatsapps = await ListWhatsAppsForQueueService(queueId, "CONNECTED"); | ||||||
|   }); |  | ||||||
| 
 | 
 | ||||||
|   if (whatsapp) { |     if (!whatsapps || whatsapps?.length == 0) { | ||||||
|     const { id: whatsappId, number, status } = whatsapp; |       return res.status(500).json({ | ||||||
| 
 |         msg: `queueId ${queueId} does not have a WhatsApp number associated with it or the number's session is disconnected.` | ||||||
|     const queue: any = await WhatsappQueue.findOne({ |  | ||||||
|       where: { whatsappId }, |  | ||||||
|       attributes: ["queueId"] |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     const { queueId } = queue; |  | ||||||
| 
 |  | ||||||
|     // const validNumber = await CheckIsValidContact(contact_to, true);
 |  | ||||||
|     const validNumber = contact_to; |  | ||||||
| 
 |  | ||||||
|     if (validNumber) { |  | ||||||
|       let contact = await Contact.findOne({ where: { number: validNumber } }); |  | ||||||
| 
 |  | ||||||
|       if (!contact) { |  | ||||||
|         // const profilePicUrl = await GetProfilePicUrl(validNumber);
 |  | ||||||
| 
 |  | ||||||
|         contact = await CreateContactService({ |  | ||||||
|           name: contact_name ? contact_name : contact_to, |  | ||||||
|           number: validNumber |  | ||||||
|           // profilePicUrl
 |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         const io = getIO(); |  | ||||||
|         io.emit("contact", { |  | ||||||
|           action: "create", |  | ||||||
|           contact |  | ||||||
|         }); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       const { id: contactId } = contact; |  | ||||||
| 
 |  | ||||||
|       const botInfo = await BotIsOnQueue("botqueue"); |  | ||||||
| 
 |  | ||||||
|       let ticket = await Ticket.findOne({ |  | ||||||
|         where: { |  | ||||||
|           [Op.or]: [ |  | ||||||
|             { contactId, status: "queueChoice" }, |  | ||||||
|             { contactId, status: "open", userId: botInfo.userIdBot } |  | ||||||
|           ] |  | ||||||
|         } |  | ||||||
|       }); |       }); | ||||||
| 
 |  | ||||||
|       if (getSettingValue("whatsaAppCloudApi")?.value == "enabled") { |  | ||||||
|         if (ticket) { |  | ||||||
|           await UpdateTicketService({ |  | ||||||
|             ticketData: { status: "closed" }, |  | ||||||
|             ticketId: ticket.id |  | ||||||
|           }); |  | ||||||
|           ticket = null; |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         if (ticket) { |  | ||||||
|           await UpdateTicketService({ |  | ||||||
|             ticketData: { status: "closed" }, |  | ||||||
|             ticketId: ticket.id |  | ||||||
|           }); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       if (!ticket) { |  | ||||||
|         ticket = await FindOrCreateTicketService( |  | ||||||
|           contact, |  | ||||||
|           whatsappId, |  | ||||||
|           0, |  | ||||||
|           undefined, |  | ||||||
|           queueId |  | ||||||
|         ); |  | ||||||
|         botSendMessage(ticket, msg); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       const io = getIO(); |  | ||||||
|       io.to(ticket.status).emit("ticket", { |  | ||||||
|         action: "update", |  | ||||||
|         ticket |  | ||||||
|       }); |  | ||||||
| 
 |  | ||||||
|       console.log( |  | ||||||
|         `REMOTE TICKET CREATION FROM ENDPOINT | STATUS: 200 | MSG: success` |  | ||||||
|       ); |  | ||||||
|       return res.status(200).json({ msg: "success" }); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     const { id } = whatsapps[0]; | ||||||
|  | 
 | ||||||
|  |     whatsappId = id; | ||||||
|  |   } else if (contact_from) { | ||||||
|  |     const whatsapp = await Whatsapp.findOne({ | ||||||
|  |       where: { number: contact_from, status: "CONNECTED" } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     if (!whatsapp) { | ||||||
|  |       return res.status(404).json({ | ||||||
|  |         msg: `Whatsapp number ${contact_from} not found or disconnected!` | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const { id } = whatsapp; | ||||||
|  | 
 | ||||||
|  |     const { queues } = await ShowWhatsAppService(id); | ||||||
|  | 
 | ||||||
|  |     if (!queues || queues.length == 0) { | ||||||
|  |       return res.status(500).json({ | ||||||
|  |         msg: `The WhatsApp number ${contact_from} is not associated with any queue! ` | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     queueId = queues[0].id; | ||||||
|  |     whatsappId = id; | ||||||
|  |   } else if (cc) { | ||||||
|  |     const queue = await Queue.findOne({ where: { cc } }); | ||||||
|  |     if (!queue) { | ||||||
|  |       return res.status(404).json({ | ||||||
|  |         msg: `Queue with cc ${cc} not found! ` | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     queueId = queue.id; | ||||||
|  | 
 | ||||||
|  |     const whatsapps = await ListWhatsAppsForQueueService(queueId, "CONNECTED"); | ||||||
|  | 
 | ||||||
|  |     if (whatsapps.length === 0) { | ||||||
|  |       return res.status(500).json({ | ||||||
|  |         msg: `No WhatsApp found for this cc ${cc} or the WhatsApp number is disconnected! ` | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     whatsappId = whatsapps[0].id; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // const validNumber = await CheckIsValidContact(contact_to, true);
 | ||||||
|  |   const validNumber = contact_to; | ||||||
|  | 
 | ||||||
|  |   if (validNumber) { | ||||||
|  |     let contact = await Contact.findOne({ where: { number: validNumber } }); | ||||||
|  | 
 | ||||||
|  |     if (!contact) { | ||||||
|  |       // const profilePicUrl = await GetProfilePicUrl(validNumber);
 | ||||||
|  | 
 | ||||||
|  |       contact = await CreateContactService({ | ||||||
|  |         name: contact_name ? contact_name : contact_to, | ||||||
|  |         number: validNumber | ||||||
|  |         // profilePicUrl
 | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|  |       const io = getIO(); | ||||||
|  |       io.emit("contact", { | ||||||
|  |         action: "create", | ||||||
|  |         contact | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const { id: contactId } = contact; | ||||||
|  | 
 | ||||||
|  |     const botInfo = await BotIsOnQueue("botqueue"); | ||||||
|  | 
 | ||||||
|  |     let ticket = await Ticket.findOne({ | ||||||
|  |       where: { | ||||||
|  |         [Op.or]: [ | ||||||
|  |           { contactId, status: "queueChoice" }, | ||||||
|  |           { contactId, status: "open", userId: botInfo.userIdBot } | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     if (getSettingValue("whatsaAppCloudApi")?.value == "enabled") { | ||||||
|  |       if (ticket) { | ||||||
|  |         await UpdateTicketService({ | ||||||
|  |           ticketData: { status: "closed" }, | ||||||
|  |           ticketId: ticket.id | ||||||
|  |         }); | ||||||
|  |         ticket = null; | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       if (ticket) { | ||||||
|  |         await UpdateTicketService({ | ||||||
|  |           ticketData: { status: "closed" }, | ||||||
|  |           ticketId: ticket.id | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (!ticket) { | ||||||
|  |       ticket = await FindOrCreateTicketService( | ||||||
|  |         contact, | ||||||
|  |         whatsappId, | ||||||
|  |         0, | ||||||
|  |         undefined, | ||||||
|  |         queueId, | ||||||
|  |         true | ||||||
|  |       ); | ||||||
|  |       botSendMessage(ticket, `${msg}`); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const io = getIO(); | ||||||
|  |     io.to(ticket.status).emit("ticket", { | ||||||
|  |       action: "update", | ||||||
|  |       ticket | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|     console.log( |     console.log( | ||||||
|       `REMOTE TICKET CREATION FROM ENDPOINT | STATUS: 500 | MSG: The number ${contact_to} does not exist on WhatsApp` |       `REMOTE TICKET CREATION FROM ENDPOINT | STATUS: 200 | MSG: success` | ||||||
|     ); |     ); | ||||||
|     return res |     return res.status(200).json({ msg: "success" }); | ||||||
|       .status(500) |  | ||||||
|       .json({ msg: `The number ${contact_to} does not exist on WhatsApp` }); |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   console.log( |   console.log( | ||||||
|     `REMOTE TICKET CREATION FROM ENDPOINT | STATUS: 500 | MSG: Whatsapp number ${contact_from} disconnected or it doesn't exist in omnihit` |     `REMOTE TICKET CREATION FROM ENDPOINT | STATUS: 500 | MSG: The number ${contact_to} does not exist on WhatsApp` | ||||||
|   ); |   ); | ||||||
|   return res.status(500).json({ |   return res | ||||||
|     msg: `Whatsapp number ${contact_from} disconnected or it doesn't exist in omnihit` |     .status(500) | ||||||
|   }); |     .json({ msg: `The number ${contact_to} does not exist on WhatsApp` }); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const store = async (req: Request, res: Response): Promise<Response> => { | export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|  | @ -340,34 +387,75 @@ export const update = async ( | ||||||
|     // lembrete
 |     // lembrete
 | ||||||
|     const scheduleData = JSON.parse(schedulingNotifyData); |     const scheduleData = JSON.parse(schedulingNotifyData); | ||||||
| 
 | 
 | ||||||
|     const statusChatEndName = await ShowStatusChatEndService( |     console.log("scheduleData: ", scheduleData); | ||||||
|       scheduleData.statusChatEndId | 
 | ||||||
|     ); |     const statusChatEnd = await ShowStatusChatEndService({ | ||||||
|  |       name: scheduleData.statusChatEndName | ||||||
|  |     }); | ||||||
| 
 | 
 | ||||||
|     const { ticket } = await UpdateTicketService({ |     const { ticket } = await UpdateTicketService({ | ||||||
|       ticketData: { |       ticketData: { | ||||||
|         status: status, |         status: status, | ||||||
|         userId: userId, |         userId: userId, | ||||||
|         statusChatEnd: statusChatEndName.name |         statusChatEnd: statusChatEnd.name, | ||||||
|  |         statusChatEndId: statusChatEnd.id | ||||||
|       }, |       }, | ||||||
|       ticketId |       ticketId | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     if (scheduleData.farewellMessage) { |     let _farewellMessage; | ||||||
|  | 
 | ||||||
|  |     if (getSettingValue("farewellMessageByStatusChatEnd")?.value == "enabled") { | ||||||
|  |       const statusChatEndData = await get({ | ||||||
|  |         key: `statusChatEnd:${statusChatEnd.id}`, | ||||||
|  |         parse: true | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|  |       if ( | ||||||
|  |         statusChatEndData && | ||||||
|  |         statusChatEndData?.farewellMessage && | ||||||
|  |         statusChatEndData?.farewellMessage?.trim()?.length > 0 | ||||||
|  |       ) { | ||||||
|  |         const { farewellMessage } = statusChatEndData; | ||||||
|  | 
 | ||||||
|  |         _farewellMessage = farewellMessage; | ||||||
|  |       } | ||||||
|  |     } else if (getSettingValue("farewellMessageByQueue")?.value == "enabled") { | ||||||
|  |       const queueData = await get({ | ||||||
|  |         key: `queue:${ticket.queueId}`, | ||||||
|  |         parse: true | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|  |       if ( | ||||||
|  |         queueData && | ||||||
|  |         queueData?.farewellMessage && | ||||||
|  |         queueData?.farewellMessage?.trim()?.length > 0 | ||||||
|  |       ) { | ||||||
|  |         const { farewellMessage } = queueData; | ||||||
|  | 
 | ||||||
|  |         _farewellMessage = farewellMessage; | ||||||
|  |       } | ||||||
|  |     } else if (scheduleData.farewellMessage) { | ||||||
|       const whatsapp = await ShowWhatsAppService(ticket.whatsappId); |       const whatsapp = await ShowWhatsAppService(ticket.whatsappId); | ||||||
| 
 | 
 | ||||||
|       const { farewellMessage } = whatsapp; |       const { farewellMessage } = whatsapp; | ||||||
| 
 | 
 | ||||||
|       if (farewellMessage) { |       if (farewellMessage) { | ||||||
|         await SendWhatsAppMessage({ body: farewellMessage, ticket }); |         _farewellMessage = farewellMessage; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     //  lembrete                              // agendamento
 |     if (_farewellMessage) { | ||||||
|  |       await SendWhatsAppMessage({ body: `\u200e${_farewellMessage}`, ticket }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     console.log("tatusChatEnd.name: ", statusChatEnd.name); | ||||||
|  | 
 | ||||||
|     if ( |     if ( | ||||||
|       scheduleData.statusChatEndId === "2" || |       statusChatEnd.name === "LEMBRETE" || | ||||||
|       scheduleData.statusChatEndId === "3" |       statusChatEnd.name === "AGENDAMENTO À CONFIRMAR" | ||||||
|     ) { |     ) { | ||||||
|  |       //  lembrete                              // agendamento
 | ||||||
|       if ( |       if ( | ||||||
|         isScheduling(scheduleData.schedulingDate, scheduleData.schedulingTime) |         isScheduling(scheduleData.schedulingDate, scheduleData.schedulingTime) | ||||||
|       ) { |       ) { | ||||||
|  | @ -377,8 +465,8 @@ export const update = async ( | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       const schedulingNotifyCreate = await CreateSchedulingNotifyService({ |       const schedulingNotifyCreate = await CreateSchedulingNotifyService({ | ||||||
|         ticketId: scheduleData.ticketId, |         ticketId: scheduleData.ticketId,  | ||||||
|         statusChatEndId: scheduleData.statusChatEndId, |         statusChatEndId: `${statusChatEnd.id}`, | ||||||
|         schedulingDate: scheduleData.schedulingDate, |         schedulingDate: scheduleData.schedulingDate, | ||||||
|         schedulingTime: scheduleData.schedulingTime, |         schedulingTime: scheduleData.schedulingTime, | ||||||
|         message: scheduleData.message |         message: scheduleData.message | ||||||
|  | @ -406,7 +494,6 @@ export const update = async ( | ||||||
|           for (const w of whatsappsByqueue) { |           for (const w of whatsappsByqueue) { | ||||||
|             let whats = await ListWhatsAppsNumber(w.id); |             let whats = await ListWhatsAppsNumber(w.id); | ||||||
| 
 | 
 | ||||||
|             console.log("-------> WHATS: ", JSON.stringify(whats, null, 6)); |  | ||||||
|             const ticket = await Ticket.findOne({ |             const ticket = await Ticket.findOne({ | ||||||
|               where: { |               where: { | ||||||
|                 [Op.and]: [ |                 [Op.and]: [ | ||||||
|  |  | ||||||
|  | @ -138,8 +138,15 @@ export const all = async (req: Request, res: Response): Promise<Response> => { | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const store = async (req: Request, res: Response): Promise<Response> => { | export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|   const { email, password, name, profile, positionCompany, queueIds } = |   const { | ||||||
|     req.body; |     email, | ||||||
|  |     password, | ||||||
|  |     name, | ||||||
|  |     profile, | ||||||
|  |     positionCompany, | ||||||
|  |     positionId, | ||||||
|  |     queueIds | ||||||
|  |   } = req.body; | ||||||
| 
 | 
 | ||||||
|   console.log("===========> req.url: ", req.url); |   console.log("===========> req.url: ", req.url); | ||||||
| 
 | 
 | ||||||
|  | @ -163,6 +170,7 @@ export const store = async (req: Request, res: Response): Promise<Response> => { | ||||||
|     password, |     password, | ||||||
|     name, |     name, | ||||||
|     positionCompany, |     positionCompany, | ||||||
|  |     positionId, | ||||||
|     profile, |     profile, | ||||||
|     queueIds |     queueIds | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|  | @ -13,8 +13,11 @@ import QuickAnswer from "../models/QuickAnswer"; | ||||||
| 
 | 
 | ||||||
| import SchedulingNotify from "../models/SchedulingNotify"; | import SchedulingNotify from "../models/SchedulingNotify"; | ||||||
| import StatusChatEnd from "../models/StatusChatEnd"; | import StatusChatEnd from "../models/StatusChatEnd"; | ||||||
| import UserOnlineTime from "../models/UserOnlineTime";  | import UserOnlineTime from "../models/UserOnlineTime"; | ||||||
| import SettingTicket from "../models/SettingTicket"; | import SettingTicket from "../models/SettingTicket"; | ||||||
|  | import QuickAnswerQueue from "../models/QuickAnswerQueue"; | ||||||
|  | import Position from "../models/Position" | ||||||
|  | import ContactQueue from "../models/ContactQueues" | ||||||
| // eslint-disable-next-line
 | // eslint-disable-next-line
 | ||||||
| const dbConfig = require("../config/database"); | const dbConfig = require("../config/database"); | ||||||
| // import dbConfig from "../config/database";
 | // import dbConfig from "../config/database";
 | ||||||
|  | @ -33,11 +36,13 @@ const models = [ | ||||||
|   WhatsappQueue, |   WhatsappQueue, | ||||||
|   UserQueue, |   UserQueue, | ||||||
|   QuickAnswer, |   QuickAnswer, | ||||||
| 
 |   QuickAnswerQueue, | ||||||
|   SchedulingNotify, |   SchedulingNotify, | ||||||
|   StatusChatEnd, |   StatusChatEnd, | ||||||
|   UserOnlineTime, |   UserOnlineTime, | ||||||
|   SettingTicket |   SettingTicket, | ||||||
|  |   Position, | ||||||
|  |   ContactQueue | ||||||
| ]; | ]; | ||||||
| 
 | 
 | ||||||
| sequelize.addModels(models); | sequelize.addModels(models); | ||||||
|  |  | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("Tickets", "statusChatEndId", { | ||||||
|  |       type: DataTypes.INTEGER, | ||||||
|  |       references: { model: "StatusChatEnds", key: "id" }, | ||||||
|  |       onUpdate: "CASCADE", | ||||||
|  |       onDelete: "SET NULL", | ||||||
|  |       allowNull: true | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("Tickets", "statusChatEndId"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("Queues", "cc", { | ||||||
|  |       type: DataTypes.STRING, | ||||||
|  |       allowNull: true | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("Queues", "cc"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,28 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.createTable("QuickAnswerQueues", { | ||||||
|  |       quickAnswerId: { | ||||||
|  |         type: DataTypes.INTEGER, | ||||||
|  |         primaryKey: true | ||||||
|  |       }, | ||||||
|  |       queueId: { | ||||||
|  |         type: DataTypes.INTEGER, | ||||||
|  |         primaryKey: true | ||||||
|  |       }, | ||||||
|  |       createdAt: { | ||||||
|  |         type: DataTypes.DATE, | ||||||
|  |         allowNull: false | ||||||
|  |       }, | ||||||
|  |       updatedAt: { | ||||||
|  |         type: DataTypes.DATE, | ||||||
|  |         allowNull: false | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.dropTable("QuickAnswerQueues"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("Queues", "farewellMessage", { | ||||||
|  |       type: DataTypes.STRING, | ||||||
|  |       allowNull: true | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("Queues", "farewellMessage"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,14 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("StatusChatEnds", "farewellMessage", { | ||||||
|  |       type: DataTypes.STRING, | ||||||
|  |       allowNull: true | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("StatusChatEnds", "farewellMessage"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("StatusChatEnds", "isDefault", { | ||||||
|  |       type: DataTypes.BOOLEAN, | ||||||
|  |       allowNull: false, | ||||||
|  |       defaultValue: false | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("StatusChatEnds", "isDefault"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,30 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.createTable("Positions", { | ||||||
|  |       id: { | ||||||
|  |         type: DataTypes.INTEGER, | ||||||
|  |         autoIncrement: true, | ||||||
|  |         primaryKey: true, | ||||||
|  |         allowNull: false | ||||||
|  |       }, | ||||||
|  |       name: { | ||||||
|  |         type: DataTypes.TEXT, | ||||||
|  |         allowNull: false | ||||||
|  |       }, | ||||||
|  |       createdAt: { | ||||||
|  |         type: DataTypes.DATE, | ||||||
|  |         allowNull: false | ||||||
|  |       }, | ||||||
|  |       updatedAt: { | ||||||
|  |         type: DataTypes.DATE, | ||||||
|  |         allowNull: false | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.dropTable("Positions"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,19 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("Users", "positionId", { | ||||||
|  |       type: DataTypes.INTEGER, | ||||||
|  |         references: { model: "Positions", key: "id" }, | ||||||
|  |         onUpdate: "CASCADE", | ||||||
|  |         onDelete: "SET NULL" | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("Users", "positionId"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   | ||||||
|  | @ -0,0 +1,28 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.createTable("ContactQueues", { | ||||||
|  |       contactId: { | ||||||
|  |         type: DataTypes.INTEGER, | ||||||
|  |         primaryKey: true | ||||||
|  |       }, | ||||||
|  |       queueId: { | ||||||
|  |         type: DataTypes.INTEGER, | ||||||
|  |         primaryKey: true | ||||||
|  |       }, | ||||||
|  |       createdAt: { | ||||||
|  |         type: DataTypes.DATE, | ||||||
|  |         allowNull: false | ||||||
|  |       }, | ||||||
|  |       updatedAt: { | ||||||
|  |         type: DataTypes.DATE, | ||||||
|  |         allowNull: false | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.dropTable("ContactQueues"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | import { QueryInterface, DataTypes } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.addColumn("Tickets", "isRemote", { | ||||||
|  |       type: DataTypes.BOOLEAN, | ||||||
|  |       allowNull: true, | ||||||
|  |       defaultValue: false | ||||||
|  |     }); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.removeColumn("Tickets", "isRemote"); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -1,26 +0,0 @@ | ||||||
| 'use strict'; |  | ||||||
| 
 |  | ||||||
| module.exports = { |  | ||||||
|   up: (queryInterface, Sequelize) => { |  | ||||||
|     /* |  | ||||||
|       Add altering commands here. |  | ||||||
|       Return a promise to correctly handle asynchronicity. |  | ||||||
| 
 |  | ||||||
|       Example: |  | ||||||
|       return queryInterface.bulkInsert('People', [{ |  | ||||||
|         name: 'John Doe', |  | ||||||
|         isBetaMember: false |  | ||||||
|       }], {}); |  | ||||||
|     */ |  | ||||||
|   }, |  | ||||||
| 
 |  | ||||||
|   down: (queryInterface, Sequelize) => { |  | ||||||
|     /* |  | ||||||
|       Add reverting commands here. |  | ||||||
|       Return a promise to correctly handle asynchronicity. |  | ||||||
| 
 |  | ||||||
|       Example: |  | ||||||
|       return queryInterface.bulkDelete('People', null, {}); |  | ||||||
|     */ |  | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  | @ -0,0 +1,25 @@ | ||||||
|  | import { QueryInterface } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkInsert( | ||||||
|  |       "SettingTickets", | ||||||
|  |       [ | ||||||
|  |         { | ||||||
|  |           message: "", | ||||||
|  |           startTime: new Date(), | ||||||
|  |           endTime: new Date(), | ||||||
|  |           value: "disabled", | ||||||
|  |           key: "saturdayBusinessTime", | ||||||
|  |           createdAt: new Date(), | ||||||
|  |           updatedAt: new Date() | ||||||
|  |         }  | ||||||
|  |       ], | ||||||
|  |       {} | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkDelete("SettingTickets", {}); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | import { QueryInterface } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkInsert( | ||||||
|  |       "Settings", | ||||||
|  |       [ | ||||||
|  |         { | ||||||
|  |           key: "quickAnswerByQueue", | ||||||
|  |           value: "disabled", | ||||||
|  |           createdAt: new Date(), | ||||||
|  |           updatedAt: new Date() | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       {} | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkDelete("Settings", {}); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | import { QueryInterface } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkInsert( | ||||||
|  |       "Settings", | ||||||
|  |       [ | ||||||
|  |         { | ||||||
|  |           key: "farewellMessageByQueue", | ||||||
|  |           value: "disabled", | ||||||
|  |           createdAt: new Date(), | ||||||
|  |           updatedAt: new Date() | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       {} | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkDelete("Settings", {}); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | import { QueryInterface } from "sequelize"; | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkInsert( | ||||||
|  |       "Settings", | ||||||
|  |       [ | ||||||
|  |         { | ||||||
|  |           key: "farewellMessageByStatusChatEnd", | ||||||
|  |           value: "disabled", | ||||||
|  |           createdAt: new Date(), | ||||||
|  |           updatedAt: new Date() | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       {} | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkDelete("Settings", {}); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,22 @@ | ||||||
|  | import { QueryInterface } from "sequelize" | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |   up: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkInsert( | ||||||
|  |       "Settings", | ||||||
|  |       [ | ||||||
|  |         { | ||||||
|  |           key: "contactByqueues", | ||||||
|  |           value: "disabled", | ||||||
|  |           createdAt: new Date(), | ||||||
|  |           updatedAt: new Date() | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       {} | ||||||
|  |     ) | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   down: (queryInterface: QueryInterface) => { | ||||||
|  |     return queryInterface.bulkDelete("Settings", {}) | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,29 @@ | ||||||
|  | import QuickAnswer from "../models/QuickAnswer"; | ||||||
|  | 
 | ||||||
|  | export default function quickAnswearByQueueFiltered( | ||||||
|  |   queueIds: any[], | ||||||
|  |   quickAnswers: QuickAnswer[] | ||||||
|  | ) { | ||||||
|  |   let auxQuickAnswear = []; | ||||||
|  |   let repet: any[] = []; | ||||||
|  |   const userQueues = queueIds.map((uq: any) => uq.queueId); | ||||||
|  | 
 | ||||||
|  |   for (const quickAnswer of quickAnswers) { | ||||||
|  |     const { queues, id } = quickAnswer; | ||||||
|  | 
 | ||||||
|  |     if (queues.length == 0) { | ||||||
|  |       auxQuickAnswear.push(quickAnswer); | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (const q of queues) { | ||||||
|  |       if (userQueues.includes(q.id)) { | ||||||
|  |         if (repet.includes(id)) continue; | ||||||
|  |         repet.push(id); | ||||||
|  | 
 | ||||||
|  |         auxQuickAnswear.push(quickAnswer); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return auxQuickAnswear; | ||||||
|  | } | ||||||
|  | @ -5,6 +5,8 @@ interface SerializedUser { | ||||||
|   id: number; |   id: number; | ||||||
|   name: string; |   name: string; | ||||||
|   positionCompany: string; |   positionCompany: string; | ||||||
|  |   positionId: string | number; | ||||||
|  |   position: object; | ||||||
|   email: string; |   email: string; | ||||||
|   profile: string; |   profile: string; | ||||||
|   queues: Queue[]; |   queues: Queue[]; | ||||||
|  | @ -15,8 +17,11 @@ export const SerializeUser = (user: User): SerializedUser => { | ||||||
|     id: user.id, |     id: user.id, | ||||||
|     name: user.name, |     name: user.name, | ||||||
|     positionCompany: user.positionCompany, |     positionCompany: user.positionCompany, | ||||||
|  |     positionId: user.positionId, | ||||||
|  |     position: user.position, | ||||||
|     email: user.email, |     email: user.email, | ||||||
|     profile: user.profile, |     profile: user.profile, | ||||||
|     queues: user.queues |     queues: user.queues, | ||||||
|  |      | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -39,7 +39,7 @@ const isHoliday = async (number: string | number) => { | ||||||
|           locale: ptBR |           locale: ptBR | ||||||
|         }) |         }) | ||||||
|       ) |       ) | ||||||
|     );  |     ); | ||||||
| 
 | 
 | ||||||
|     if (currentDate.fullDate == startTime.fullDate) { |     if (currentDate.fullDate == startTime.fullDate) { | ||||||
|       obj.set = true; |       obj.set = true; | ||||||
|  | @ -62,21 +62,8 @@ const isWeekend = async (number: string | number) => { | ||||||
|     weekend.value == "enabled" && |     weekend.value == "enabled" && | ||||||
|     weekend.message?.trim()?.length > 0 |     weekend.message?.trim()?.length > 0 | ||||||
|   ) { |   ) { | ||||||
|     // Specify your desired timezone
 |  | ||||||
|     const brazilTimeZone = "America/Sao_Paulo"; |  | ||||||
| 
 |  | ||||||
|     const currentDateUtc = new Date(); |  | ||||||
| 
 |  | ||||||
|     // Convert UTC date to Brazil time zone
 |  | ||||||
|     const currentDate = utcToZonedTime(currentDateUtc, brazilTimeZone); |  | ||||||
| 
 |  | ||||||
|     // Format the date using the desired format
 |  | ||||||
|     const formattedDate = _format(currentDate, "yyyy-MM-dd HH:mm:ssXXX"); |  | ||||||
| 
 |  | ||||||
|     const parsedDate = parseISO(formattedDate); |  | ||||||
| 
 |  | ||||||
|     // Convert parsed date to Brazil time zone
 |     // Convert parsed date to Brazil time zone
 | ||||||
|     const localDate = utcToZonedTime(parsedDate, brazilTimeZone); |     const localDate = localDateConvert(); | ||||||
| 
 | 
 | ||||||
|     // Check if it's Saturday or Sunday
 |     // Check if it's Saturday or Sunday
 | ||||||
|     if (isSaturday(localDate)) { |     if (isSaturday(localDate)) { | ||||||
|  | @ -173,8 +160,104 @@ async function isOutBusinessTime(number: string | number) { | ||||||
|   return obj; |   return obj; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export { | async function isOutBusinessTimeSaturday(number: string | number) { | ||||||
|   isWeekend, |   let obj = { set: false, msg: "" }; | ||||||
|   isHoliday, | 
 | ||||||
|   isOutBusinessTime |   // Convert parsed date to Brazil time zone
 | ||||||
| }; |   const localDate = localDateConvert(); | ||||||
|  | 
 | ||||||
|  |   // Check if it's Saturday or Sunday
 | ||||||
|  |   if (!isSaturday(localDate)) { | ||||||
|  |     return obj; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const outBusinessHoursSaturday = await SettingTicket.findOne({ | ||||||
|  |     where: { key: "saturdayBusinessTime", number } | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   let isWithinRange = false; | ||||||
|  | 
 | ||||||
|  |   if ( | ||||||
|  |     outBusinessHoursSaturday && | ||||||
|  |     outBusinessHoursSaturday.value == "enabled" && | ||||||
|  |     outBusinessHoursSaturday?.message?.trim()?.length > 0 | ||||||
|  |   ) { | ||||||
|  |     const ticketDateTimeUpdate = splitDateTime( | ||||||
|  |       new Date( | ||||||
|  |         _format(new Date(), "yyyy-MM-dd HH:mm:ss", { | ||||||
|  |           locale: ptBR | ||||||
|  |         }) | ||||||
|  |       ) | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     const startTime = splitDateTime( | ||||||
|  |       new Date( | ||||||
|  |         _format( | ||||||
|  |           new Date(outBusinessHoursSaturday.startTime), | ||||||
|  |           "yyyy-MM-dd HH:mm:ss", | ||||||
|  |           { | ||||||
|  |             locale: ptBR | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|  |       ) | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     const endTime = splitDateTime( | ||||||
|  |       new Date( | ||||||
|  |         _format( | ||||||
|  |           new Date(outBusinessHoursSaturday.endTime), | ||||||
|  |           "yyyy-MM-dd HH:mm:ss", | ||||||
|  |           { | ||||||
|  |             locale: ptBR | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|  |       ) | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     const format = "HH:mm:ss"; | ||||||
|  |     const parsedStartTime = parse( | ||||||
|  |       ticketDateTimeUpdate.fullTime, | ||||||
|  |       format, | ||||||
|  |       new Date() | ||||||
|  |     ); | ||||||
|  |     const parsedEndTime = parse(startTime.fullTime, format, new Date()); | ||||||
|  |     const parsedTimeToCheck = parse(endTime.fullTime, format, new Date()); | ||||||
|  |     const timeInterval = { start: parsedStartTime, end: parsedEndTime }; | ||||||
|  | 
 | ||||||
|  |     // If the time range spans across different days, handle the date part
 | ||||||
|  |     if (parsedEndTime < parsedStartTime) { | ||||||
|  |       const nextDay = new Date(parsedStartTime); | ||||||
|  |       nextDay.setDate(nextDay.getDate() + 1); | ||||||
|  |       timeInterval.end = nextDay; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     isWithinRange = isWithinInterval(parsedTimeToCheck, timeInterval); | ||||||
|  | 
 | ||||||
|  |     if (!isWithinRange) { | ||||||
|  |       obj.set = true; | ||||||
|  |       obj.msg = outBusinessHoursSaturday.message; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return obj; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function localDateConvert() { | ||||||
|  |   const brazilTimeZone = "America/Sao_Paulo"; | ||||||
|  | 
 | ||||||
|  |   const currentDateUtc = new Date(); | ||||||
|  | 
 | ||||||
|  |   // Convert UTC date to Brazil time zone
 | ||||||
|  |   const currentDate = utcToZonedTime(currentDateUtc, brazilTimeZone); | ||||||
|  | 
 | ||||||
|  |   // Format the date using the desired format
 | ||||||
|  |   const formattedDate = _format(currentDate, "yyyy-MM-dd HH:mm:ssXXX"); | ||||||
|  | 
 | ||||||
|  |   const parsedDate = parseISO(formattedDate); | ||||||
|  | 
 | ||||||
|  |   // Convert parsed date to Brazil time zone
 | ||||||
|  |   const localDate = utcToZonedTime(parsedDate, brazilTimeZone); | ||||||
|  |   return localDate; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export { isWeekend, isHoliday, isOutBusinessTime, isOutBusinessTimeSaturday }; | ||||||
|  |  | ||||||
|  | @ -1,11 +1,16 @@ | ||||||
| import axios from "axios"; | import axios from "axios"; | ||||||
|  | import https from "https" | ||||||
|  | import http from "http" | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| const api = axios.create({ | const api = axios.create({ | ||||||
|   baseURL: process.env.URL_WHATSAPP_API, |   baseURL: process.env.URL_WHATSAPP_API, | ||||||
|   headers: { |   headers: { | ||||||
|     Accept: "application/json", |     Accept: "application/json", | ||||||
|     Authorization: `Bearer ${process.env.TOKEN}` |     Authorization: `Bearer ${process.env.TOKEN}` | ||||||
|   } |   }, | ||||||
|  |   httpAgent: new http.Agent({ keepAlive: true }),  | ||||||
|  |   httpsAgent: new https.Agent({ keepAlive: true }) | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| export default api; | export default api; | ||||||
|  |  | ||||||
|  | @ -53,7 +53,7 @@ async function sendWhatsAppMessageOfficialAPI( | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   console.log("SEND MESSAGE: ", JSON.stringify(data, null,2)); |   console.log("SEND MESSAGE: ", JSON.stringify(data, null,2)); | ||||||
| 
 |   | ||||||
|   whatsappOfficialAPI |   whatsappOfficialAPI | ||||||
|     .post(`/${process.env.VERSION}/${phoneNumberId}/messages`, data) |     .post(`/${process.env.VERSION}/${phoneNumberId}/messages`, data) | ||||||
|     .then(response => { |     .then(response => { | ||||||
|  |  | ||||||
|  | @ -19,10 +19,11 @@ const isAuth = (req: Request, res: Response, next: NextFunction): void => { | ||||||
|     throw new AppError("ERR_SESSION_EXPIRED", 401); |     throw new AppError("ERR_SESSION_EXPIRED", 401); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const [, token] = authHeader.split(" ");   |   const [, token] = authHeader.split(" ");    | ||||||
| 
 | 
 | ||||||
|   if ( |   if ( | ||||||
|     req.originalUrl == "/tickets/remote/create" && |     (req.originalUrl == "/queue/remote/list" || | ||||||
|  |       req.originalUrl == "/tickets/remote/create") && | ||||||
|     token === process.env.TOKEN_REMOTE_TICKET_CREATION |     token === process.env.TOKEN_REMOTE_TICKET_CREATION | ||||||
|   ) { |   ) { | ||||||
|     return next(); |     return next(); | ||||||
|  |  | ||||||
|  | @ -9,10 +9,13 @@ import { | ||||||
|   AllowNull, |   AllowNull, | ||||||
|   Unique, |   Unique, | ||||||
|   Default, |   Default, | ||||||
|   HasMany |   HasMany, | ||||||
|  |   BelongsToMany | ||||||
| } from "sequelize-typescript"; | } from "sequelize-typescript"; | ||||||
| import ContactCustomField from "./ContactCustomField"; | import ContactCustomField from "./ContactCustomField"; | ||||||
| import Ticket from "./Ticket"; | import Ticket from "./Ticket"; | ||||||
|  | import Queue from "./Queue" | ||||||
|  | import ContactQueue from "./ContactQueues" | ||||||
| 
 | 
 | ||||||
| @Table | @Table | ||||||
| class Contact extends Model<Contact> { | class Contact extends Model<Contact> { | ||||||
|  | @ -52,6 +55,12 @@ class Contact extends Model<Contact> { | ||||||
| 
 | 
 | ||||||
|   @HasMany(() => ContactCustomField) |   @HasMany(() => ContactCustomField) | ||||||
|   extraInfo: ContactCustomField[]; |   extraInfo: ContactCustomField[]; | ||||||
|  | 
 | ||||||
|  |   @BelongsToMany(() => Queue, () => ContactQueue) | ||||||
|  |   queues: Array<Queue & { WhatsappQueue: ContactQueue }>; | ||||||
|  | 
 | ||||||
|  |   @HasMany(() => ContactQueue) | ||||||
|  |   contactQueue: ContactQueue[]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export default Contact; | export default Contact; | ||||||
|  |  | ||||||
|  | @ -0,0 +1,33 @@ | ||||||
|  | import { | ||||||
|  |   Table, | ||||||
|  |   Column, | ||||||
|  |   CreatedAt, | ||||||
|  |   UpdatedAt, | ||||||
|  |   Model, | ||||||
|  |   ForeignKey, | ||||||
|  |   BelongsTo | ||||||
|  | } from "sequelize-typescript"; | ||||||
|  | import Queue from "./Queue";  | ||||||
|  | import Contact from "./Contact" | ||||||
|  | 
 | ||||||
|  | @Table | ||||||
|  | class ContactQueue extends Model<ContactQueue> { | ||||||
|  |   @ForeignKey(() => Contact) | ||||||
|  |   @Column | ||||||
|  |   contactId: number; | ||||||
|  | 
 | ||||||
|  |   @ForeignKey(() => Queue) | ||||||
|  |   @Column | ||||||
|  |   queueId: number; | ||||||
|  | 
 | ||||||
|  |   @CreatedAt | ||||||
|  |   createdAt: Date; | ||||||
|  | 
 | ||||||
|  |   @UpdatedAt | ||||||
|  |   updatedAt: Date; | ||||||
|  | 
 | ||||||
|  |   @BelongsTo(() => Queue) | ||||||
|  |   queue: Queue; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default ContactQueue; | ||||||
|  | @ -0,0 +1,48 @@ | ||||||
|  | import { | ||||||
|  |   Table, | ||||||
|  |   Column, | ||||||
|  |   CreatedAt, | ||||||
|  |   UpdatedAt, | ||||||
|  |   Model, | ||||||
|  |   PrimaryKey, | ||||||
|  |   ForeignKey, | ||||||
|  |   BelongsTo, | ||||||
|  |   HasMany, | ||||||
|  |   HasOne, | ||||||
|  |   AutoIncrement, | ||||||
|  |   Default, | ||||||
|  |   DataType,  | ||||||
|  |   Unique | ||||||
|  | } from "sequelize-typescript"; | ||||||
|  | 
 | ||||||
|  | import Contact from "./Contact"; | ||||||
|  | import Message from "./Message"; | ||||||
|  | import Queue from "./Queue"; | ||||||
|  | import User from "./User"; | ||||||
|  | import Whatsapp from "./Whatsapp"; | ||||||
|  | 
 | ||||||
|  | import SchedulingNotify from "./SchedulingNotify"; | ||||||
|  | import StatusChatEnd from "./StatusChatEnd"; | ||||||
|  | 
 | ||||||
|  | @Table | ||||||
|  | class Position extends Model<Position> { | ||||||
|  |   @PrimaryKey | ||||||
|  |   @AutoIncrement | ||||||
|  |   @Column | ||||||
|  |   id: number; | ||||||
|  |   | ||||||
|  |   @Unique | ||||||
|  |   @Column(DataType.TEXT) | ||||||
|  |   name: string; | ||||||
|  | 
 | ||||||
|  |   @CreatedAt | ||||||
|  |   createdAt: Date; | ||||||
|  | 
 | ||||||
|  |   @UpdatedAt | ||||||
|  |   updatedAt: Date; | ||||||
|  | 
 | ||||||
|  |   @HasMany(() => User) | ||||||
|  |   users: User[]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Position; | ||||||
|  | @ -36,6 +36,12 @@ class Queue extends Model<Queue> { | ||||||
|   @Column |   @Column | ||||||
|   greetingMessage: string; |   greetingMessage: string; | ||||||
| 
 | 
 | ||||||
|  |   @Column | ||||||
|  |   farewellMessage: string; | ||||||
|  | 
 | ||||||
|  |   @Column | ||||||
|  |   cc: string; | ||||||
|  | 
 | ||||||
|   @CreatedAt |   @CreatedAt | ||||||
|   createdAt: Date; |   createdAt: Date; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -6,8 +6,12 @@ import { | ||||||
|   UpdatedAt, |   UpdatedAt, | ||||||
|   Model, |   Model, | ||||||
|   PrimaryKey, |   PrimaryKey, | ||||||
|   AutoIncrement |   AutoIncrement, | ||||||
|  |   BelongsToMany, | ||||||
|  |   HasMany | ||||||
| } from "sequelize-typescript"; | } from "sequelize-typescript"; | ||||||
|  | import Queue from "./Queue"; | ||||||
|  | import QuickAnswerQueue from "./QuickAnswerQueue"; | ||||||
| 
 | 
 | ||||||
| @Table | @Table | ||||||
| class QuickAnswer extends Model<QuickAnswer> { | class QuickAnswer extends Model<QuickAnswer> { | ||||||
|  | @ -27,6 +31,12 @@ class QuickAnswer extends Model<QuickAnswer> { | ||||||
| 
 | 
 | ||||||
|   @UpdatedAt |   @UpdatedAt | ||||||
|   updatedAt: Date; |   updatedAt: Date; | ||||||
|  | 
 | ||||||
|  |   @BelongsToMany(() => Queue, () => QuickAnswerQueue) | ||||||
|  |   queues: Array<Queue & { QuickAnswerQueue: QuickAnswerQueue }>; | ||||||
|  | 
 | ||||||
|  |   @HasMany(() => QuickAnswerQueue) | ||||||
|  |   quickAnswerQueue: QuickAnswerQueue[]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export default QuickAnswer; | export default QuickAnswer; | ||||||
|  |  | ||||||
|  | @ -0,0 +1,34 @@ | ||||||
|  | import { | ||||||
|  |   Table, | ||||||
|  |   Column, | ||||||
|  |   CreatedAt, | ||||||
|  |   UpdatedAt, | ||||||
|  |   Model, | ||||||
|  |   ForeignKey, | ||||||
|  |   BelongsTo | ||||||
|  | } from "sequelize-typescript"; | ||||||
|  | import Queue from "./Queue"; | ||||||
|  | import Whatsapp from "./Whatsapp"; | ||||||
|  | import QuickAnswer from "./QuickAnswer" | ||||||
|  | 
 | ||||||
|  | @Table | ||||||
|  | class QuickAnswerQueue extends Model<QuickAnswerQueue> { | ||||||
|  |   @ForeignKey(() => QuickAnswer) | ||||||
|  |   @Column | ||||||
|  |   quickAnswerId: number; | ||||||
|  | 
 | ||||||
|  |   @ForeignKey(() => Queue) | ||||||
|  |   @Column | ||||||
|  |   queueId: number; | ||||||
|  | 
 | ||||||
|  |   @CreatedAt | ||||||
|  |   createdAt: Date; | ||||||
|  | 
 | ||||||
|  |   @UpdatedAt | ||||||
|  |   updatedAt: Date; | ||||||
|  | 
 | ||||||
|  |   @BelongsTo(() => Queue) | ||||||
|  |   queue: Queue; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default QuickAnswerQueue; | ||||||
|  | @ -1,35 +1,44 @@ | ||||||
| import { | import { | ||||||
|     Table, |   Table, | ||||||
|     AutoIncrement, |   AutoIncrement, | ||||||
|     Column, |   Column, | ||||||
|     CreatedAt, |   CreatedAt, | ||||||
|     UpdatedAt, |   UpdatedAt, | ||||||
|     Model, |   Model, | ||||||
|     PrimaryKey, |   PrimaryKey, | ||||||
|     HasMany |   HasMany | ||||||
|   } from "sequelize-typescript"; | } from "sequelize-typescript"; | ||||||
| 
 | 
 | ||||||
|   import SchedulingNotify from "./SchedulingNotify"; | import SchedulingNotify from "./SchedulingNotify"; | ||||||
|  | import Ticket from "./Ticket"; | ||||||
| 
 | 
 | ||||||
|   @Table | @Table | ||||||
|   class StatusChatEnd extends Model<StatusChatEnd> { | class StatusChatEnd extends Model<StatusChatEnd> { | ||||||
|     @PrimaryKey |   @PrimaryKey | ||||||
|     @AutoIncrement |   @AutoIncrement | ||||||
|     @Column |   @Column | ||||||
|     id: number;   |   id: number; | ||||||
|    |  | ||||||
|     @Column |  | ||||||
|     name: string; |  | ||||||
|    |  | ||||||
|     @CreatedAt |  | ||||||
|     createdAt: Date; |  | ||||||
|    |  | ||||||
|     @UpdatedAt |  | ||||||
|     updatedAt: Date; |  | ||||||
| 
 | 
 | ||||||
|     @HasMany(() => SchedulingNotify) |   @Column | ||||||
|     SchedulingNotifies: SchedulingNotify[]; |   name: string; | ||||||
|   } | 
 | ||||||
|    |   @Column | ||||||
|   export default StatusChatEnd; |   farewellMessage: string; | ||||||
|    | 
 | ||||||
|  |   @Column | ||||||
|  |   isDefault: boolean; | ||||||
|  | 
 | ||||||
|  |   @CreatedAt | ||||||
|  |   createdAt: Date; | ||||||
|  | 
 | ||||||
|  |   @UpdatedAt | ||||||
|  |   updatedAt: Date; | ||||||
|  | 
 | ||||||
|  |   @HasMany(() => SchedulingNotify) | ||||||
|  |   SchedulingNotifies: SchedulingNotify[]; | ||||||
|  | 
 | ||||||
|  |   @HasMany(() => Ticket) | ||||||
|  |   tickets: Ticket[]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default StatusChatEnd; | ||||||
|  |  | ||||||
|  | @ -21,6 +21,7 @@ import User from "./User"; | ||||||
| import Whatsapp from "./Whatsapp"; | import Whatsapp from "./Whatsapp"; | ||||||
| 
 | 
 | ||||||
| import SchedulingNotify from "./SchedulingNotify"; | import SchedulingNotify from "./SchedulingNotify"; | ||||||
|  | import StatusChatEnd from "./StatusChatEnd" | ||||||
| 
 | 
 | ||||||
| @Table | @Table | ||||||
| class Ticket extends Model<Ticket> { | class Ticket extends Model<Ticket> { | ||||||
|  | @ -42,6 +43,14 @@ class Ticket extends Model<Ticket> { | ||||||
|   @Column |   @Column | ||||||
|   isGroup: boolean; |   isGroup: boolean; | ||||||
| 
 | 
 | ||||||
|  |   @Default(false) | ||||||
|  |   @Column | ||||||
|  |   isRemote: boolean; | ||||||
|  | 
 | ||||||
|  |   @ForeignKey(() => StatusChatEnd) | ||||||
|  |   @Column | ||||||
|  |   statusChatEndId: number; | ||||||
|  | 
 | ||||||
|   @Column |   @Column | ||||||
|   statusChatEnd: string; |   statusChatEnd: string; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -11,13 +11,16 @@ import { | ||||||
|   AutoIncrement, |   AutoIncrement, | ||||||
|   Default, |   Default, | ||||||
|   HasMany, |   HasMany, | ||||||
|   BelongsToMany,  |   BelongsToMany, | ||||||
|  |   BelongsTo, | ||||||
|  |   ForeignKey | ||||||
| } from "sequelize-typescript"; | } from "sequelize-typescript"; | ||||||
| import { hash, compare } from "bcryptjs"; | import { hash, compare } from "bcryptjs"; | ||||||
| import Ticket from "./Ticket"; | import Ticket from "./Ticket"; | ||||||
| import Queue from "./Queue"; | import Queue from "./Queue"; | ||||||
| import UserQueue from "./UserQueue"; | import UserQueue from "./UserQueue"; | ||||||
| import UserOnlineTime from "./UserOnlineTime";  | import UserOnlineTime from "./UserOnlineTime";  | ||||||
|  | import Position from "./Position" | ||||||
| 
 | 
 | ||||||
| @Table | @Table | ||||||
| class User extends Model<User> { | class User extends Model<User> { | ||||||
|  | @ -66,6 +69,13 @@ class User extends Model<User> { | ||||||
|   @BelongsToMany(() => Queue, () => UserQueue) |   @BelongsToMany(() => Queue, () => UserQueue) | ||||||
|   queues: Queue[]; |   queues: Queue[]; | ||||||
| 
 | 
 | ||||||
|  |   @ForeignKey(() => Position) | ||||||
|  |   @Column | ||||||
|  |   positionId: number; | ||||||
|  | 
 | ||||||
|  |   @BelongsTo(() => Position) | ||||||
|  |   position: Position; | ||||||
|  | 
 | ||||||
|   @BeforeUpdate |   @BeforeUpdate | ||||||
|   @BeforeCreate |   @BeforeCreate | ||||||
|   static hashPassword = async (instance: User): Promise<void> => { |   static hashPassword = async (instance: User): Promise<void> => { | ||||||
|  |  | ||||||
|  | @ -0,0 +1,30 @@ | ||||||
|  | import express from "express"; | ||||||
|  | import isAuth from "../middleware/isAuth"; | ||||||
|  | 
 | ||||||
|  | import * as PositionController from "../controllers/PositionController"; | ||||||
|  | 
 | ||||||
|  | const positionRoutes = express.Router(); | ||||||
|  | 
 | ||||||
|  | positionRoutes.get("/positions", isAuth, PositionController.index); | ||||||
|  | 
 | ||||||
|  | positionRoutes.get( | ||||||
|  |   "/positions/:positionId", | ||||||
|  |   isAuth, | ||||||
|  |   PositionController.show | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | positionRoutes.post("/positions", isAuth, PositionController.store); | ||||||
|  | 
 | ||||||
|  | positionRoutes.put( | ||||||
|  |   "/positions/:positionId", | ||||||
|  |   isAuth, | ||||||
|  |   PositionController.update | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | positionRoutes.delete( | ||||||
|  |   "/positions/:positionId", | ||||||
|  |   isAuth, | ||||||
|  |   PositionController.remove | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | export default positionRoutes; | ||||||
|  | @ -15,7 +15,7 @@ import schedulingNotifiyRoutes from "./SchedulingNotifyRoutes"; | ||||||
| import statusChatEndRoutes from "./statusChatEndRoutes"; | import statusChatEndRoutes from "./statusChatEndRoutes"; | ||||||
| import wbotMonitorRoutes from "./wbotMonitorRoutes"; | import wbotMonitorRoutes from "./wbotMonitorRoutes"; | ||||||
| import iamRoutesEL from "./iamRoutesEL"; | import iamRoutesEL from "./iamRoutesEL"; | ||||||
| 
 | import positionRoutes from "./PositionRoutes" | ||||||
| const routes = Router();     | const routes = Router();     | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -35,5 +35,6 @@ routes.use(schedulingNotifiyRoutes); | ||||||
| routes.use(reportRoutes); | routes.use(reportRoutes); | ||||||
| routes.use(statusChatEndRoutes); | routes.use(statusChatEndRoutes); | ||||||
| routes.use(wbotMonitorRoutes); | routes.use(wbotMonitorRoutes); | ||||||
|  | routes.use(positionRoutes); | ||||||
| 
 | 
 | ||||||
| export default routes; | export default routes; | ||||||
|  |  | ||||||
|  | @ -11,6 +11,8 @@ queueRoutes.post("/queue", isAuth, QueueController.store); | ||||||
| 
 | 
 | ||||||
| queueRoutes.post("/queue/customization", QueueController.customization); | queueRoutes.post("/queue/customization", QueueController.customization); | ||||||
| 
 | 
 | ||||||
|  | queueRoutes.get("/queue/remote/list", isAuth, QueueController.listQueues); | ||||||
|  | 
 | ||||||
| queueRoutes.get("/queue/:queueId", isAuth, QueueController.show); | queueRoutes.get("/queue/:queueId", isAuth, QueueController.show); | ||||||
| 
 | 
 | ||||||
| queueRoutes.put("/queue/:queueId", isAuth, QueueController.update); | queueRoutes.put("/queue/:queueId", isAuth, QueueController.update); | ||||||
|  |  | ||||||
|  | @ -1,26 +1,48 @@ | ||||||
| //relatorio
 | //relatorio
 | ||||||
| import express from "express";  | import express from "express"; | ||||||
| 
 | 
 | ||||||
| import isAuth from "../middleware/isAuth";   | import isAuth from "../middleware/isAuth"; | ||||||
|   |  | ||||||
| import * as ReportController from "../controllers/ReportController";  |  | ||||||
| 
 | 
 | ||||||
| const reportRoutes = express.Router();    | import * as ReportController from "../controllers/ReportController"; | ||||||
| 
 | 
 | ||||||
| reportRoutes.get("/reports", isAuth, ReportController.reportUserByDateStartDateEnd);   | const reportRoutes = express.Router(); | ||||||
| 
 | 
 | ||||||
| reportRoutes.post("/reports/onqueue", ReportController.reportOnQueue);   | reportRoutes.get( | ||||||
|  |   "/reports", | ||||||
|  |   isAuth, | ||||||
|  |   ReportController.reportUserByDateStartDateEnd | ||||||
|  | ); | ||||||
| 
 | 
 | ||||||
| reportRoutes.get("/reports/user/services", isAuth, ReportController.reportUserService);   | reportRoutes.post("/reports/onqueue", ReportController.reportOnQueue); | ||||||
|  | 
 | ||||||
|  | reportRoutes.get( | ||||||
|  |   "/reports/user/services", | ||||||
|  |   isAuth, | ||||||
|  |   ReportController.reportUserService | ||||||
|  | ); | ||||||
| 
 | 
 | ||||||
| reportRoutes.get( | reportRoutes.get( | ||||||
|   "/reports/services/numbers", |   "/reports/services/numbers", | ||||||
|   isAuth, |   isAuth, | ||||||
|   ReportController.reportService |   ReportController.reportService | ||||||
| );    | ); | ||||||
| 
 | 
 | ||||||
| reportRoutes.get("/reports/services/queues", isAuth, ReportController.reportServiceByQueue);    | reportRoutes.get( | ||||||
|  |   "/reports/services/queues", | ||||||
|  |   isAuth, | ||||||
|  |   ReportController.reportServiceByQueue | ||||||
|  | ); | ||||||
| 
 | 
 | ||||||
| reportRoutes.get("/reports/messages", isAuth, ReportController.reportMessagesUserByDateStartDateEnd);   | reportRoutes.get( | ||||||
|  |   "/reports/messages", | ||||||
|  |   isAuth, | ||||||
|  |   ReportController.reportMessagesUserByDateStartDateEnd | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | reportRoutes.get( | ||||||
|  |   "/reports/count/statusChatEnd", | ||||||
|  |   isAuth, | ||||||
|  |   ReportController.reportTicksCountByStatusChatEnds | ||||||
|  | ); | ||||||
| 
 | 
 | ||||||
| export default reportRoutes; | export default reportRoutes; | ||||||
|  |  | ||||||
|  | @ -3,8 +3,29 @@ import isAuth from "../middleware/isAuth"; | ||||||
| 
 | 
 | ||||||
| import * as StatusChatEnd from "../controllers/StatusChatEndController"; | import * as StatusChatEnd from "../controllers/StatusChatEndController"; | ||||||
| 
 | 
 | ||||||
| const statusChatEndRoutes = Router();   | const statusChatEndRoutes = Router(); | ||||||
| 
 | 
 | ||||||
| statusChatEndRoutes.get("/statusChatEnd", isAuth, StatusChatEnd.show); | statusChatEndRoutes.post("/statusChatEnd", isAuth, StatusChatEnd.store); | ||||||
| 
 | 
 | ||||||
| export default statusChatEndRoutes; | // statusChatEndRoutes.get("/statusChatEnd", isAuth, StatusChatEnd.show);
 | ||||||
|  | statusChatEndRoutes.get("/statusChatEnd", isAuth, StatusChatEnd.index); | ||||||
|  | 
 | ||||||
|  | statusChatEndRoutes.get( | ||||||
|  |   "/statusChatEnd/:statusChatEndId", | ||||||
|  |   isAuth, | ||||||
|  |   StatusChatEnd.show | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | statusChatEndRoutes.put( | ||||||
|  |   "/statusChatEnd/:statusChatEndId", | ||||||
|  |   isAuth, | ||||||
|  |   StatusChatEnd.update | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | statusChatEndRoutes.delete( | ||||||
|  |   "/statusChatEnd/:statusChatEndId", | ||||||
|  |   isAuth, | ||||||
|  |   StatusChatEnd.remove | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | export default statusChatEndRoutes; | ||||||
|  |  | ||||||
|  | @ -28,6 +28,7 @@ import ShowUserService from "./services/UserServices/ShowUserService"; | ||||||
| import { json } from "sequelize"; | import { json } from "sequelize"; | ||||||
| import { setBotInfo } from "./helpers/SetBotInfo"; | import { setBotInfo } from "./helpers/SetBotInfo"; | ||||||
| import Queue from "./models/Queue"; | import Queue from "./models/Queue"; | ||||||
|  | import StatusChatEnd from "./models/StatusChatEnd"; | ||||||
| 
 | 
 | ||||||
| const server = app.listen(process.env.PORT, () => { | const server = app.listen(process.env.PORT, () => { | ||||||
|   logger.info(`Server started on port: ${process.env.PORT}`); |   logger.info(`Server started on port: ${process.env.PORT}`); | ||||||
|  | @ -48,7 +49,14 @@ gracefulShutdown(server); | ||||||
| (async () => { | (async () => { | ||||||
|   console.log("os.tmpdir(): ", os.tmpdir()); |   console.log("os.tmpdir(): ", os.tmpdir()); | ||||||
| 
 | 
 | ||||||
|   await clearAllKeys("user:*", "whatsapp:*", "queue:*"); |   await clearAllKeys("user:*", "whatsapp:*", "queue:*", "statusChatEnd:*"); | ||||||
|  | 
 | ||||||
|  |   const statusChatEnds = await StatusChatEnd.findAll(); | ||||||
|  | 
 | ||||||
|  |   for (const statusChatEnd of statusChatEnds) { | ||||||
|  |     const { id, name, farewellMessage } = statusChatEnd; | ||||||
|  |     await set(`statusChatEnd:${id}`, { id, name, farewellMessage }); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   const users = await User.findAll(); |   const users = await User.findAll(); | ||||||
| 
 | 
 | ||||||
|  | @ -63,12 +71,12 @@ gracefulShutdown(server); | ||||||
|     await set(`user:${id}`, { id, name }); |     await set(`user:${id}`, { id, name }); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // const queues = await Queue.findAll();
 |   const queues = await Queue.findAll(); | ||||||
| 
 | 
 | ||||||
|   // for (const queue of queues) {
 |   for (const queue of queues) { | ||||||
|   //   const { id, greetingMessage, name } = queue;
 |     const { id, greetingMessage, name, farewellMessage } = queue; | ||||||
|   //   await set(`queue:${id}`, { id, name, greetingMessage });
 |     await set(`queue:${id}`, { id, name, greetingMessage, farewellMessage }); | ||||||
|   // }
 |   } | ||||||
| 
 | 
 | ||||||
|   loadSettings(); |   loadSettings(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | import Contact from "../../models/Contact"; | ||||||
|  | 
 | ||||||
|  | const AssociateContatctQueue = async ( | ||||||
|  |   contact: Contact, | ||||||
|  |   queueIds: number[] | ||||||
|  | ): Promise<void> => { | ||||||
|  |   await contact.$set("queues", queueIds); | ||||||
|  | 
 | ||||||
|  |   await contact.reload(); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default AssociateContatctQueue; | ||||||
|  | @ -1,8 +1,9 @@ | ||||||
| import AppError from "../../errors/AppError"; | import AppError from "../../errors/AppError"; | ||||||
| import Contact from "../../models/Contact"; | import Contact from "../../models/Contact"; | ||||||
| 
 | 
 | ||||||
| import { createOrUpdateContactCache } from '../../helpers/ContactsCache' | import { createOrUpdateContactCache } from "../../helpers/ContactsCache"; | ||||||
| import GetProfilePicUrl from "../WbotServices/GetProfilePicUrl"; | import GetProfilePicUrl from "../WbotServices/GetProfilePicUrl"; | ||||||
|  | import AssociateContatctQueue from "./AssociateContatctQueue"; | ||||||
| 
 | 
 | ||||||
| interface ExtraInfo { | interface ExtraInfo { | ||||||
|   name: string; |   name: string; | ||||||
|  | @ -15,26 +16,26 @@ interface Request { | ||||||
|   email?: string; |   email?: string; | ||||||
|   profilePicUrl?: string; |   profilePicUrl?: string; | ||||||
|   extraInfo?: ExtraInfo[]; |   extraInfo?: ExtraInfo[]; | ||||||
|  |   queueIds?: number[]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const CreateContactService = async ({ | const CreateContactService = async ({ | ||||||
|   name, |   name, | ||||||
|   number, |   number, | ||||||
|   email = "", |   email = "", | ||||||
|   profilePicUrl='', |   profilePicUrl = "", | ||||||
|   extraInfo = [] |   extraInfo = [], | ||||||
| }: Request): Promise<Contact> => {  |   queueIds = [] | ||||||
| 
 | }: Request): Promise<Contact> => { | ||||||
|   try { |   try { | ||||||
| 
 |  | ||||||
|     const numberExists = await Contact.findOne({ |     const numberExists = await Contact.findOne({ | ||||||
|       where: { number } |       where: { number } | ||||||
|     }); |     }); | ||||||
|    | 
 | ||||||
|     if (numberExists) { |     if (numberExists) { | ||||||
|       throw new AppError("ERR_DUPLICATED_CONTACT"); |       throw new AppError("ERR_DUPLICATED_CONTACT"); | ||||||
|     } |     } | ||||||
|    | 
 | ||||||
|     const contact = await Contact.create( |     const contact = await Contact.create( | ||||||
|       { |       { | ||||||
|         name, |         name, | ||||||
|  | @ -47,22 +48,26 @@ const CreateContactService = async ({ | ||||||
|         include: ["extraInfo"] |         include: ["extraInfo"] | ||||||
|       } |       } | ||||||
|     ); |     ); | ||||||
|    | 
 | ||||||
|       |     await AssociateContatctQueue(contact, queueIds); | ||||||
|    | 
 | ||||||
|     // TEST DEL
 |     // TEST DEL
 | ||||||
|     await createOrUpdateContactCache(`contact:${contact.id}`, {id: contact.id, name, number, profilePicUrl, isGroup:'false', extraInfo, email }) |     await createOrUpdateContactCache(`contact:${contact.id}`, { | ||||||
|  |       id: contact.id, | ||||||
|  |       name, | ||||||
|  |       number, | ||||||
|  |       profilePicUrl, | ||||||
|  |       isGroup: "false", | ||||||
|  |       extraInfo, | ||||||
|  |       email | ||||||
|  |     }); | ||||||
|     //
 |     //
 | ||||||
|    | 
 | ||||||
|    |  | ||||||
|     return contact; |     return contact; | ||||||
|      |  | ||||||
|   } catch (error: any) { |   } catch (error: any) { | ||||||
|     console.error('===> Error on CreateContactService.ts file: \n', error) |     console.error("===> Error on CreateContactService.ts file: \n", error); | ||||||
|     throw new AppError(error.message); |     throw new AppError(error.message); | ||||||
|   } |   } | ||||||
|   |  | ||||||
|    |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export default CreateContactService; | export default CreateContactService; | ||||||
|  |  | ||||||
|  | @ -1,9 +1,16 @@ | ||||||
| import { Sequelize, Op } from "sequelize"; | import { Sequelize, Op } from "sequelize"; | ||||||
| import Contact from "../../models/Contact"; | import Contact from "../../models/Contact"; | ||||||
|  | import Queue from "../../models/Queue"; | ||||||
|  | import { getSettingValue } from "../../helpers/WhaticketSettings"; | ||||||
|  | // import ListContactsService from "../services/ContactServices/ListContactsService";
 | ||||||
|  | import ListWhatsappQueueByUserQueue from "../WhatsappService/ListWhatsAppsForQueueService"; | ||||||
|  | import QueuesByUser from "../UserServices/ShowQueuesByUser"; | ||||||
|  | import { number } from "yup"; | ||||||
| 
 | 
 | ||||||
| interface Request { | interface Request { | ||||||
|   searchParam?: string; |   searchParam?: string; | ||||||
|   pageNumber?: string; |   pageNumber?: string; | ||||||
|  |   userId?: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| interface Response { | interface Response { | ||||||
|  | @ -14,13 +21,14 @@ interface Response { | ||||||
| 
 | 
 | ||||||
| const ListContactsService = async ({ | const ListContactsService = async ({ | ||||||
|   searchParam = "", |   searchParam = "", | ||||||
|   pageNumber = "1" |   pageNumber = "1", | ||||||
|  |   userId | ||||||
| }: Request): Promise<Response> => { | }: Request): Promise<Response> => { | ||||||
|   const whereCondition = { |   const whereCondition = { | ||||||
|     [Op.or]: [ |     [Op.or]: [ | ||||||
|       { |       { | ||||||
|         name: Sequelize.where( |         name: Sequelize.where( | ||||||
|           Sequelize.fn("LOWER", Sequelize.col("name")), |           Sequelize.fn("LOWER", Sequelize.col("Contact.name")), | ||||||
|           "LIKE", |           "LIKE", | ||||||
|           `%${searchParam.toLowerCase().trim()}%` |           `%${searchParam.toLowerCase().trim()}%` | ||||||
|         ) |         ) | ||||||
|  | @ -29,22 +37,61 @@ const ListContactsService = async ({ | ||||||
|     ] |     ] | ||||||
|   }; |   }; | ||||||
|   const limit = 20; |   const limit = 20; | ||||||
|   const offset = limit * (+pageNumber - 1);  |   const offset = limit * (+pageNumber - 1); | ||||||
| 
 | 
 | ||||||
|   const { count, rows: contacts } = await Contact.findAndCountAll({ |   let { count, rows: contacts } = await Contact.findAndCountAll({ | ||||||
|     where: whereCondition, |     where: whereCondition, | ||||||
|     limit, |     limit, | ||||||
|  |     include: [ | ||||||
|  |       { | ||||||
|  |         model: Queue, | ||||||
|  |         as: "queues", | ||||||
|  |         // where: whereConditionQueue,
 | ||||||
|  |         attributes: ["id", "name", "color", "greetingMessage"] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|     offset, |     offset, | ||||||
|     order: [["name", "ASC"]] |     order: [["name", "ASC"]] | ||||||
|   });  |   }); | ||||||
| 
 | 
 | ||||||
|   const hasMore = count > offset + contacts.length; |   const hasMore = count > offset + contacts.length; | ||||||
| 
 | 
 | ||||||
|  |   if (getSettingValue("contactByqueues")?.value == "enabled") { | ||||||
|  |     const queueIds = await QueuesByUser({ userId }); | ||||||
|  | 
 | ||||||
|  |     contacts = contactQueueFilter(queueIds, contacts); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   return { |   return { | ||||||
|     contacts, |     contacts, | ||||||
|     count, |     count, | ||||||
|     hasMore |     hasMore | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
| export default ListContactsService; | export default ListContactsService; | ||||||
|  | 
 | ||||||
|  | function contactQueueFilter(queueIds: any[], contacts: Contact[]) { | ||||||
|  |   let auxContact: any[] = []; | ||||||
|  |   let repet: any[] = []; | ||||||
|  |   const userQueues = queueIds.map((uq: any) => uq.queueId); | ||||||
|  | 
 | ||||||
|  |   for (const contact of contacts) { | ||||||
|  |     const { queues, id } = contact; | ||||||
|  | 
 | ||||||
|  |     if (queues.length == 0) { | ||||||
|  |       auxContact.push(contact); | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (const q of queues) { | ||||||
|  |       if (userQueues.includes(q.id)) { | ||||||
|  |         if (repet.includes(id)) continue; | ||||||
|  |         repet.push(id); | ||||||
|  | 
 | ||||||
|  |         auxContact.push(contact); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return auxContact; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -1,8 +1,20 @@ | ||||||
| import Contact from "../../models/Contact"; | import Contact from "../../models/Contact"; | ||||||
| import AppError from "../../errors/AppError"; | import AppError from "../../errors/AppError"; | ||||||
|  | import { getSettingValue } from "../../helpers/WhaticketSettings"; | ||||||
|  | import Queue from "../../models/Queue"; | ||||||
| 
 | 
 | ||||||
| const ShowContactService = async (id: string | number): Promise<Contact> => { | const ShowContactService = async (id: string | number): Promise<Contact> => { | ||||||
|   const contact = await Contact.findByPk(id, { include: ["extraInfo"] }); |   let includeQueue: any[] = []; | ||||||
|  | 
 | ||||||
|  |   if (getSettingValue("contactByqueues")?.value == "enabled") { | ||||||
|  |     includeQueue = [ | ||||||
|  |       { model: Queue, as: "queues", attributes: ["id", "name", "color"] } | ||||||
|  |     ]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const contact = await Contact.findByPk(id, { | ||||||
|  |     include: ["extraInfo", ...includeQueue] | ||||||
|  |   }); | ||||||
| 
 | 
 | ||||||
|   if (!contact) { |   if (!contact) { | ||||||
|     throw new AppError("ERR_NO_CONTACT_FOUND", 404); |     throw new AppError("ERR_NO_CONTACT_FOUND", 404); | ||||||
|  |  | ||||||
|  | @ -2,9 +2,10 @@ import AppError from "../../errors/AppError"; | ||||||
| import Contact from "../../models/Contact"; | import Contact from "../../models/Contact"; | ||||||
| import ContactCustomField from "../../models/ContactCustomField"; | import ContactCustomField from "../../models/ContactCustomField"; | ||||||
| 
 | 
 | ||||||
| import { updateTicketsByContactsCache } from '../../helpers/TicketCache' | import { updateTicketsByContactsCache } from "../../helpers/TicketCache"; | ||||||
| import { updateContactCacheById } from '../../helpers/ContactsCache' | import { updateContactCacheById } from "../../helpers/ContactsCache"; | ||||||
| import { tr } from "date-fns/locale"; | import { tr } from "date-fns/locale"; | ||||||
|  | import AssociateContatctQueue from "./AssociateContatctQueue"; | ||||||
| 
 | 
 | ||||||
| interface ExtraInfo { | interface ExtraInfo { | ||||||
|   id?: number; |   id?: number; | ||||||
|  | @ -16,6 +17,7 @@ interface ContactData { | ||||||
|   number?: string; |   number?: string; | ||||||
|   name?: string; |   name?: string; | ||||||
|   extraInfo?: ExtraInfo[]; |   extraInfo?: ExtraInfo[]; | ||||||
|  |   queueIds?: number[]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| interface Request { | interface Request { | ||||||
|  | @ -23,15 +25,12 @@ interface Request { | ||||||
|   contactId: string; |   contactId: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| const UpdateContactService = async ({ | const UpdateContactService = async ({ | ||||||
|   contactData, |   contactData, | ||||||
|   contactId |   contactId | ||||||
| }: Request): Promise<Contact> => { | }: Request): Promise<Contact> => { | ||||||
| 
 |  | ||||||
|   try { |   try { | ||||||
| 
 |     const { email, name, number, extraInfo, queueIds } = contactData; | ||||||
|     const { email, name, number, extraInfo } = contactData; |  | ||||||
| 
 | 
 | ||||||
|     // console.log('email, name, number, extraInfo: ', email, name, number, extraInfo)
 |     // console.log('email, name, number, extraInfo: ', email, name, number, extraInfo)
 | ||||||
| 
 | 
 | ||||||
|  | @ -54,7 +53,9 @@ const UpdateContactService = async ({ | ||||||
| 
 | 
 | ||||||
|       await Promise.all( |       await Promise.all( | ||||||
|         contact.extraInfo.map(async oldInfo => { |         contact.extraInfo.map(async oldInfo => { | ||||||
|           const stillExists = extraInfo.findIndex(info => info.id === oldInfo.id); |           const stillExists = extraInfo.findIndex( | ||||||
|  |             info => info.id === oldInfo.id | ||||||
|  |           ); | ||||||
| 
 | 
 | ||||||
|           if (stillExists === -1) { |           if (stillExists === -1) { | ||||||
|             await ContactCustomField.destroy({ where: { id: oldInfo.id } }); |             await ContactCustomField.destroy({ where: { id: oldInfo.id } }); | ||||||
|  | @ -63,11 +64,10 @@ const UpdateContactService = async ({ | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const oldNumber = contact.number |     const oldNumber = contact.number; | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
|     //Solução para o erro tcp_wrap.cc
 |     //Solução para o erro tcp_wrap.cc
 | ||||||
|     console.log('----------> oldNumber: ', oldNumber) |     console.log("----------> oldNumber: ", oldNumber); | ||||||
|     if (number) { |     if (number) { | ||||||
|       const numberExists = await Contact.findOne({ |       const numberExists = await Contact.findOne({ | ||||||
|         where: { number } |         where: { number } | ||||||
|  | @ -78,37 +78,35 @@ const UpdateContactService = async ({ | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     await contact.update({ |     await contact.update({ | ||||||
|       name, |       name, | ||||||
|       number, |       number, | ||||||
|       email |       email | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
| 
 |      | ||||||
|  |     if (queueIds) await AssociateContatctQueue(contact, queueIds); | ||||||
| 
 | 
 | ||||||
|     //TEST DEL
 |     //TEST DEL
 | ||||||
|     await updateTicketsByContactsCache(oldNumber, contact.name, contact.number) |     await updateTicketsByContactsCache(oldNumber, contact.name, contact.number); | ||||||
|     //
 |     //
 | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     await contact.reload({ |     await contact.reload({ | ||||||
|       attributes: ["id", "name", "number", "email", "profilePicUrl"], |       attributes: ["id", "name", "number", "email", "profilePicUrl"], | ||||||
|       include: ["extraInfo"] |       include: ["extraInfo"] | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     // console.log('contactcontactcontactcontact: ',flatten(JSON.parse(JSON.stringify(contact))))
 |     // console.log('contactcontactcontactcontact: ',flatten(JSON.parse(JSON.stringify(contact))))
 | ||||||
|     await updateContactCacheById(contact.id, JSON.parse(JSON.stringify(contact))) |     await updateContactCacheById( | ||||||
|  |       contact.id, | ||||||
|  |       JSON.parse(JSON.stringify(contact)) | ||||||
|  |     ); | ||||||
| 
 | 
 | ||||||
|     return contact; |     return contact; | ||||||
| 
 |  | ||||||
|   } catch (error: any) { |   } catch (error: any) { | ||||||
|     console.error('===> Error on UpdateContactService.ts file: \n', error) |     console.error("===> Error on UpdateContactService.ts file: \n", error); | ||||||
|     throw new AppError(error.message); |     throw new AppError(error.message); | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export default UpdateContactService; | export default UpdateContactService; | ||||||
|  |  | ||||||
|  | @ -0,0 +1,23 @@ | ||||||
|  | import AppError from "../../errors/AppError"; | ||||||
|  | import Position from "../../models/Position"; | ||||||
|  | import AssociateQuickAnswerQueue from "../QueueService/AssociateQuickAnswerQueue"; | ||||||
|  | 
 | ||||||
|  | interface Request { | ||||||
|  |   name: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const CreatePositionService = async ({ name }: Request): Promise<Position> => { | ||||||
|  |   const nameExists = await Position.findOne({ | ||||||
|  |     where: { name } | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   if (nameExists) { | ||||||
|  |     throw new AppError("ERR_POSITION_DUPLICATED"); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const position = await Position.create({ name }); | ||||||
|  | 
 | ||||||
|  |   return position; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default CreatePositionService; | ||||||
|  | @ -0,0 +1,16 @@ | ||||||
|  | import Position from "../../models/Position"; | ||||||
|  | import AppError from "../../errors/AppError"; | ||||||
|  | 
 | ||||||
|  | const DeletePositionService = async (id: string): Promise<void> => { | ||||||
|  |   const position = await Position.findOne({ | ||||||
|  |     where: { id } | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   if (!position) { | ||||||
|  |     throw new AppError("ERR_NO_POSITION_FOUND", 404); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   await position.destroy(); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default DeletePositionService; | ||||||
|  | @ -0,0 +1,48 @@ | ||||||
|  | import { Op, Sequelize } from "sequelize";  | ||||||
|  | import Position from "../../models/Position"; | ||||||
|  | 
 | ||||||
|  | interface Request { | ||||||
|  |   searchParam?: string; | ||||||
|  |   pageNumber?: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | interface Response { | ||||||
|  |   positions: Position[]; | ||||||
|  |   count: number; | ||||||
|  |   hasMore: boolean; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const ListPositionService = async ({ | ||||||
|  |   searchParam = "", | ||||||
|  |   pageNumber = "1" | ||||||
|  | }: Request): Promise<Response> => { | ||||||
|  |   console.log("searchParam: ", searchParam); | ||||||
|  | 
 | ||||||
|  |   const whereCondition = { | ||||||
|  |     message: Sequelize.where( | ||||||
|  |       Sequelize.fn("LOWER", Sequelize.col("name")), | ||||||
|  |       "LIKE", | ||||||
|  |       `%${searchParam.toLowerCase().trim()}%` | ||||||
|  |     ) | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   const limit = 20; | ||||||
|  |   const offset = limit * (+pageNumber - 1); | ||||||
|  | 
 | ||||||
|  |   let { count, rows: positions } = await Position.findAndCountAll({ | ||||||
|  |     where: whereCondition, | ||||||
|  |     limit, | ||||||
|  |     offset, | ||||||
|  |     order: [["name", "ASC"]] | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const hasMore = count > offset + positions.length; | ||||||
|  | 
 | ||||||
|  |   return { | ||||||
|  |     positions, | ||||||
|  |     count, | ||||||
|  |     hasMore | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default ListPositionService; | ||||||
|  | @ -0,0 +1,16 @@ | ||||||
|  | import AppError from "../../errors/AppError"; | ||||||
|  | import Position from "../../models/Position"; | ||||||
|  | 
 | ||||||
|  | const ShowPositionService = async ( | ||||||
|  |   id: string,  | ||||||
|  | ): Promise<Position> => { | ||||||
|  |   const position = await Position.findByPk(id); | ||||||
|  | 
 | ||||||
|  |   if (!position) { | ||||||
|  |     throw new AppError("ERR_NO_POSITION_FOUND", 404); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return position; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default ShowPositionService; | ||||||
|  | @ -0,0 +1,38 @@ | ||||||
|  | import Position from "../../models/Position"; | ||||||
|  | import AppError from "../../errors/AppError"; | ||||||
|  | import AssociateQuickAnswerQueue from "../QueueService/AssociateQuickAnswerQueue"; | ||||||
|  | 
 | ||||||
|  | interface PositionData { | ||||||
|  |   name: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | interface Request { | ||||||
|  |   positionData: PositionData; | ||||||
|  |   positionId: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const UpdatePositionService = async ({ | ||||||
|  |   positionData, | ||||||
|  |   positionId | ||||||
|  | }: Request): Promise<Position> => { | ||||||
|  |   const { name } = positionData; | ||||||
|  | 
 | ||||||
|  |   const position = await Position.findOne({ | ||||||
|  |     where: { id: positionId } | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   if (!position) { | ||||||
|  |     throw new AppError("ERR_NO_POSITION_FOUND", 404); | ||||||
|  |   } | ||||||
|  |   await position.update({ | ||||||
|  |     name | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   await position.reload({ | ||||||
|  |     attributes: ["id", "name"] | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   return position; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default UpdatePositionService; | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | import QuickAnswer from "../../models/QuickAnswer"; | ||||||
|  | 
 | ||||||
|  | const AssociateQuickAnswerQueue = async ( | ||||||
|  |   QuickAnswer: QuickAnswer, | ||||||
|  |   queueIds: number[] | ||||||
|  | ): Promise<void> => { | ||||||
|  |   await QuickAnswer.$set("queues", queueIds); | ||||||
|  | 
 | ||||||
|  |   await QuickAnswer.reload(); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default AssociateQuickAnswerQueue; | ||||||
|  | @ -7,6 +7,8 @@ interface QueueData { | ||||||
|   name: string; |   name: string; | ||||||
|   color: string; |   color: string; | ||||||
|   greetingMessage?: string; |   greetingMessage?: string; | ||||||
|  |   farewellMessage?: string; | ||||||
|  |   cc?: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const CreateQueueService = async (queueData: QueueData): Promise<Queue> => { | const CreateQueueService = async (queueData: QueueData): Promise<Queue> => { | ||||||
|  | @ -63,8 +65,13 @@ const CreateQueueService = async (queueData: QueueData): Promise<Queue> => { | ||||||
| 
 | 
 | ||||||
|     const queue = await Queue.create(queueData); |     const queue = await Queue.create(queueData); | ||||||
| 
 | 
 | ||||||
|     // const { id, greetingMessage } = queue;
 |     const { id, greetingMessage, farewellMessage } = queue; | ||||||
|     // await set(`queue:${id}`, { id, name, greetingMessage });
 |     await set(`queue:${id}`, { | ||||||
|  |       id, | ||||||
|  |       name, | ||||||
|  |       greetingMessage, | ||||||
|  |       farewellMessage | ||||||
|  |     }); | ||||||
| 
 | 
 | ||||||
|     return queue; |     return queue; | ||||||
|   } catch (error: any) { |   } catch (error: any) { | ||||||
|  |  | ||||||
|  | @ -4,28 +4,28 @@ import UserQueue from "../../models/UserQueue"; | ||||||
| 
 | 
 | ||||||
| import ListTicketsServiceCache from "../TicketServices/ListTicketServiceCache"; | import ListTicketsServiceCache from "../TicketServices/ListTicketServiceCache"; | ||||||
| 
 | 
 | ||||||
| import { deleteTicketsFieldsCache } from '../../helpers/TicketCache' | import { deleteTicketsFieldsCache } from "../../helpers/TicketCache"; | ||||||
|  | import { del } from "../../helpers/RedisClient"; | ||||||
| 
 | 
 | ||||||
| const DeleteQueueService = async (queueId: number | string): Promise<void> => { | const DeleteQueueService = async (queueId: number | string): Promise<void> => { | ||||||
| 
 |  | ||||||
|   const queue = await ShowQueueService(queueId); |   const queue = await ShowQueueService(queueId); | ||||||
| 
 | 
 | ||||||
|   if (queue.id) { |   if (queue.id) { | ||||||
|  |     const tickets = await ListTicketsServiceCache({ queueId }); | ||||||
| 
 | 
 | ||||||
|     const tickets = await ListTicketsServiceCache({ queueId }) |     await deleteTicketsFieldsCache(tickets, [ | ||||||
| 
 |       "queue.id", | ||||||
|     await deleteTicketsFieldsCache(tickets, ['queue.id',  'queue.name', 'queue.color']) |       "queue.name", | ||||||
|   |       "queue.color" | ||||||
|  |     ]); | ||||||
|   } |   } | ||||||
|     |  | ||||||
|   try { |  | ||||||
| 
 | 
 | ||||||
|  |   try { | ||||||
|     await UserQueue.destroy({ where: { queueId: queueId } }); |     await UserQueue.destroy({ where: { queueId: queueId } }); | ||||||
| 
 | 
 | ||||||
|  |     del(`queue:${queueId}`); | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
| 
 |     console.log("Error on delete UserQueue by queueId: ", queueId); | ||||||
|     console.log('Error on delete UserQueue by queueId: ', queueId) |  | ||||||
| 
 |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   await queue.destroy(); |   await queue.destroy(); | ||||||
|  |  | ||||||
|  | @ -3,12 +3,14 @@ import * as Yup from "yup"; | ||||||
| import AppError from "../../errors/AppError"; | import AppError from "../../errors/AppError"; | ||||||
| import Queue from "../../models/Queue"; | import Queue from "../../models/Queue"; | ||||||
| import ShowQueueService from "./ShowQueueService"; | import ShowQueueService from "./ShowQueueService"; | ||||||
| import { set } from "../../helpers/RedisClient" | import { set } from "../../helpers/RedisClient"; | ||||||
| 
 | 
 | ||||||
| interface QueueData { | interface QueueData { | ||||||
|   name?: string; |   name?: string; | ||||||
|   color?: string; |   color?: string; | ||||||
|   greetingMessage?: string; |   greetingMessage?: string; | ||||||
|  |   farewellMessage?: string; | ||||||
|  |   cc?: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const UpdateQueueService = async ( | const UpdateQueueService = async ( | ||||||
|  | @ -69,8 +71,13 @@ const UpdateQueueService = async ( | ||||||
| 
 | 
 | ||||||
|     await queue.update(queueData); |     await queue.update(queueData); | ||||||
| 
 | 
 | ||||||
|     // const { id, greetingMessage } = queue;
 |     const { greetingMessage, farewellMessage } = queue; | ||||||
|     // await set(`queue:${id}`, { id, name, greetingMessage });
 |     await set(`queue:${queueId}`, { | ||||||
|  |       id: queueId, | ||||||
|  |       name, | ||||||
|  |       greetingMessage, | ||||||
|  |       farewellMessage | ||||||
|  |     }); | ||||||
| 
 | 
 | ||||||
|     return queue; |     return queue; | ||||||
|   } catch (error: any) { |   } catch (error: any) { | ||||||
|  |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue