mirror of
https://github.com/EvolutionAPI/evolution-api.git
synced 2025-07-24 17:38:40 -06:00
Merge branch 'release/1.2.0'
This commit is contained in:
commit
be54331d05
25
CHANGELOG.md
25
CHANGELOG.md
@ -1,3 +1,28 @@
|
|||||||
|
# 1.2.0 (homolog)
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* Native integration with chatwoot
|
||||||
|
* Added returning or non-returning participants option in fetchAllGroups
|
||||||
|
* Added group integration to chatwoot
|
||||||
|
* Added automation on create instance to chatwoot
|
||||||
|
* Added verbose logs and format chatwoot service
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
* Adjusts in docker-compose files
|
||||||
|
* Adjusts in number validation for AR and MX numbers
|
||||||
|
* Adjusts in env files, removed save old_messages
|
||||||
|
* Fix when sending a message to a group I don't belong returns a bad request
|
||||||
|
* Fits the format on return from the fetchAllGroups endpoint
|
||||||
|
* Adjust in send document with caption from chatwoot
|
||||||
|
* Fixed message with undefind in chatwoot
|
||||||
|
* Changed message in path /
|
||||||
|
* Test duplicate message media in groups chatwoot
|
||||||
|
* Optimize send message from group with mentions
|
||||||
|
* Fixed name of the profile status in fetchInstances
|
||||||
|
* Fixed error 500 when logout in instance with status = close
|
||||||
|
|
||||||
# 1.1.5 (2023-07-12 07:17)
|
# 1.1.5 (2023-07-12 07:17)
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
SERVER_URL='<url>' # ex.: http://localhost:3333
|
||||||
|
|
||||||
CORS_ORIGIN='*' # Or separate by commas - ex.: 'yourdomain1.com, yourdomain2.com'
|
CORS_ORIGIN='*' # Or separate by commas - ex.: 'yourdomain1.com, yourdomain2.com'
|
||||||
CORS_METHODS='POST,GET,PUT,DELETE'
|
CORS_METHODS='POST,GET,PUT,DELETE'
|
||||||
CORS_CREDENTIALS=true
|
CORS_CREDENTIALS=true
|
||||||
@ -31,7 +33,6 @@ DATABASE_CONNECTION_DB_PREFIX_NAME=evolution
|
|||||||
|
|
||||||
# Choose the data you want to save in the application's database or store
|
# Choose the data you want to save in the application's database or store
|
||||||
DATABASE_SAVE_DATA_INSTANCE=false
|
DATABASE_SAVE_DATA_INSTANCE=false
|
||||||
DATABASE_SAVE_DATA_OLD_MESSAGE=false
|
|
||||||
DATABASE_SAVE_DATA_NEW_MESSAGE=false
|
DATABASE_SAVE_DATA_NEW_MESSAGE=false
|
||||||
DATABASE_SAVE_MESSAGE_UPDATE=false
|
DATABASE_SAVE_MESSAGE_UPDATE=false
|
||||||
DATABASE_SAVE_DATA_CONTACTS=false
|
DATABASE_SAVE_DATA_CONTACTS=false
|
||||||
@ -41,7 +42,8 @@ REDIS_ENABLED=false
|
|||||||
REDIS_URI=redis://redis:6379
|
REDIS_URI=redis://redis:6379
|
||||||
REDIS_PREFIX_KEY=evolution
|
REDIS_PREFIX_KEY=evolution
|
||||||
|
|
||||||
# Webhook Settings
|
# Global Webhook Settings
|
||||||
|
# Each instance's Webhook URL and events will be requested at the time it is created
|
||||||
## Define a global webhook that will listen for enabled events from all instances
|
## Define a global webhook that will listen for enabled events from all instances
|
||||||
WEBHOOK_GLOBAL_URL='<url>'
|
WEBHOOK_GLOBAL_URL='<url>'
|
||||||
WEBHOOK_GLOBAL_ENABLED=false
|
WEBHOOK_GLOBAL_ENABLED=false
|
||||||
@ -53,6 +55,7 @@ WEBHOOK_EVENTS_QRCODE_UPDATED=true
|
|||||||
WEBHOOK_EVENTS_MESSAGES_SET=true
|
WEBHOOK_EVENTS_MESSAGES_SET=true
|
||||||
WEBHOOK_EVENTS_MESSAGES_UPSERT=true
|
WEBHOOK_EVENTS_MESSAGES_UPSERT=true
|
||||||
WEBHOOK_EVENTS_MESSAGES_UPDATE=true
|
WEBHOOK_EVENTS_MESSAGES_UPDATE=true
|
||||||
|
WEBHOOK_EVENTS_SEND_MESSAGE=true
|
||||||
WEBHOOK_EVENTS_CONTACTS_SET=true
|
WEBHOOK_EVENTS_CONTACTS_SET=true
|
||||||
WEBHOOK_EVENTS_CONTACTS_UPSERT=true
|
WEBHOOK_EVENTS_CONTACTS_UPSERT=true
|
||||||
WEBHOOK_EVENTS_CONTACTS_UPDATE=true
|
WEBHOOK_EVENTS_CONTACTS_UPDATE=true
|
||||||
@ -76,6 +79,8 @@ CONFIG_SESSION_PHONE_NAME=chrome # chrome | firefox | edge | opera | safari
|
|||||||
QRCODE_LIMIT=30
|
QRCODE_LIMIT=30
|
||||||
|
|
||||||
# Defines an authentication type for the api
|
# Defines an authentication type for the api
|
||||||
|
# We recommend using the apikey because it will allow you to use a custom token,
|
||||||
|
# if you use jwt, a random token will be generated and may be expired and you will have to generate a new token
|
||||||
AUTHENTICATION_TYPE='apikey' # jwt or 'apikey'
|
AUTHENTICATION_TYPE='apikey' # jwt or 'apikey'
|
||||||
## Define a global apikey to access all instances.
|
## Define a global apikey to access all instances.
|
||||||
### OBS: This key must be inserted in the request header to create an instance.
|
### OBS: This key must be inserted in the request header to create an instance.
|
||||||
@ -89,4 +94,7 @@ AUTHENTICATION_JWT_SECRET='L0YWtjb2w554WFqPG'
|
|||||||
AUTHENTICATION_INSTANCE_MODE=server # container or server
|
AUTHENTICATION_INSTANCE_MODE=server # container or server
|
||||||
# if you are using container mode, set the container name and the webhook url to default instance
|
# if you are using container mode, set the container name and the webhook url to default instance
|
||||||
AUTHENTICATION_INSTANCE_NAME=evolution
|
AUTHENTICATION_INSTANCE_NAME=evolution
|
||||||
AUTHENTICATION_INSTANCE_WEBHOOK_URL='<url>'
|
AUTHENTICATION_INSTANCE_WEBHOOK_URL='<url>'
|
||||||
|
AUTHENTICATION_INSTANCE_CHATWOOT_ACCOUNT_ID=1
|
||||||
|
AUTHENTICATION_INSTANCE_CHATWOOT_TOKEN=123456
|
||||||
|
AUTHENTICATION_INSTANCE_CHATWOOT_URL='<url>'
|
@ -9,14 +9,16 @@ services:
|
|||||||
container_name: mongodb
|
container_name: mongodb
|
||||||
image: mongo
|
image: mongo
|
||||||
restart: always
|
restart: always
|
||||||
volumes:
|
|
||||||
- evolution_mongodb_data:/data/db
|
|
||||||
- evolution_mongodb_configdb:/data/configdb
|
|
||||||
ports:
|
ports:
|
||||||
- 27017:27017
|
- 27017:27017
|
||||||
environment:
|
environment:
|
||||||
MONGO_INITDB_ROOT_USERNAME: root
|
- MONGO_INITDB_ROOT_USERNAME=root
|
||||||
MONGO_INITDB_ROOT_PASSWORD: root
|
- MONGO_INITDB_ROOT_PASSWORD=root
|
||||||
|
- PUID=1000
|
||||||
|
- PGID=1000
|
||||||
|
volumes:
|
||||||
|
- evolution_mongodb_data:/data/db
|
||||||
|
- evolution_mongodb_configdb:/data/configdb
|
||||||
networks:
|
networks:
|
||||||
- evolution-net
|
- evolution-net
|
||||||
expose:
|
expose:
|
||||||
|
@ -8,6 +8,12 @@ services:
|
|||||||
redis:
|
redis:
|
||||||
image: redis:latest
|
image: redis:latest
|
||||||
container_name: redis
|
container_name: redis
|
||||||
|
command: >
|
||||||
|
redis-server
|
||||||
|
--port 6379
|
||||||
|
--appendonly yes
|
||||||
|
volumes:
|
||||||
|
- evolution_redis:/data
|
||||||
ports:
|
ports:
|
||||||
- 6379:6379
|
- 6379:6379
|
||||||
networks:
|
networks:
|
||||||
|
@ -15,6 +15,7 @@ ENV DOCKER_ENV=true
|
|||||||
|
|
||||||
ENV SERVER_TYPE="http"
|
ENV SERVER_TYPE="http"
|
||||||
ENV SERVER_PORT=8080
|
ENV SERVER_PORT=8080
|
||||||
|
ENV SERVER_URL=$SERVER_URL
|
||||||
|
|
||||||
ENV CORS_ORIGIN="*"
|
ENV CORS_ORIGIN="*"
|
||||||
ENV CORS_METHODS="POST,GET,PUT,DELETE"
|
ENV CORS_METHODS="POST,GET,PUT,DELETE"
|
||||||
@ -40,7 +41,6 @@ ENV DATABASE_ENABLED=$DATABASE_ENABLED
|
|||||||
ENV DATABASE_CONNECTION_URI=$DATABASE_CONNECTION_URI
|
ENV DATABASE_CONNECTION_URI=$DATABASE_CONNECTION_URI
|
||||||
ENV DATABASE_CONNECTION_DB_PREFIX_NAME=$DATABASE_CONNECTION_DB_PREFIX_NAME
|
ENV DATABASE_CONNECTION_DB_PREFIX_NAME=$DATABASE_CONNECTION_DB_PREFIX_NAME
|
||||||
ENV DATABASE_SAVE_DATA_INSTANCE=$DATABASE_SAVE_DATA_INSTANCE
|
ENV DATABASE_SAVE_DATA_INSTANCE=$DATABASE_SAVE_DATA_INSTANCE
|
||||||
ENV DATABASE_SAVE_DATA_OLD_MESSAGE=$DATABASE_SAVE_DATA_OLD_MESSAGE
|
|
||||||
ENV DATABASE_SAVE_DATA_NEW_MESSAGE=$DATABASE_SAVE_DATA_NEW_MESSAGE
|
ENV DATABASE_SAVE_DATA_NEW_MESSAGE=$DATABASE_SAVE_DATA_NEW_MESSAGE
|
||||||
ENV DATABASE_SAVE_MESSAGE_UPDATE=$DATABASE_SAVE_MESSAGE_UPDATE
|
ENV DATABASE_SAVE_MESSAGE_UPDATE=$DATABASE_SAVE_MESSAGE_UPDATE
|
||||||
ENV DATABASE_SAVE_DATA_CONTACTS=$DATABASE_SAVE_DATA_CONTACTS
|
ENV DATABASE_SAVE_DATA_CONTACTS=$DATABASE_SAVE_DATA_CONTACTS
|
||||||
@ -57,8 +57,9 @@ ENV WEBHOOK_GLOBAL_WEBHOOK_BY_EVENTS=$WEBHOOK_GLOBAL_WEBHOOK_BY_EVENTS
|
|||||||
ENV WEBHOOK_EVENTS_APPLICATION_STARTUP=$WEBHOOK_EVENTS_APPLICATION_STARTUP
|
ENV WEBHOOK_EVENTS_APPLICATION_STARTUP=$WEBHOOK_EVENTS_APPLICATION_STARTUP
|
||||||
ENV WEBHOOK_EVENTS_QRCODE_UPDATED=$WEBHOOK_EVENTS_QRCODE_UPDATED
|
ENV WEBHOOK_EVENTS_QRCODE_UPDATED=$WEBHOOK_EVENTS_QRCODE_UPDATED
|
||||||
ENV WEBHOOK_EVENTS_MESSAGES_SET=$WEBHOOK_EVENTS_MESSAGES_SET
|
ENV WEBHOOK_EVENTS_MESSAGES_SET=$WEBHOOK_EVENTS_MESSAGES_SET
|
||||||
ENV WEBHOOK_EVENTS_MESSAGES_UPDATE=$WEBHOOK_EVENTS_MESSAGES_UPDATE
|
|
||||||
ENV WEBHOOK_EVENTS_MESSAGES_UPSERT=$WEBHOOK_EVENTS_MESSAGES_UPSERT
|
ENV WEBHOOK_EVENTS_MESSAGES_UPSERT=$WEBHOOK_EVENTS_MESSAGES_UPSERT
|
||||||
|
ENV WEBHOOK_EVENTS_MESSAGES_UPDATE=$WEBHOOK_EVENTS_MESSAGES_UPDATE
|
||||||
|
ENV WEBHOOK_EVENTS_SEND_MESSAGE=$WEBHOOK_EVENTS_SEND_MESSAGE
|
||||||
ENV WEBHOOK_EVENTS_CONTACTS_SET=$WEBHOOK_EVENTS_CONTACTS_SET
|
ENV WEBHOOK_EVENTS_CONTACTS_SET=$WEBHOOK_EVENTS_CONTACTS_SET
|
||||||
ENV WEBHOOK_EVENTS_CONTACTS_UPSERT=$WEBHOOK_EVENTS_CONTACTS_UPSERT
|
ENV WEBHOOK_EVENTS_CONTACTS_UPSERT=$WEBHOOK_EVENTS_CONTACTS_UPSERT
|
||||||
ENV WEBHOOK_EVENTS_CONTACTS_UPDATE=$WEBHOOK_EVENTS_CONTACTS_UPDATE
|
ENV WEBHOOK_EVENTS_CONTACTS_UPDATE=$WEBHOOK_EVENTS_CONTACTS_UPDATE
|
||||||
@ -88,6 +89,9 @@ ENV AUTHENTICATION_JWT_SECRET="L=0YWt]b2w[WF>#>:&E`"
|
|||||||
|
|
||||||
ENV AUTHENTICATION_INSTANCE_NAME=$AUTHENTICATION_INSTANCE_NAME
|
ENV AUTHENTICATION_INSTANCE_NAME=$AUTHENTICATION_INSTANCE_NAME
|
||||||
ENV AUTHENTICATION_INSTANCE_WEBHOOK_URL=$AUTHENTICATION_INSTANCE_WEBHOOK_URL
|
ENV AUTHENTICATION_INSTANCE_WEBHOOK_URL=$AUTHENTICATION_INSTANCE_WEBHOOK_URL
|
||||||
|
ENV AUTHENTICATION_INSTANCE_CHATWOOT_ACCOUNT_ID=$AUTHENTICATION_INSTANCE_CHATWOOT_ACCOUNT_ID
|
||||||
|
ENV AUTHENTICATION_INSTANCE_CHATWOOT_TOKEN=$AUTHENTICATION_INSTANCE_CHATWOOT_TOKEN
|
||||||
|
ENV AUTHENTICATION_INSTANCE_CHATWOOT_URL=$AUTHENTICATION_INSTANCE_CHATWOOT_URL
|
||||||
ENV AUTHENTICATION_INSTANCE_MODE=$AUTHENTICATION_INSTANCE_MODE
|
ENV AUTHENTICATION_INSTANCE_MODE=$AUTHENTICATION_INSTANCE_MODE
|
||||||
|
|
||||||
RUN npm install
|
RUN npm install
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "evolution-api",
|
"name": "evolution-api",
|
||||||
"version": "1.1.4",
|
"version": "1.2.0",
|
||||||
"description": "Rest api for communication with WhatsApp",
|
"description": "Rest api for communication with WhatsApp",
|
||||||
"main": "./dist/src/main.js",
|
"main": "./dist/src/main.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@ -43,7 +43,8 @@
|
|||||||
"@adiwajshing/keyed-db": "^0.2.4",
|
"@adiwajshing/keyed-db": "^0.2.4",
|
||||||
"@ffmpeg-installer/ffmpeg": "^1.1.0",
|
"@ffmpeg-installer/ffmpeg": "^1.1.0",
|
||||||
"@hapi/boom": "^10.0.1",
|
"@hapi/boom": "^10.0.1",
|
||||||
"@whiskeysockets/baileys": "github:EvolutionAPI/Baileys",
|
"@whiskeysockets/baileys": "github:vphelipe/WhiskeySockets-Baileys#master",
|
||||||
|
"@figuro/chatwoot-sdk": "^1.1.14",
|
||||||
"axios": "^1.3.5",
|
"axios": "^1.3.5",
|
||||||
"class-validator": "^0.13.2",
|
"class-validator": "^0.13.2",
|
||||||
"compression": "^1.7.4",
|
"compression": "^1.7.4",
|
||||||
|
@ -4,7 +4,7 @@ import { join } from 'path';
|
|||||||
import { SRC_DIR } from './path.config';
|
import { SRC_DIR } from './path.config';
|
||||||
import { isBooleanString } from 'class-validator';
|
import { isBooleanString } from 'class-validator';
|
||||||
|
|
||||||
export type HttpServer = { TYPE: 'http' | 'https'; PORT: number };
|
export type HttpServer = { TYPE: 'http' | 'https'; PORT: number; URL: string };
|
||||||
|
|
||||||
export type HttpMethods = 'POST' | 'GET' | 'PUT' | 'DELETE';
|
export type HttpMethods = 'POST' | 'GET' | 'PUT' | 'DELETE';
|
||||||
export type Cors = {
|
export type Cors = {
|
||||||
@ -33,7 +33,6 @@ export type Log = {
|
|||||||
|
|
||||||
export type SaveData = {
|
export type SaveData = {
|
||||||
INSTANCE: boolean;
|
INSTANCE: boolean;
|
||||||
OLD_MESSAGE: boolean;
|
|
||||||
NEW_MESSAGE: boolean;
|
NEW_MESSAGE: boolean;
|
||||||
MESSAGE_UPDATE: boolean;
|
MESSAGE_UPDATE: boolean;
|
||||||
CONTACTS: boolean;
|
CONTACTS: boolean;
|
||||||
@ -77,6 +76,7 @@ export type EventsWebhook = {
|
|||||||
MESSAGES_SET: boolean;
|
MESSAGES_SET: boolean;
|
||||||
MESSAGES_UPSERT: boolean;
|
MESSAGES_UPSERT: boolean;
|
||||||
MESSAGES_UPDATE: boolean;
|
MESSAGES_UPDATE: boolean;
|
||||||
|
SEND_MESSAGE: boolean;
|
||||||
CONTACTS_SET: boolean;
|
CONTACTS_SET: boolean;
|
||||||
CONTACTS_UPDATE: boolean;
|
CONTACTS_UPDATE: boolean;
|
||||||
CONTACTS_UPSERT: boolean;
|
CONTACTS_UPSERT: boolean;
|
||||||
@ -98,6 +98,9 @@ export type Instance = {
|
|||||||
NAME: string;
|
NAME: string;
|
||||||
WEBHOOK_URL: string;
|
WEBHOOK_URL: string;
|
||||||
MODE: string;
|
MODE: string;
|
||||||
|
CHATWOOT_ACCOUNT_ID?: string;
|
||||||
|
CHATWOOT_TOKEN?: string;
|
||||||
|
CHATWOOT_URL?: string;
|
||||||
};
|
};
|
||||||
export type Auth = {
|
export type Auth = {
|
||||||
API_KEY: ApiKey;
|
API_KEY: ApiKey;
|
||||||
@ -170,6 +173,7 @@ export class ConfigService {
|
|||||||
SERVER: {
|
SERVER: {
|
||||||
TYPE: process.env.SERVER_TYPE as 'http' | 'https',
|
TYPE: process.env.SERVER_TYPE as 'http' | 'https',
|
||||||
PORT: Number.parseInt(process.env.SERVER_PORT),
|
PORT: Number.parseInt(process.env.SERVER_PORT),
|
||||||
|
URL: process.env.SERVER_URL,
|
||||||
},
|
},
|
||||||
CORS: {
|
CORS: {
|
||||||
ORIGIN: process.env.CORS_ORIGIN.split(','),
|
ORIGIN: process.env.CORS_ORIGIN.split(','),
|
||||||
@ -203,7 +207,6 @@ export class ConfigService {
|
|||||||
ENABLED: process.env?.DATABASE_ENABLED === 'true',
|
ENABLED: process.env?.DATABASE_ENABLED === 'true',
|
||||||
SAVE_DATA: {
|
SAVE_DATA: {
|
||||||
INSTANCE: process.env?.DATABASE_SAVE_DATA_INSTANCE === 'true',
|
INSTANCE: process.env?.DATABASE_SAVE_DATA_INSTANCE === 'true',
|
||||||
OLD_MESSAGE: process.env?.DATABASE_SAVE_DATA_OLD_MESSAGE === 'true',
|
|
||||||
NEW_MESSAGE: process.env?.DATABASE_SAVE_DATA_NEW_MESSAGE === 'true',
|
NEW_MESSAGE: process.env?.DATABASE_SAVE_DATA_NEW_MESSAGE === 'true',
|
||||||
MESSAGE_UPDATE: process.env?.DATABASE_SAVE_MESSAGE_UPDATE === 'true',
|
MESSAGE_UPDATE: process.env?.DATABASE_SAVE_MESSAGE_UPDATE === 'true',
|
||||||
CONTACTS: process.env?.DATABASE_SAVE_DATA_CONTACTS === 'true',
|
CONTACTS: process.env?.DATABASE_SAVE_DATA_CONTACTS === 'true',
|
||||||
@ -235,6 +238,7 @@ export class ConfigService {
|
|||||||
MESSAGES_SET: process.env?.WEBHOOK_EVENTS_MESSAGES_SET === 'true',
|
MESSAGES_SET: process.env?.WEBHOOK_EVENTS_MESSAGES_SET === 'true',
|
||||||
MESSAGES_UPSERT: process.env?.WEBHOOK_EVENTS_MESSAGES_UPSERT === 'true',
|
MESSAGES_UPSERT: process.env?.WEBHOOK_EVENTS_MESSAGES_UPSERT === 'true',
|
||||||
MESSAGES_UPDATE: process.env?.WEBHOOK_EVENTS_MESSAGES_UPDATE === 'true',
|
MESSAGES_UPDATE: process.env?.WEBHOOK_EVENTS_MESSAGES_UPDATE === 'true',
|
||||||
|
SEND_MESSAGE: process.env?.WEBHOOK_EVENTS_SEND_MESSAGE === 'true',
|
||||||
CONTACTS_SET: process.env?.WEBHOOK_EVENTS_CONTACTS_SET === 'true',
|
CONTACTS_SET: process.env?.WEBHOOK_EVENTS_CONTACTS_SET === 'true',
|
||||||
CONTACTS_UPDATE: process.env?.WEBHOOK_EVENTS_CONTACTS_UPDATE === 'true',
|
CONTACTS_UPDATE: process.env?.WEBHOOK_EVENTS_CONTACTS_UPDATE === 'true',
|
||||||
CONTACTS_UPSERT: process.env?.WEBHOOK_EVENTS_CONTACTS_UPSERT === 'true',
|
CONTACTS_UPSERT: process.env?.WEBHOOK_EVENTS_CONTACTS_UPSERT === 'true',
|
||||||
@ -275,6 +279,10 @@ export class ConfigService {
|
|||||||
NAME: process.env.AUTHENTICATION_INSTANCE_NAME,
|
NAME: process.env.AUTHENTICATION_INSTANCE_NAME,
|
||||||
WEBHOOK_URL: process.env.AUTHENTICATION_INSTANCE_WEBHOOK_URL,
|
WEBHOOK_URL: process.env.AUTHENTICATION_INSTANCE_WEBHOOK_URL,
|
||||||
MODE: process.env.AUTHENTICATION_INSTANCE_MODE,
|
MODE: process.env.AUTHENTICATION_INSTANCE_MODE,
|
||||||
|
CHATWOOT_ACCOUNT_ID:
|
||||||
|
process.env.AUTHENTICATION_INSTANCE_CHATWOOT_ACCOUNT_ID || '',
|
||||||
|
CHATWOOT_TOKEN: process.env.AUTHENTICATION_INSTANCE_CHATWOOT_TOKEN || '',
|
||||||
|
CHATWOOT_URL: process.env.AUTHENTICATION_INSTANCE_CHATWOOT_URL || '',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -8,10 +8,11 @@
|
|||||||
SERVER:
|
SERVER:
|
||||||
TYPE: http # https
|
TYPE: http # https
|
||||||
PORT: 8080 # 443
|
PORT: 8080 # 443
|
||||||
|
URL: localhost
|
||||||
|
|
||||||
CORS:
|
CORS:
|
||||||
ORIGIN:
|
ORIGIN:
|
||||||
- '*'
|
- "*"
|
||||||
# - yourdomain.com
|
# - yourdomain.com
|
||||||
METHODS:
|
METHODS:
|
||||||
- POST
|
- POST
|
||||||
@ -63,12 +64,11 @@ CLEAN_STORE:
|
|||||||
DATABASE:
|
DATABASE:
|
||||||
ENABLED: false
|
ENABLED: false
|
||||||
CONNECTION:
|
CONNECTION:
|
||||||
URI: 'mongodb://root:root@localhost:27017/?authSource=admin&readPreference=primary&ssl=false&directConnection=true'
|
URI: "mongodb://root:root@localhost:27017/?authSource=admin&readPreference=primary&ssl=false&directConnection=true"
|
||||||
DB_PREFIX_NAME: evolution
|
DB_PREFIX_NAME: evolution
|
||||||
# Choose the data you want to save in the application's database or store
|
# Choose the data you want to save in the application's database or store
|
||||||
SAVE_DATA:
|
SAVE_DATA:
|
||||||
INSTANCE: false
|
INSTANCE: false
|
||||||
OLD_MESSAGE: false
|
|
||||||
NEW_MESSAGE: false
|
NEW_MESSAGE: false
|
||||||
MESSAGE_UPDATE: false
|
MESSAGE_UPDATE: false
|
||||||
CONTACTS: false
|
CONTACTS: false
|
||||||
@ -76,10 +76,11 @@ DATABASE:
|
|||||||
|
|
||||||
REDIS:
|
REDIS:
|
||||||
ENABLED: false
|
ENABLED: false
|
||||||
URI: 'redis://localhost:6379'
|
URI: "redis://localhost:6379"
|
||||||
PREFIX_KEY: 'evolution'
|
PREFIX_KEY: "evolution"
|
||||||
|
|
||||||
# Webhook Settings
|
# Global Webhook Settings
|
||||||
|
# Each instance's Webhook URL and events will be requested at the time it is created
|
||||||
WEBHOOK:
|
WEBHOOK:
|
||||||
# Define a global webhook that will listen for enabled events from all instances
|
# Define a global webhook that will listen for enabled events from all instances
|
||||||
GLOBAL:
|
GLOBAL:
|
||||||
@ -88,13 +89,14 @@ WEBHOOK:
|
|||||||
# With this option activated, you work with a url per webhook event, respecting the global url and the name of each event
|
# With this option activated, you work with a url per webhook event, respecting the global url and the name of each event
|
||||||
WEBHOOK_BY_EVENTS: false
|
WEBHOOK_BY_EVENTS: false
|
||||||
# Automatically maps webhook paths
|
# Automatically maps webhook paths
|
||||||
# Set the events you want to hear
|
# Set the events you want to hear
|
||||||
EVENTS:
|
EVENTS:
|
||||||
APPLICATION_STARTUP: false
|
APPLICATION_STARTUP: false
|
||||||
QRCODE_UPDATED: true
|
QRCODE_UPDATED: true
|
||||||
MESSAGES_SET: true
|
MESSAGES_SET: true
|
||||||
MESSAGES_UPSERT: true
|
MESSAGES_UPSERT: true
|
||||||
MESSAGES_UPDATE: true
|
MESSAGES_UPDATE: true
|
||||||
|
SEND_MESSAGE: true
|
||||||
CONTACTS_SET: true
|
CONTACTS_SET: true
|
||||||
CONTACTS_UPSERT: true
|
CONTACTS_UPSERT: true
|
||||||
CONTACTS_UPDATE: true
|
CONTACTS_UPDATE: true
|
||||||
@ -112,7 +114,7 @@ WEBHOOK:
|
|||||||
|
|
||||||
CONFIG_SESSION_PHONE:
|
CONFIG_SESSION_PHONE:
|
||||||
# Name that will be displayed on smartphone connection
|
# Name that will be displayed on smartphone connection
|
||||||
CLIENT: 'Evolution API'
|
CLIENT: "Evolution API"
|
||||||
NAME: chrome # chrome | firefox | edge | opera | safari
|
NAME: chrome # chrome | firefox | edge | opera | safari
|
||||||
|
|
||||||
# Set qrcode display limit
|
# Set qrcode display limit
|
||||||
@ -120,6 +122,8 @@ QRCODE:
|
|||||||
LIMIT: 30
|
LIMIT: 30
|
||||||
|
|
||||||
# Defines an authentication type for the api
|
# Defines an authentication type for the api
|
||||||
|
# We recommend using the apikey because it will allow you to use a custom token,
|
||||||
|
# if you use jwt, a random token will be generated and may be expired and you will have to generate a new token
|
||||||
AUTHENTICATION:
|
AUTHENTICATION:
|
||||||
TYPE: apikey # jwt or apikey
|
TYPE: apikey # jwt or apikey
|
||||||
# Define a global apikey to access all instances
|
# Define a global apikey to access all instances
|
||||||
@ -134,8 +138,11 @@ AUTHENTICATION:
|
|||||||
SECRET: L=0YWt]b2w[WF>#>:&E`
|
SECRET: L=0YWt]b2w[WF>#>:&E`
|
||||||
# Set the instance name and webhook url to create an instance in init the application
|
# Set the instance name and webhook url to create an instance in init the application
|
||||||
INSTANCE:
|
INSTANCE:
|
||||||
# With this option activated, you work with a url per webhook event, respecting the local url and the name of each event
|
# With this option activated, you work with a url per webhook event, respecting the local url and the name of each event
|
||||||
MODE: server # container or server
|
MODE: server # container or server
|
||||||
# if you are using container mode, set the container name and the webhook url to default instance
|
# if you are using container mode, set the container name and the webhook url to default instance
|
||||||
NAME: evolution
|
NAME: evolution
|
||||||
WEBHOOK_URL: <url>
|
WEBHOOK_URL: <url>
|
||||||
|
CHATWOOT_ACCOUNT_ID: 1
|
||||||
|
CHATWOOT_TOKEN: 123456
|
||||||
|
CHATWOOT_URL: <url>
|
||||||
|
@ -39,6 +39,7 @@ export const instanceNameSchema: JSONSchema7 = {
|
|||||||
'MESSAGES_SET',
|
'MESSAGES_SET',
|
||||||
'MESSAGES_UPSERT',
|
'MESSAGES_UPSERT',
|
||||||
'MESSAGES_UPDATE',
|
'MESSAGES_UPDATE',
|
||||||
|
'SEND_MESSAGE',
|
||||||
'CONTACTS_SET',
|
'CONTACTS_SET',
|
||||||
'CONTACTS_UPSERT',
|
'CONTACTS_UPSERT',
|
||||||
'CONTACTS_UPDATE',
|
'CONTACTS_UPDATE',
|
||||||
@ -699,6 +700,16 @@ export const groupJidSchema: JSONSchema7 = {
|
|||||||
...isNotEmpty('groupJid'),
|
...isNotEmpty('groupJid'),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const getParticipantsSchema: JSONSchema7 = {
|
||||||
|
$id: v4(),
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
getParticipants: { type: 'string', enum: ['true', 'false'] },
|
||||||
|
},
|
||||||
|
required: ['getParticipants'],
|
||||||
|
...isNotEmpty('getParticipants'),
|
||||||
|
};
|
||||||
|
|
||||||
export const groupSendInviteSchema: JSONSchema7 = {
|
export const groupSendInviteSchema: JSONSchema7 = {
|
||||||
$id: v4(),
|
$id: v4(),
|
||||||
type: 'object',
|
type: 'object',
|
||||||
@ -835,6 +846,7 @@ export const webhookSchema: JSONSchema7 = {
|
|||||||
'MESSAGES_SET',
|
'MESSAGES_SET',
|
||||||
'MESSAGES_UPSERT',
|
'MESSAGES_UPSERT',
|
||||||
'MESSAGES_UPDATE',
|
'MESSAGES_UPDATE',
|
||||||
|
'SEND_MESSAGE',
|
||||||
'CONTACTS_SET',
|
'CONTACTS_SET',
|
||||||
'CONTACTS_UPSERT',
|
'CONTACTS_UPSERT',
|
||||||
'CONTACTS_UPDATE',
|
'CONTACTS_UPDATE',
|
||||||
@ -855,3 +867,17 @@ export const webhookSchema: JSONSchema7 = {
|
|||||||
required: ['url', 'enabled'],
|
required: ['url', 'enabled'],
|
||||||
...isNotEmpty('url'),
|
...isNotEmpty('url'),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const chatwootSchema: JSONSchema7 = {
|
||||||
|
$id: v4(),
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
enabled: { type: 'boolean', enum: [true, false] },
|
||||||
|
account_id: { type: 'string' },
|
||||||
|
token: { type: 'string' },
|
||||||
|
url: { type: 'string' },
|
||||||
|
sign_msg: { type: 'boolean', enum: [true, false] },
|
||||||
|
},
|
||||||
|
required: ['enabled', 'account_id', 'token', 'url', 'sign_msg'],
|
||||||
|
...isNotEmpty('account_id', 'token', 'url', 'sign_msg'),
|
||||||
|
};
|
||||||
|
@ -5,7 +5,7 @@ import { validate } from 'jsonschema';
|
|||||||
import { BadRequestException } from '../../exceptions';
|
import { BadRequestException } from '../../exceptions';
|
||||||
import 'express-async-errors';
|
import 'express-async-errors';
|
||||||
import { Logger } from '../../config/logger.config';
|
import { Logger } from '../../config/logger.config';
|
||||||
import { GroupInvite, GroupJid } from '../dto/group.dto';
|
import { GetParticipant, GroupInvite, GroupJid } from '../dto/group.dto';
|
||||||
|
|
||||||
type DataValidate<T> = {
|
type DataValidate<T> = {
|
||||||
request: Request;
|
request: Request;
|
||||||
@ -181,4 +181,47 @@ export abstract class RouterBroker {
|
|||||||
|
|
||||||
return await execute(instance, ref);
|
return await execute(instance, ref);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async getParticipantsValidate<T>(args: DataValidate<T>) {
|
||||||
|
const { request, ClassRef, schema, execute } = args;
|
||||||
|
|
||||||
|
const getParticipants = request.query as unknown as GetParticipant;
|
||||||
|
|
||||||
|
if (!getParticipants?.getParticipants) {
|
||||||
|
throw new BadRequestException(
|
||||||
|
'The getParticipants needs to be informed in the query',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const instance = request.params as unknown as InstanceDto;
|
||||||
|
const body = request.body;
|
||||||
|
|
||||||
|
const ref = new ClassRef();
|
||||||
|
|
||||||
|
Object.assign(body, getParticipants);
|
||||||
|
Object.assign(ref, body);
|
||||||
|
|
||||||
|
const v = validate(ref, schema);
|
||||||
|
|
||||||
|
console.log(v, '@checkei aqui');
|
||||||
|
|
||||||
|
if (!v.valid) {
|
||||||
|
const message: any[] = v.errors.map(({ property, stack, schema }) => {
|
||||||
|
let message: string;
|
||||||
|
if (schema['description']) {
|
||||||
|
message = schema['description'];
|
||||||
|
} else {
|
||||||
|
message = stack.replace('instance.', '');
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
property: property.replace('instance.', ''),
|
||||||
|
message,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
logger.error([...message]);
|
||||||
|
throw new BadRequestException(...message);
|
||||||
|
}
|
||||||
|
|
||||||
|
return await execute(instance, ref);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
85
src/whatsapp/controllers/chatwoot.controller.ts
Normal file
85
src/whatsapp/controllers/chatwoot.controller.ts
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
import { isURL } from 'class-validator';
|
||||||
|
import { BadRequestException } from '../../exceptions';
|
||||||
|
import { InstanceDto } from '../dto/instance.dto';
|
||||||
|
import { ChatwootDto } from '../dto/chatwoot.dto';
|
||||||
|
import { ChatwootService } from '../services/chatwoot.service';
|
||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
import { waMonitor } from '../whatsapp.module';
|
||||||
|
import { ConfigService, HttpServer } from '../../config/env.config';
|
||||||
|
|
||||||
|
const logger = new Logger('ChatwootController');
|
||||||
|
|
||||||
|
export class ChatwootController {
|
||||||
|
constructor(
|
||||||
|
private readonly chatwootService: ChatwootService,
|
||||||
|
private readonly configService: ConfigService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async createChatwoot(instance: InstanceDto, data: ChatwootDto) {
|
||||||
|
logger.verbose(
|
||||||
|
'requested createChatwoot from ' + instance.instanceName + ' instance',
|
||||||
|
);
|
||||||
|
|
||||||
|
if (data.enabled) {
|
||||||
|
if (!isURL(data.url, { require_tld: false })) {
|
||||||
|
throw new BadRequestException('url is not valid');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.account_id) {
|
||||||
|
throw new BadRequestException('account_id is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.token) {
|
||||||
|
throw new BadRequestException('token is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.sign_msg) {
|
||||||
|
throw new BadRequestException('sign_msg is required');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.enabled) {
|
||||||
|
logger.verbose('chatwoot disabled');
|
||||||
|
data.account_id = '';
|
||||||
|
data.token = '';
|
||||||
|
data.url = '';
|
||||||
|
data.sign_msg = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
data.name_inbox = instance.instanceName;
|
||||||
|
|
||||||
|
const result = this.chatwootService.create(instance, data);
|
||||||
|
|
||||||
|
const urlServer = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
const response = {
|
||||||
|
...result,
|
||||||
|
webhook_url: `${urlServer}/chatwoot/webhook/${instance.instanceName}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async findChatwoot(instance: InstanceDto) {
|
||||||
|
logger.verbose('requested findChatwoot from ' + instance.instanceName + ' instance');
|
||||||
|
const result = await this.chatwootService.find(instance);
|
||||||
|
|
||||||
|
const urlServer = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
const response = {
|
||||||
|
...result,
|
||||||
|
webhook_url: `${urlServer}/chatwoot/webhook/${instance.instanceName}`,
|
||||||
|
};
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async receiveWebhook(instance: InstanceDto, data: any) {
|
||||||
|
logger.verbose(
|
||||||
|
'requested receiveWebhook from ' + instance.instanceName + ' instance',
|
||||||
|
);
|
||||||
|
const chatwootService = new ChatwootService(waMonitor);
|
||||||
|
|
||||||
|
return chatwootService.receiveWebhook(instance, data);
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,6 @@
|
|||||||
import {
|
import {
|
||||||
CreateGroupDto,
|
CreateGroupDto,
|
||||||
|
GetParticipant,
|
||||||
GroupDescriptionDto,
|
GroupDescriptionDto,
|
||||||
GroupInvite,
|
GroupInvite,
|
||||||
GroupJid,
|
GroupJid,
|
||||||
@ -59,11 +60,13 @@ export class GroupController {
|
|||||||
return await this.waMonitor.waInstances[instance.instanceName].findGroup(groupJid);
|
return await this.waMonitor.waInstances[instance.instanceName].findGroup(groupJid);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async fetchAllGroups(instance: InstanceDto) {
|
public async fetchAllGroups(instance: InstanceDto, getPaticipants: GetParticipant) {
|
||||||
logger.verbose(
|
logger.verbose(
|
||||||
'requested fetchAllGroups from ' + instance.instanceName + ' instance',
|
'requested fetchAllGroups from ' + instance.instanceName + ' instance',
|
||||||
);
|
);
|
||||||
return await this.waMonitor.waInstances[instance.instanceName].fetchAllGroups();
|
return await this.waMonitor.waInstances[instance.instanceName].fetchAllGroups(
|
||||||
|
getPaticipants,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async inviteCode(instance: InstanceDto, groupJid: GroupJid) {
|
public async inviteCode(instance: InstanceDto, groupJid: GroupJid) {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { delay } from '@whiskeysockets/baileys';
|
import { delay } from '@whiskeysockets/baileys';
|
||||||
import EventEmitter2 from 'eventemitter2';
|
import EventEmitter2 from 'eventemitter2';
|
||||||
import { Auth, ConfigService } from '../../config/env.config';
|
import { Auth, ConfigService, HttpServer } from '../../config/env.config';
|
||||||
import { BadRequestException, InternalServerErrorException } from '../../exceptions';
|
import { BadRequestException, InternalServerErrorException } from '../../exceptions';
|
||||||
import { InstanceDto } from '../dto/instance.dto';
|
import { InstanceDto } from '../dto/instance.dto';
|
||||||
import { RepositoryBroker } from '../repository/repository.manager';
|
import { RepositoryBroker } from '../repository/repository.manager';
|
||||||
@ -8,6 +8,7 @@ import { AuthService, OldToken } from '../services/auth.service';
|
|||||||
import { WAMonitoringService } from '../services/monitor.service';
|
import { WAMonitoringService } from '../services/monitor.service';
|
||||||
import { WAStartupService } from '../services/whatsapp.service';
|
import { WAStartupService } from '../services/whatsapp.service';
|
||||||
import { WebhookService } from '../services/webhook.service';
|
import { WebhookService } from '../services/webhook.service';
|
||||||
|
import { ChatwootService } from '../services/chatwoot.service';
|
||||||
import { Logger } from '../../config/logger.config';
|
import { Logger } from '../../config/logger.config';
|
||||||
import { wa } from '../types/wa.types';
|
import { wa } from '../types/wa.types';
|
||||||
import { RedisCache } from '../../db/redis.client';
|
import { RedisCache } from '../../db/redis.client';
|
||||||
@ -20,6 +21,7 @@ export class InstanceController {
|
|||||||
private readonly eventEmitter: EventEmitter2,
|
private readonly eventEmitter: EventEmitter2,
|
||||||
private readonly authService: AuthService,
|
private readonly authService: AuthService,
|
||||||
private readonly webhookService: WebhookService,
|
private readonly webhookService: WebhookService,
|
||||||
|
private readonly chatwootService: ChatwootService,
|
||||||
private readonly cache: RedisCache,
|
private readonly cache: RedisCache,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@ -32,6 +34,10 @@ export class InstanceController {
|
|||||||
events,
|
events,
|
||||||
qrcode,
|
qrcode,
|
||||||
token,
|
token,
|
||||||
|
chatwoot_account_id,
|
||||||
|
chatwoot_token,
|
||||||
|
chatwoot_url,
|
||||||
|
chatwoot_sign_msg,
|
||||||
}: InstanceDto) {
|
}: InstanceDto) {
|
||||||
this.logger.verbose('requested createInstance from ' + instanceName + ' instance');
|
this.logger.verbose('requested createInstance from ' + instanceName + ' instance');
|
||||||
|
|
||||||
@ -91,16 +97,71 @@ export class InstanceController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.logger.verbose('instance created');
|
if (
|
||||||
this.logger.verbose({
|
!chatwoot_account_id ||
|
||||||
instance: {
|
!chatwoot_token ||
|
||||||
instanceName: instance.instanceName,
|
!chatwoot_url ||
|
||||||
status: 'created',
|
!chatwoot_sign_msg
|
||||||
},
|
) {
|
||||||
hash,
|
this.logger.verbose('instance created');
|
||||||
webhook,
|
this.logger.verbose({
|
||||||
events: getEvents,
|
instance: {
|
||||||
});
|
instanceName: instance.instanceName,
|
||||||
|
status: 'created',
|
||||||
|
},
|
||||||
|
hash,
|
||||||
|
webhook,
|
||||||
|
events: getEvents,
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
instance: {
|
||||||
|
instanceName: instance.instanceName,
|
||||||
|
status: 'created',
|
||||||
|
},
|
||||||
|
hash,
|
||||||
|
webhook,
|
||||||
|
events: getEvents,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!chatwoot_account_id) {
|
||||||
|
throw new BadRequestException('account_id is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!chatwoot_token) {
|
||||||
|
throw new BadRequestException('token is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!chatwoot_url) {
|
||||||
|
throw new BadRequestException('url is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!chatwoot_sign_msg) {
|
||||||
|
throw new BadRequestException('sign_msg is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
const urlServer = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
try {
|
||||||
|
this.chatwootService.create(instance, {
|
||||||
|
enabled: true,
|
||||||
|
account_id: chatwoot_account_id,
|
||||||
|
token: chatwoot_token,
|
||||||
|
url: chatwoot_url,
|
||||||
|
sign_msg: chatwoot_sign_msg,
|
||||||
|
name_inbox: instance.instanceName,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.chatwootService.initInstanceChatwoot(
|
||||||
|
instance,
|
||||||
|
instance.instanceName,
|
||||||
|
`${urlServer}/chatwoot/webhook/${instance.instanceName}`,
|
||||||
|
qrcode,
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
this.logger.log(error);
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
instance: {
|
instance: {
|
||||||
@ -108,8 +169,15 @@ export class InstanceController {
|
|||||||
status: 'created',
|
status: 'created',
|
||||||
},
|
},
|
||||||
hash,
|
hash,
|
||||||
webhook,
|
chatwoot: {
|
||||||
events: getEvents,
|
enabled: true,
|
||||||
|
account_id: chatwoot_account_id,
|
||||||
|
token: chatwoot_token,
|
||||||
|
url: chatwoot_url,
|
||||||
|
sign_msg: chatwoot_sign_msg,
|
||||||
|
name_inbox: instance.instanceName,
|
||||||
|
webhook_url: `${urlServer}/chatwoot/webhook/${instance.instanceName}`,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
this.logger.verbose('server mode');
|
this.logger.verbose('server mode');
|
||||||
@ -159,27 +227,84 @@ export class InstanceController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let getQrcode: wa.QrCode;
|
if (
|
||||||
|
!chatwoot_account_id ||
|
||||||
|
!chatwoot_token ||
|
||||||
|
!chatwoot_url ||
|
||||||
|
!chatwoot_sign_msg
|
||||||
|
) {
|
||||||
|
let getQrcode: wa.QrCode;
|
||||||
|
|
||||||
if (qrcode) {
|
if (qrcode) {
|
||||||
this.logger.verbose('creating qrcode');
|
this.logger.verbose('creating qrcode');
|
||||||
await instance.connectToWhatsapp();
|
await instance.connectToWhatsapp();
|
||||||
await delay(2000);
|
await delay(2000);
|
||||||
getQrcode = instance.qrCode;
|
getQrcode = instance.qrCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('instance created');
|
||||||
|
this.logger.verbose({
|
||||||
|
instance: {
|
||||||
|
instanceName: instance.instanceName,
|
||||||
|
status: 'created',
|
||||||
|
},
|
||||||
|
hash,
|
||||||
|
webhook,
|
||||||
|
webhook_by_events,
|
||||||
|
events: getEvents,
|
||||||
|
qrcode: getQrcode,
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
instance: {
|
||||||
|
instanceName: instance.instanceName,
|
||||||
|
status: 'created',
|
||||||
|
},
|
||||||
|
hash,
|
||||||
|
webhook,
|
||||||
|
webhook_by_events,
|
||||||
|
events: getEvents,
|
||||||
|
qrcode: getQrcode,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
this.logger.verbose('instance created');
|
if (!chatwoot_account_id) {
|
||||||
this.logger.verbose({
|
throw new BadRequestException('account_id is required');
|
||||||
instance: {
|
}
|
||||||
instanceName: instance.instanceName,
|
|
||||||
status: 'created',
|
if (!chatwoot_token) {
|
||||||
},
|
throw new BadRequestException('token is required');
|
||||||
hash,
|
}
|
||||||
webhook,
|
|
||||||
webhook_by_events,
|
if (!chatwoot_url) {
|
||||||
events: getEvents,
|
throw new BadRequestException('url is required');
|
||||||
qrcode: getQrcode,
|
}
|
||||||
});
|
|
||||||
|
if (!chatwoot_sign_msg) {
|
||||||
|
throw new BadRequestException('sign_msg is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
const urlServer = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
try {
|
||||||
|
this.chatwootService.create(instance, {
|
||||||
|
enabled: true,
|
||||||
|
account_id: chatwoot_account_id,
|
||||||
|
token: chatwoot_token,
|
||||||
|
url: chatwoot_url,
|
||||||
|
sign_msg: chatwoot_sign_msg,
|
||||||
|
name_inbox: instance.instanceName,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.chatwootService.initInstanceChatwoot(
|
||||||
|
instance,
|
||||||
|
instance.instanceName,
|
||||||
|
`${urlServer}/chatwoot/webhook/${instance.instanceName}`,
|
||||||
|
qrcode,
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
this.logger.log(error);
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
instance: {
|
instance: {
|
||||||
@ -190,7 +315,15 @@ export class InstanceController {
|
|||||||
webhook,
|
webhook,
|
||||||
webhook_by_events,
|
webhook_by_events,
|
||||||
events: getEvents,
|
events: getEvents,
|
||||||
qrcode: getQrcode,
|
chatwoot: {
|
||||||
|
enabled: true,
|
||||||
|
account_id: chatwoot_account_id,
|
||||||
|
token: chatwoot_token,
|
||||||
|
url: chatwoot_url,
|
||||||
|
sign_msg: chatwoot_sign_msg,
|
||||||
|
name_inbox: instance.instanceName,
|
||||||
|
webhook_url: `${urlServer}/chatwoot/webhook/${instance.instanceName}`,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -268,6 +401,14 @@ export class InstanceController {
|
|||||||
|
|
||||||
public async logout({ instanceName }: InstanceDto) {
|
public async logout({ instanceName }: InstanceDto) {
|
||||||
this.logger.verbose('requested logout from ' + instanceName + ' instance');
|
this.logger.verbose('requested logout from ' + instanceName + ' instance');
|
||||||
|
const stateConn = await this.connectionState({ instanceName });
|
||||||
|
|
||||||
|
if (stateConn.state === 'close') {
|
||||||
|
throw new BadRequestException(
|
||||||
|
'The "' + instanceName + '" instance is not connected',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
this.logger.verbose('logging out instance: ' + instanceName);
|
this.logger.verbose('logging out instance: ' + instanceName);
|
||||||
await this.waMonitor.waInstances[instanceName]?.client?.logout(
|
await this.waMonitor.waInstances[instanceName]?.client?.logout(
|
||||||
@ -288,10 +429,9 @@ export class InstanceController {
|
|||||||
const stateConn = await this.connectionState({ instanceName });
|
const stateConn = await this.connectionState({ instanceName });
|
||||||
|
|
||||||
if (stateConn.state === 'open') {
|
if (stateConn.state === 'open') {
|
||||||
throw new BadRequestException([
|
throw new BadRequestException(
|
||||||
'Deletion failed',
|
'The "' + instanceName + '" instance needs to be disconnected',
|
||||||
'The instance needs to be disconnected',
|
);
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
if (stateConn.state === 'connecting') {
|
if (stateConn.state === 'connecting') {
|
||||||
|
8
src/whatsapp/dto/chatwoot.dto.ts
Normal file
8
src/whatsapp/dto/chatwoot.dto.ts
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
export class ChatwootDto {
|
||||||
|
enabled?: boolean;
|
||||||
|
account_id?: string;
|
||||||
|
token?: string;
|
||||||
|
url?: string;
|
||||||
|
name_inbox?: string;
|
||||||
|
sign_msg?: boolean;
|
||||||
|
}
|
@ -23,6 +23,10 @@ export class GroupJid {
|
|||||||
groupJid: string;
|
groupJid: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class GetParticipant {
|
||||||
|
getParticipants: string;
|
||||||
|
}
|
||||||
|
|
||||||
export class GroupInvite {
|
export class GroupInvite {
|
||||||
inviteCode: string;
|
inviteCode: string;
|
||||||
}
|
}
|
||||||
|
@ -5,4 +5,8 @@ export class InstanceDto {
|
|||||||
events?: string[];
|
events?: string[];
|
||||||
qrcode?: boolean;
|
qrcode?: boolean;
|
||||||
token?: string;
|
token?: string;
|
||||||
|
chatwoot_account_id?: string;
|
||||||
|
chatwoot_token?: string;
|
||||||
|
chatwoot_url?: string;
|
||||||
|
chatwoot_sign_msg?: boolean;
|
||||||
}
|
}
|
||||||
|
29
src/whatsapp/models/chatwoot.model.ts
Normal file
29
src/whatsapp/models/chatwoot.model.ts
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
import { Schema } from 'mongoose';
|
||||||
|
import { dbserver } from '../../db/db.connect';
|
||||||
|
|
||||||
|
export class ChatwootRaw {
|
||||||
|
_id?: string;
|
||||||
|
enabled?: boolean;
|
||||||
|
account_id?: string;
|
||||||
|
token?: string;
|
||||||
|
url?: string;
|
||||||
|
name_inbox?: string;
|
||||||
|
sign_msg?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const chatwootSchema = new Schema<ChatwootRaw>({
|
||||||
|
_id: { type: String, _id: true },
|
||||||
|
enabled: { type: Boolean, required: true },
|
||||||
|
account_id: { type: String, required: true },
|
||||||
|
token: { type: String, required: true },
|
||||||
|
url: { type: String, required: true },
|
||||||
|
name_inbox: { type: String, required: true },
|
||||||
|
sign_msg: { type: Boolean, required: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ChatwootModel = dbserver?.model(
|
||||||
|
ChatwootRaw.name,
|
||||||
|
chatwootSchema,
|
||||||
|
'chatwoot',
|
||||||
|
);
|
||||||
|
export type IChatwootModel = typeof ChatwootModel;
|
@ -3,3 +3,4 @@ export * from './contact.model';
|
|||||||
export * from './message.model';
|
export * from './message.model';
|
||||||
export * from './auth.model';
|
export * from './auth.model';
|
||||||
export * from './webhook.model';
|
export * from './webhook.model';
|
||||||
|
export * from './chatwoot.model';
|
||||||
|
75
src/whatsapp/repository/chatwoot.repository.ts
Normal file
75
src/whatsapp/repository/chatwoot.repository.ts
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
import { IInsert, Repository } from '../abstract/abstract.repository';
|
||||||
|
import { ConfigService } from '../../config/env.config';
|
||||||
|
import { join } from 'path';
|
||||||
|
import { readFileSync } from 'fs';
|
||||||
|
import { IChatwootModel, ChatwootRaw } from '../models';
|
||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
|
||||||
|
export class ChatwootRepository extends Repository {
|
||||||
|
constructor(
|
||||||
|
private readonly chatwootModel: IChatwootModel,
|
||||||
|
private readonly configService: ConfigService,
|
||||||
|
) {
|
||||||
|
super(configService);
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly logger = new Logger('ChatwootRepository');
|
||||||
|
|
||||||
|
public async create(data: ChatwootRaw, instance: string): Promise<IInsert> {
|
||||||
|
try {
|
||||||
|
this.logger.verbose('creating chatwoot');
|
||||||
|
if (this.dbSettings.ENABLED) {
|
||||||
|
this.logger.verbose('saving chatwoot to db');
|
||||||
|
const insert = await this.chatwootModel.replaceOne(
|
||||||
|
{ _id: instance },
|
||||||
|
{ ...data },
|
||||||
|
{ upsert: true },
|
||||||
|
);
|
||||||
|
|
||||||
|
this.logger.verbose(
|
||||||
|
'chatwoot saved to db: ' + insert.modifiedCount + ' chatwoot',
|
||||||
|
);
|
||||||
|
return { insertCount: insert.modifiedCount };
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('saving chatwoot to store');
|
||||||
|
|
||||||
|
this.writeStore<ChatwootRaw>({
|
||||||
|
path: join(this.storePath, 'chatwoot'),
|
||||||
|
fileName: instance,
|
||||||
|
data,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.logger.verbose(
|
||||||
|
'chatwoot saved to store in path: ' +
|
||||||
|
join(this.storePath, 'chatwoot') +
|
||||||
|
'/' +
|
||||||
|
instance,
|
||||||
|
);
|
||||||
|
|
||||||
|
this.logger.verbose('chatwoot created');
|
||||||
|
return { insertCount: 1 };
|
||||||
|
} catch (error) {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async find(instance: string): Promise<ChatwootRaw> {
|
||||||
|
try {
|
||||||
|
this.logger.verbose('finding chatwoot');
|
||||||
|
if (this.dbSettings.ENABLED) {
|
||||||
|
this.logger.verbose('finding chatwoot in db');
|
||||||
|
return await this.chatwootModel.findOne({ _id: instance });
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('finding chatwoot in store');
|
||||||
|
return JSON.parse(
|
||||||
|
readFileSync(join(this.storePath, 'chatwoot', instance + '.json'), {
|
||||||
|
encoding: 'utf-8',
|
||||||
|
}),
|
||||||
|
) as ChatwootRaw;
|
||||||
|
} catch (error) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -4,6 +4,7 @@ import { ContactRepository } from './contact.repository';
|
|||||||
import { MessageUpRepository } from './messageUp.repository';
|
import { MessageUpRepository } from './messageUp.repository';
|
||||||
import { MongoClient } from 'mongodb';
|
import { MongoClient } from 'mongodb';
|
||||||
import { WebhookRepository } from './webhook.repository';
|
import { WebhookRepository } from './webhook.repository';
|
||||||
|
import { ChatwootRepository } from './chatwoot.repository';
|
||||||
import { AuthRepository } from './auth.repository';
|
import { AuthRepository } from './auth.repository';
|
||||||
import { Auth, ConfigService, Database } from '../../config/env.config';
|
import { Auth, ConfigService, Database } from '../../config/env.config';
|
||||||
import { execSync } from 'child_process';
|
import { execSync } from 'child_process';
|
||||||
@ -17,6 +18,7 @@ export class RepositoryBroker {
|
|||||||
public readonly contact: ContactRepository,
|
public readonly contact: ContactRepository,
|
||||||
public readonly messageUpdate: MessageUpRepository,
|
public readonly messageUpdate: MessageUpRepository,
|
||||||
public readonly webhook: WebhookRepository,
|
public readonly webhook: WebhookRepository,
|
||||||
|
public readonly chatwoot: ChatwootRepository,
|
||||||
public readonly auth: AuthRepository,
|
public readonly auth: AuthRepository,
|
||||||
private configService: ConfigService,
|
private configService: ConfigService,
|
||||||
dbServer?: MongoClient,
|
dbServer?: MongoClient,
|
||||||
@ -64,6 +66,9 @@ export class RepositoryBroker {
|
|||||||
this.logger.verbose('creating webhook path: ' + join(storePath, 'webhook'));
|
this.logger.verbose('creating webhook path: ' + join(storePath, 'webhook'));
|
||||||
execSync(`mkdir -p ${join(storePath, 'webhook')}`);
|
execSync(`mkdir -p ${join(storePath, 'webhook')}`);
|
||||||
|
|
||||||
|
this.logger.verbose('creating chatwoot path: ' + join(storePath, 'chatwoot'));
|
||||||
|
execSync(`mkdir -p ${join(storePath, 'chatwoot')}`);
|
||||||
|
|
||||||
this.logger.verbose('creating temp path: ' + join(storePath, 'temp'));
|
this.logger.verbose('creating temp path: ' + join(storePath, 'temp'));
|
||||||
execSync(`mkdir -p ${join(storePath, 'temp')}`);
|
execSync(`mkdir -p ${join(storePath, 'temp')}`);
|
||||||
}
|
}
|
||||||
|
68
src/whatsapp/routers/chatwoot.router.ts
Normal file
68
src/whatsapp/routers/chatwoot.router.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import { RequestHandler, Router } from 'express';
|
||||||
|
import { instanceNameSchema, chatwootSchema } from '../../validate/validate.schema';
|
||||||
|
import { RouterBroker } from '../abstract/abstract.router';
|
||||||
|
import { InstanceDto } from '../dto/instance.dto';
|
||||||
|
import { ChatwootDto } from '../dto/chatwoot.dto';
|
||||||
|
import { chatwootController } from '../whatsapp.module';
|
||||||
|
import { ChatwootService } from '../services/chatwoot.service';
|
||||||
|
import { HttpStatus } from './index.router';
|
||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
|
||||||
|
const logger = new Logger('ChatwootRouter');
|
||||||
|
|
||||||
|
export class ChatwootRouter extends RouterBroker {
|
||||||
|
constructor(...guards: RequestHandler[]) {
|
||||||
|
super();
|
||||||
|
this.router
|
||||||
|
.post(this.routerPath('set'), ...guards, async (req, res) => {
|
||||||
|
logger.verbose('request received in setChatwoot');
|
||||||
|
logger.verbose('request body: ');
|
||||||
|
logger.verbose(req.body);
|
||||||
|
|
||||||
|
logger.verbose('request query: ');
|
||||||
|
logger.verbose(req.query);
|
||||||
|
const response = await this.dataValidate<ChatwootDto>({
|
||||||
|
request: req,
|
||||||
|
schema: chatwootSchema,
|
||||||
|
ClassRef: ChatwootDto,
|
||||||
|
execute: (instance, data) => chatwootController.createChatwoot(instance, data),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(HttpStatus.CREATED).json(response);
|
||||||
|
})
|
||||||
|
.get(this.routerPath('find'), ...guards, async (req, res) => {
|
||||||
|
logger.verbose('request received in findChatwoot');
|
||||||
|
logger.verbose('request body: ');
|
||||||
|
logger.verbose(req.body);
|
||||||
|
|
||||||
|
logger.verbose('request query: ');
|
||||||
|
logger.verbose(req.query);
|
||||||
|
const response = await this.dataValidate<InstanceDto>({
|
||||||
|
request: req,
|
||||||
|
schema: instanceNameSchema,
|
||||||
|
ClassRef: InstanceDto,
|
||||||
|
execute: (instance) => chatwootController.findChatwoot(instance),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(HttpStatus.OK).json(response);
|
||||||
|
})
|
||||||
|
.post(this.routerPath('webhook'), async (req, res) => {
|
||||||
|
logger.verbose('request received in findChatwoot');
|
||||||
|
logger.verbose('request body: ');
|
||||||
|
logger.verbose(req.body);
|
||||||
|
|
||||||
|
logger.verbose('request query: ');
|
||||||
|
logger.verbose(req.query);
|
||||||
|
const response = await this.dataValidate<InstanceDto>({
|
||||||
|
request: req,
|
||||||
|
schema: instanceNameSchema,
|
||||||
|
ClassRef: InstanceDto,
|
||||||
|
execute: (instance, data) => chatwootController.receiveWebhook(instance, data),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(HttpStatus.OK).json(response);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public readonly router = Router();
|
||||||
|
}
|
@ -10,6 +10,7 @@ import {
|
|||||||
updateGroupDescriptionSchema,
|
updateGroupDescriptionSchema,
|
||||||
groupInviteSchema,
|
groupInviteSchema,
|
||||||
groupSendInviteSchema,
|
groupSendInviteSchema,
|
||||||
|
getParticipantsSchema,
|
||||||
} from '../../validate/validate.schema';
|
} from '../../validate/validate.schema';
|
||||||
import { RouterBroker } from '../abstract/abstract.router';
|
import { RouterBroker } from '../abstract/abstract.router';
|
||||||
import {
|
import {
|
||||||
@ -23,6 +24,7 @@ import {
|
|||||||
GroupUpdateSettingDto,
|
GroupUpdateSettingDto,
|
||||||
GroupToggleEphemeralDto,
|
GroupToggleEphemeralDto,
|
||||||
GroupSendInvite,
|
GroupSendInvite,
|
||||||
|
GetParticipant,
|
||||||
} from '../dto/group.dto';
|
} from '../dto/group.dto';
|
||||||
import { groupController } from '../whatsapp.module';
|
import { groupController } from '../whatsapp.module';
|
||||||
import { HttpStatus } from './index.router';
|
import { HttpStatus } from './index.router';
|
||||||
@ -123,11 +125,11 @@ export class GroupRouter extends RouterBroker {
|
|||||||
|
|
||||||
logger.verbose('request query: ');
|
logger.verbose('request query: ');
|
||||||
logger.verbose(req.query);
|
logger.verbose(req.query);
|
||||||
const response = await this.groupNoValidate<GroupJid>({
|
const response = await this.getParticipantsValidate<GetParticipant>({
|
||||||
request: req,
|
request: req,
|
||||||
schema: {},
|
schema: getParticipantsSchema,
|
||||||
ClassRef: GroupJid,
|
ClassRef: GetParticipant,
|
||||||
execute: (instance) => groupController.fetchAllGroups(instance),
|
execute: (instance, data) => groupController.fetchAllGroups(instance, data),
|
||||||
});
|
});
|
||||||
|
|
||||||
res.status(HttpStatus.OK).json(response);
|
res.status(HttpStatus.OK).json(response);
|
||||||
|
@ -8,6 +8,7 @@ import { InstanceRouter } from './instance.router';
|
|||||||
import { MessageRouter } from './sendMessage.router';
|
import { MessageRouter } from './sendMessage.router';
|
||||||
import { ViewsRouter } from './view.router';
|
import { ViewsRouter } from './view.router';
|
||||||
import { WebhookRouter } from './webhook.router';
|
import { WebhookRouter } from './webhook.router';
|
||||||
|
import { ChatwootRouter } from './chatwoot.router';
|
||||||
|
|
||||||
enum HttpStatus {
|
enum HttpStatus {
|
||||||
OK = 200,
|
OK = 200,
|
||||||
@ -24,6 +25,12 @@ const authType = configService.get<Auth>('AUTHENTICATION').TYPE;
|
|||||||
const guards = [instanceExistsGuard, instanceLoggedGuard, authGuard[authType]];
|
const guards = [instanceExistsGuard, instanceLoggedGuard, authGuard[authType]];
|
||||||
|
|
||||||
router
|
router
|
||||||
|
.get('/', (req, res) => {
|
||||||
|
res.status(HttpStatus.OK).json({
|
||||||
|
status: HttpStatus.OK,
|
||||||
|
message: 'Welcome to the Evolution API, it is working!',
|
||||||
|
});
|
||||||
|
})
|
||||||
.use(
|
.use(
|
||||||
'/instance',
|
'/instance',
|
||||||
new InstanceRouter(configService, ...guards).router,
|
new InstanceRouter(configService, ...guards).router,
|
||||||
@ -32,6 +39,7 @@ router
|
|||||||
.use('/message', new MessageRouter(...guards).router)
|
.use('/message', new MessageRouter(...guards).router)
|
||||||
.use('/chat', new ChatRouter(...guards).router)
|
.use('/chat', new ChatRouter(...guards).router)
|
||||||
.use('/group', new GroupRouter(...guards).router)
|
.use('/group', new GroupRouter(...guards).router)
|
||||||
.use('/webhook', new WebhookRouter(...guards).router);
|
.use('/webhook', new WebhookRouter(...guards).router)
|
||||||
|
.use('/chatwoot', new ChatwootRouter(...guards).router);
|
||||||
|
|
||||||
export { router, HttpStatus };
|
export { router, HttpStatus };
|
||||||
|
1430
src/whatsapp/services/chatwoot.service.ts
Normal file
1430
src/whatsapp/services/chatwoot.service.ts
Normal file
File diff suppressed because it is too large
Load Diff
@ -9,6 +9,7 @@ import {
|
|||||||
ConfigService,
|
ConfigService,
|
||||||
Database,
|
Database,
|
||||||
DelInstance,
|
DelInstance,
|
||||||
|
HttpServer,
|
||||||
Redis,
|
Redis,
|
||||||
} from '../../config/env.config';
|
} from '../../config/env.config';
|
||||||
import { RepositoryBroker } from '../repository/repository.manager';
|
import { RepositoryBroker } from '../repository/repository.manager';
|
||||||
@ -83,6 +84,19 @@ export class WAMonitoringService {
|
|||||||
for await (const [key, value] of Object.entries(this.waInstances)) {
|
for await (const [key, value] of Object.entries(this.waInstances)) {
|
||||||
if (value) {
|
if (value) {
|
||||||
this.logger.verbose('get instance info: ' + key);
|
this.logger.verbose('get instance info: ' + key);
|
||||||
|
let chatwoot: any;
|
||||||
|
|
||||||
|
const urlServer = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
const findChatwoot = await this.waInstances[key].findChatwoot();
|
||||||
|
|
||||||
|
if (findChatwoot.enabled) {
|
||||||
|
chatwoot = {
|
||||||
|
...findChatwoot,
|
||||||
|
webhook_url: `${urlServer}/chatwoot/webhook/${key}`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
if (value.connectionStatus.state === 'open') {
|
if (value.connectionStatus.state === 'open') {
|
||||||
this.logger.verbose('instance: ' + key + ' - connectionStatus: open');
|
this.logger.verbose('instance: ' + key + ' - connectionStatus: open');
|
||||||
let apikey: string;
|
let apikey: string;
|
||||||
@ -99,8 +113,10 @@ export class WAMonitoringService {
|
|||||||
owner: value.wuid,
|
owner: value.wuid,
|
||||||
profileName: (await value.getProfileName()) || 'not loaded',
|
profileName: (await value.getProfileName()) || 'not loaded',
|
||||||
profilePictureUrl: value.profilePictureUrl,
|
profilePictureUrl: value.profilePictureUrl,
|
||||||
status: (await value.getProfileStatus()) || '',
|
profileStatus: (await value.getProfileStatus()) || '',
|
||||||
|
status: value.connectionStatus.state,
|
||||||
apikey,
|
apikey,
|
||||||
|
chatwoot,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
@ -113,7 +129,8 @@ export class WAMonitoringService {
|
|||||||
owner: value.wuid,
|
owner: value.wuid,
|
||||||
profileName: (await value.getProfileName()) || 'not loaded',
|
profileName: (await value.getProfileName()) || 'not loaded',
|
||||||
profilePictureUrl: value.profilePictureUrl,
|
profilePictureUrl: value.profilePictureUrl,
|
||||||
status: (await value.getProfileStatus()) || '',
|
profileStatus: (await value.getProfileStatus()) || '',
|
||||||
|
status: value.connectionStatus.state,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -134,6 +151,7 @@ export class WAMonitoringService {
|
|||||||
instanceName: key,
|
instanceName: key,
|
||||||
status: value.connectionStatus.state,
|
status: value.connectionStatus.state,
|
||||||
apikey,
|
apikey,
|
||||||
|
chatwoot,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
@ -232,6 +250,7 @@ export class WAMonitoringService {
|
|||||||
|
|
||||||
execSync(`rm -rf ${join(STORE_DIR, 'auth', 'apikey', instanceName + '.json')}`);
|
execSync(`rm -rf ${join(STORE_DIR, 'auth', 'apikey', instanceName + '.json')}`);
|
||||||
execSync(`rm -rf ${join(STORE_DIR, 'webhook', instanceName + '.json')}`);
|
execSync(`rm -rf ${join(STORE_DIR, 'webhook', instanceName + '.json')}`);
|
||||||
|
execSync(`rm -rf ${join(STORE_DIR, 'chatwoot', instanceName + '*')}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,7 +30,6 @@ import makeWASocket, {
|
|||||||
WAMessageUpdate,
|
WAMessageUpdate,
|
||||||
WASocket,
|
WASocket,
|
||||||
getAggregateVotesInPollMessage,
|
getAggregateVotesInPollMessage,
|
||||||
Browsers,
|
|
||||||
} from '@whiskeysockets/baileys';
|
} from '@whiskeysockets/baileys';
|
||||||
import {
|
import {
|
||||||
Auth,
|
Auth,
|
||||||
@ -38,6 +37,7 @@ import {
|
|||||||
ConfigService,
|
ConfigService,
|
||||||
ConfigSessionPhone,
|
ConfigSessionPhone,
|
||||||
Database,
|
Database,
|
||||||
|
HttpServer,
|
||||||
QrCode,
|
QrCode,
|
||||||
Redis,
|
Redis,
|
||||||
Webhook,
|
Webhook,
|
||||||
@ -109,11 +109,13 @@ import {
|
|||||||
GroupSubjectDto,
|
GroupSubjectDto,
|
||||||
GroupDescriptionDto,
|
GroupDescriptionDto,
|
||||||
GroupSendInvite,
|
GroupSendInvite,
|
||||||
|
GetParticipant,
|
||||||
} from '../dto/group.dto';
|
} from '../dto/group.dto';
|
||||||
import { MessageUpQuery } from '../repository/messageUp.repository';
|
import { MessageUpQuery } from '../repository/messageUp.repository';
|
||||||
import { useMultiFileAuthStateDb } from '../../utils/use-multi-file-auth-state-db';
|
import { useMultiFileAuthStateDb } from '../../utils/use-multi-file-auth-state-db';
|
||||||
import Long from 'long';
|
import Long from 'long';
|
||||||
import { WebhookRaw } from '../models/webhook.model';
|
import { WebhookRaw } from '../models/webhook.model';
|
||||||
|
import { ChatwootRaw } from '../models/chatwoot.model';
|
||||||
import { dbserver } from '../../db/db.connect';
|
import { dbserver } from '../../db/db.connect';
|
||||||
import NodeCache from 'node-cache';
|
import NodeCache from 'node-cache';
|
||||||
import { useMultiFileAuthStateRedisDb } from '../../utils/use-multi-file-auth-state-redis-db';
|
import { useMultiFileAuthStateRedisDb } from '../../utils/use-multi-file-auth-state-redis-db';
|
||||||
@ -121,6 +123,8 @@ import sharp from 'sharp';
|
|||||||
import { RedisCache } from '../../db/redis.client';
|
import { RedisCache } from '../../db/redis.client';
|
||||||
import { Log } from '../../config/env.config';
|
import { Log } from '../../config/env.config';
|
||||||
import ProxyAgent from 'proxy-agent';
|
import ProxyAgent from 'proxy-agent';
|
||||||
|
import { ChatwootService } from './chatwoot.service';
|
||||||
|
import { waMonitor } from '../whatsapp.module';
|
||||||
|
|
||||||
export class WAStartupService {
|
export class WAStartupService {
|
||||||
constructor(
|
constructor(
|
||||||
@ -138,13 +142,16 @@ export class WAStartupService {
|
|||||||
private readonly instance: wa.Instance = {};
|
private readonly instance: wa.Instance = {};
|
||||||
public client: WASocket;
|
public client: WASocket;
|
||||||
private readonly localWebhook: wa.LocalWebHook = {};
|
private readonly localWebhook: wa.LocalWebHook = {};
|
||||||
|
private readonly localChatwoot: wa.LocalChatwoot = {};
|
||||||
private stateConnection: wa.StateConnection = { state: 'close' };
|
private stateConnection: wa.StateConnection = { state: 'close' };
|
||||||
private readonly storePath = join(ROOT_DIR, 'store');
|
public readonly storePath = join(ROOT_DIR, 'store');
|
||||||
private readonly msgRetryCounterCache: CacheStore = new NodeCache();
|
private readonly msgRetryCounterCache: CacheStore = new NodeCache();
|
||||||
private readonly userDevicesCache: CacheStore = new NodeCache();
|
private readonly userDevicesCache: CacheStore = new NodeCache();
|
||||||
private endSession = false;
|
private endSession = false;
|
||||||
private logBaileys = this.configService.get<Log>('LOG').BAILEYS;
|
private logBaileys = this.configService.get<Log>('LOG').BAILEYS;
|
||||||
|
|
||||||
|
private chatwootService = new ChatwootService(waMonitor);
|
||||||
|
|
||||||
public set instanceName(name: string) {
|
public set instanceName(name: string) {
|
||||||
this.logger.verbose(`Initializing instance '${name}'`);
|
this.logger.verbose(`Initializing instance '${name}'`);
|
||||||
if (!name) {
|
if (!name) {
|
||||||
@ -159,6 +166,17 @@ export class WAStartupService {
|
|||||||
instance: this.instance.name,
|
instance: this.instance.name,
|
||||||
status: 'created',
|
status: 'created',
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
this.chatwootService.eventWhatsapp(
|
||||||
|
Events.STATUS_INSTANCE,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
{
|
||||||
|
instance: this.instance.name,
|
||||||
|
status: 'created',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public get instanceName() {
|
public get instanceName() {
|
||||||
@ -268,8 +286,64 @@ export class WAStartupService {
|
|||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async loadChatwoot() {
|
||||||
|
this.logger.verbose('Loading chatwoot');
|
||||||
|
const data = await this.repository.chatwoot.find(this.instanceName);
|
||||||
|
this.localChatwoot.enabled = data?.enabled;
|
||||||
|
this.logger.verbose(`Chatwoot enabled: ${this.localChatwoot.enabled}`);
|
||||||
|
|
||||||
|
this.localChatwoot.account_id = data?.account_id;
|
||||||
|
this.logger.verbose(`Chatwoot account id: ${this.localChatwoot.account_id}`);
|
||||||
|
|
||||||
|
this.localChatwoot.token = data?.token;
|
||||||
|
this.logger.verbose(`Chatwoot token: ${this.localChatwoot.token}`);
|
||||||
|
|
||||||
|
this.localChatwoot.url = data?.url;
|
||||||
|
this.logger.verbose(`Chatwoot url: ${this.localChatwoot.url}`);
|
||||||
|
|
||||||
|
this.localChatwoot.name_inbox = data?.name_inbox;
|
||||||
|
this.logger.verbose(`Chatwoot inbox name: ${this.localChatwoot.name_inbox}`);
|
||||||
|
|
||||||
|
this.localChatwoot.sign_msg = data?.sign_msg;
|
||||||
|
this.logger.verbose(`Chatwoot sign msg: ${this.localChatwoot.sign_msg}`);
|
||||||
|
|
||||||
|
this.logger.verbose('Chatwoot loaded');
|
||||||
|
}
|
||||||
|
|
||||||
|
public async setChatwoot(data: ChatwootRaw) {
|
||||||
|
this.logger.verbose('Setting chatwoot');
|
||||||
|
await this.repository.chatwoot.create(data, this.instanceName);
|
||||||
|
this.logger.verbose(`Chatwoot account id: ${data.account_id}`);
|
||||||
|
this.logger.verbose(`Chatwoot token: ${data.token}`);
|
||||||
|
this.logger.verbose(`Chatwoot url: ${data.url}`);
|
||||||
|
this.logger.verbose(`Chatwoot inbox name: ${data.name_inbox}`);
|
||||||
|
this.logger.verbose(`Chatwoot sign msg: ${data.sign_msg}`);
|
||||||
|
|
||||||
|
Object.assign(this.localChatwoot, data);
|
||||||
|
this.logger.verbose('Chatwoot set');
|
||||||
|
}
|
||||||
|
|
||||||
|
public async findChatwoot() {
|
||||||
|
this.logger.verbose('Finding chatwoot');
|
||||||
|
const data = await this.repository.chatwoot.find(this.instanceName);
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
this.logger.verbose('Chatwoot not found');
|
||||||
|
throw new NotFoundException('Chatwoot not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose(`Chatwoot account id: ${data.account_id}`);
|
||||||
|
this.logger.verbose(`Chatwoot token: ${data.token}`);
|
||||||
|
this.logger.verbose(`Chatwoot url: ${data.url}`);
|
||||||
|
this.logger.verbose(`Chatwoot inbox name: ${data.name_inbox}`);
|
||||||
|
this.logger.verbose(`Chatwoot sign msg: ${data.sign_msg}`);
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
public async sendDataWebhook<T = any>(event: Events, data: T, local = true) {
|
public async sendDataWebhook<T = any>(event: Events, data: T, local = true) {
|
||||||
const webhookGlobal = this.configService.get<Webhook>('WEBHOOK');
|
const webhookGlobal = this.configService.get<Webhook>('WEBHOOK');
|
||||||
|
const urlServer = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
const webhookLocal = this.localWebhook.events;
|
const webhookLocal = this.localWebhook.events;
|
||||||
const we = event.replace(/[\.-]/gm, '_').toUpperCase();
|
const we = event.replace(/[\.-]/gm, '_').toUpperCase();
|
||||||
const transformedWe = we.replace(/_/gm, '-').toLowerCase();
|
const transformedWe = we.replace(/_/gm, '-').toLowerCase();
|
||||||
@ -294,6 +368,7 @@ export class WAStartupService {
|
|||||||
instance: this.instance.name,
|
instance: this.instance.name,
|
||||||
data,
|
data,
|
||||||
destination: this.localWebhook.url,
|
destination: this.localWebhook.url,
|
||||||
|
urlServer,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -305,6 +380,7 @@ export class WAStartupService {
|
|||||||
instance: this.instance.name,
|
instance: this.instance.name,
|
||||||
data,
|
data,
|
||||||
destination: this.localWebhook.url,
|
destination: this.localWebhook.url,
|
||||||
|
urlServer,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -352,6 +428,7 @@ export class WAStartupService {
|
|||||||
instance: this.instance.name,
|
instance: this.instance.name,
|
||||||
data,
|
data,
|
||||||
destination: localUrl,
|
destination: localUrl,
|
||||||
|
urlServer,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -363,6 +440,7 @@ export class WAStartupService {
|
|||||||
instance: this.instance.name,
|
instance: this.instance.name,
|
||||||
data,
|
data,
|
||||||
destination: localUrl,
|
destination: localUrl,
|
||||||
|
urlServer,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -399,6 +477,17 @@ export class WAStartupService {
|
|||||||
statusCode: DisconnectReason.badSession,
|
statusCode: DisconnectReason.badSession,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
this.chatwootService.eventWhatsapp(
|
||||||
|
Events.QRCODE_UPDATED,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
{
|
||||||
|
message: 'QR code limit reached, please login again',
|
||||||
|
statusCode: DisconnectReason.badSession,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('Sending data to webhook in event CONNECTION_UPDATE');
|
this.logger.verbose('Sending data to webhook in event CONNECTION_UPDATE');
|
||||||
this.sendDataWebhook(Events.CONNECTION_UPDATE, {
|
this.sendDataWebhook(Events.CONNECTION_UPDATE, {
|
||||||
instance: this.instance.name,
|
instance: this.instance.name,
|
||||||
@ -412,6 +501,17 @@ export class WAStartupService {
|
|||||||
status: 'removed',
|
status: 'removed',
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
this.chatwootService.eventWhatsapp(
|
||||||
|
Events.STATUS_INSTANCE,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
{
|
||||||
|
instance: this.instance.name,
|
||||||
|
status: 'removed',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('endSession defined as true');
|
this.logger.verbose('endSession defined as true');
|
||||||
this.endSession = true;
|
this.endSession = true;
|
||||||
|
|
||||||
@ -442,6 +542,16 @@ export class WAStartupService {
|
|||||||
this.sendDataWebhook(Events.QRCODE_UPDATED, {
|
this.sendDataWebhook(Events.QRCODE_UPDATED, {
|
||||||
qrcode: { instance: this.instance.name, code: qr, base64 },
|
qrcode: { instance: this.instance.name, code: qr, base64 },
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
this.chatwootService.eventWhatsapp(
|
||||||
|
Events.QRCODE_UPDATED,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
{
|
||||||
|
qrcode: { instance: this.instance.name, code: qr, base64 },
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
this.logger.verbose('Generating QR code in terminal');
|
this.logger.verbose('Generating QR code in terminal');
|
||||||
@ -482,6 +592,17 @@ export class WAStartupService {
|
|||||||
status: 'removed',
|
status: 'removed',
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
this.chatwootService.eventWhatsapp(
|
||||||
|
Events.STATUS_INSTANCE,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
{
|
||||||
|
instance: this.instance.name,
|
||||||
|
status: 'removed',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('Emittin event logout.instance');
|
this.logger.verbose('Emittin event logout.instance');
|
||||||
this.eventEmitter.emit('logout.instance', this.instance.name, 'inner');
|
this.eventEmitter.emit('logout.instance', this.instance.name, 'inner');
|
||||||
this.client?.ws?.close();
|
this.client?.ws?.close();
|
||||||
@ -596,14 +717,14 @@ export class WAStartupService {
|
|||||||
this.logger.verbose('Connecting to whatsapp');
|
this.logger.verbose('Connecting to whatsapp');
|
||||||
try {
|
try {
|
||||||
this.loadWebhook();
|
this.loadWebhook();
|
||||||
|
this.loadChatwoot();
|
||||||
|
|
||||||
this.instance.authState = await this.defineAuthState();
|
this.instance.authState = await this.defineAuthState();
|
||||||
|
|
||||||
const { version } = await fetchLatestBaileysVersion();
|
const { version } = await fetchLatestBaileysVersion();
|
||||||
this.logger.verbose('Baileys version: ' + version);
|
this.logger.verbose('Baileys version: ' + version);
|
||||||
const session = this.configService.get<ConfigSessionPhone>('CONFIG_SESSION_PHONE');
|
const session = this.configService.get<ConfigSessionPhone>('CONFIG_SESSION_PHONE');
|
||||||
// const browser: WABrowserDescription = [session.CLIENT, session.NAME, release()];
|
const browser: WABrowserDescription = [session.CLIENT, session.NAME, release()];
|
||||||
const browser: WABrowserDescription = Browsers.appropriate(session.CLIENT);
|
|
||||||
this.logger.verbose('Browser: ' + JSON.stringify(browser));
|
this.logger.verbose('Browser: ' + JSON.stringify(browser));
|
||||||
|
|
||||||
const socketConfig: UserFacingSocketConfig = {
|
const socketConfig: UserFacingSocketConfig = {
|
||||||
@ -787,6 +908,14 @@ export class WAStartupService {
|
|||||||
this.logger.verbose('Sending data to webhook in event CONTACTS_UPDATE');
|
this.logger.verbose('Sending data to webhook in event CONTACTS_UPDATE');
|
||||||
await this.sendDataWebhook(Events.CONTACTS_UPDATE, contactsRaw);
|
await this.sendDataWebhook(Events.CONTACTS_UPDATE, contactsRaw);
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
await this.chatwootService.eventWhatsapp(
|
||||||
|
Events.CONTACTS_UPDATE,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
contactsRaw,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('Updating contacts in database');
|
this.logger.verbose('Updating contacts in database');
|
||||||
await this.repository.contact.update(
|
await this.repository.contact.update(
|
||||||
contactsRaw,
|
contactsRaw,
|
||||||
@ -910,6 +1039,14 @@ export class WAStartupService {
|
|||||||
this.logger.verbose('Sending data to webhook in event MESSAGES_UPSERT');
|
this.logger.verbose('Sending data to webhook in event MESSAGES_UPSERT');
|
||||||
await this.sendDataWebhook(Events.MESSAGES_UPSERT, messageRaw);
|
await this.sendDataWebhook(Events.MESSAGES_UPSERT, messageRaw);
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
await this.chatwootService.eventWhatsapp(
|
||||||
|
Events.MESSAGES_UPSERT,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
messageRaw,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('Inserting message in database');
|
this.logger.verbose('Inserting message in database');
|
||||||
await this.repository.message.insert(
|
await this.repository.message.insert(
|
||||||
[messageRaw],
|
[messageRaw],
|
||||||
@ -948,6 +1085,14 @@ export class WAStartupService {
|
|||||||
this.logger.verbose('Sending data to webhook in event CONTACTS_UPDATE');
|
this.logger.verbose('Sending data to webhook in event CONTACTS_UPDATE');
|
||||||
await this.sendDataWebhook(Events.CONTACTS_UPDATE, contactRaw);
|
await this.sendDataWebhook(Events.CONTACTS_UPDATE, contactRaw);
|
||||||
|
|
||||||
|
if (this.localChatwoot.enabled) {
|
||||||
|
await this.chatwootService.eventWhatsapp(
|
||||||
|
Events.CONTACTS_UPDATE,
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
contactRaw,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('Updating contact in database');
|
this.logger.verbose('Updating contact in database');
|
||||||
await this.repository.contact.update(
|
await this.repository.contact.update(
|
||||||
[contactRaw],
|
[contactRaw],
|
||||||
@ -1160,6 +1305,8 @@ export class WAStartupService {
|
|||||||
}
|
}
|
||||||
return match[1] === '52' ? '52' + match[3] : '54' + match[3];
|
return match[1] === '52' ? '52' + match[3] : '54' + match[3];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return jid;
|
||||||
}
|
}
|
||||||
return jid;
|
return jid;
|
||||||
}
|
}
|
||||||
@ -1177,6 +1324,7 @@ export class WAStartupService {
|
|||||||
}
|
}
|
||||||
return match[1] + match[2] + match[3];
|
return match[1] + match[2] + match[3];
|
||||||
}
|
}
|
||||||
|
return jid;
|
||||||
} else {
|
} else {
|
||||||
return jid;
|
return jid;
|
||||||
}
|
}
|
||||||
@ -1203,6 +1351,7 @@ export class WAStartupService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const formattedMXARNumber = this.formatMXOrARNumber(number);
|
const formattedMXARNumber = this.formatMXOrARNumber(number);
|
||||||
|
|
||||||
if (formattedMXARNumber !== number) {
|
if (formattedMXARNumber !== number) {
|
||||||
this.logger.verbose(
|
this.logger.verbose(
|
||||||
'Jid created is whatsapp in format MXAR: ' +
|
'Jid created is whatsapp in format MXAR: ' +
|
||||||
@ -1247,22 +1396,15 @@ export class WAStartupService {
|
|||||||
this.logger.verbose('Sending message with typing');
|
this.logger.verbose('Sending message with typing');
|
||||||
|
|
||||||
const jid = this.createJid(number);
|
const jid = this.createJid(number);
|
||||||
const isWA = (await this.whatsappNumber({ numbers: [jid] }))[0];
|
const numberWA = await this.whatsappNumber({ numbers: [jid] });
|
||||||
|
const isWA = numberWA[0];
|
||||||
|
|
||||||
if (!isWA.exists && !isJidGroup(isWA.jid) && !isWA.jid.includes('@broadcast')) {
|
if (!isWA.exists && !isJidGroup(isWA.jid) && !isWA.jid.includes('@broadcast')) {
|
||||||
throw new BadRequestException(isWA);
|
throw new BadRequestException(isWA);
|
||||||
}
|
}
|
||||||
|
|
||||||
const sender = isJidGroup(jid) ? jid : isWA.jid;
|
const sender = isJidGroup(jid) ? jid : isWA.jid;
|
||||||
|
|
||||||
if (isJidGroup(sender)) {
|
|
||||||
try {
|
|
||||||
this.logger.verbose('Getting group metadata');
|
|
||||||
await this.client.groupMetadata(sender);
|
|
||||||
} catch (error) {
|
|
||||||
throw new NotFoundException('Group not found');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (options?.delay) {
|
if (options?.delay) {
|
||||||
this.logger.verbose('Delaying message');
|
this.logger.verbose('Delaying message');
|
||||||
@ -1290,29 +1432,38 @@ export class WAStartupService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let mentions: string[];
|
let mentions: string[];
|
||||||
|
if (isJidGroup(sender)) {
|
||||||
|
try {
|
||||||
|
if (options?.mentions) {
|
||||||
|
this.logger.verbose('Mentions defined');
|
||||||
|
|
||||||
if (options?.mentions) {
|
if (
|
||||||
this.logger.verbose('Mentions defined');
|
!Array.isArray(options.mentions.mentioned) &&
|
||||||
|
!options.mentions.everyOne
|
||||||
if (!Array.isArray(options.mentions.mentioned) && !options.mentions.everyOne) {
|
) {
|
||||||
throw new BadRequestException('Mentions must be an array');
|
throw new BadRequestException('Mentions must be an array');
|
||||||
}
|
|
||||||
|
|
||||||
if (options.mentions.everyOne) {
|
|
||||||
this.logger.verbose('Mentions everyone');
|
|
||||||
|
|
||||||
const groupMetadata = await this.client.groupMetadata(sender);
|
|
||||||
mentions = groupMetadata.participants.map((participant) => participant.id);
|
|
||||||
this.logger.verbose('Getting group metadata for mentions');
|
|
||||||
} else {
|
|
||||||
this.logger.verbose('Mentions manually defined');
|
|
||||||
mentions = options.mentions.mentioned.map((mention) => {
|
|
||||||
const jid = this.createJid(mention);
|
|
||||||
if (isJidGroup(jid)) {
|
|
||||||
throw new BadRequestException('Mentions must be a number');
|
|
||||||
}
|
}
|
||||||
return jid;
|
|
||||||
});
|
if (options.mentions.everyOne) {
|
||||||
|
this.logger.verbose('Mentions everyone');
|
||||||
|
|
||||||
|
this.logger.verbose('Getting group metadata');
|
||||||
|
const groupMetadata = await this.client.groupMetadata(sender);
|
||||||
|
mentions = groupMetadata.participants.map((participant) => participant.id);
|
||||||
|
this.logger.verbose('Getting group metadata for mentions');
|
||||||
|
} else {
|
||||||
|
this.logger.verbose('Mentions manually defined');
|
||||||
|
mentions = options.mentions.mentioned.map((mention) => {
|
||||||
|
const jid = this.createJid(mention);
|
||||||
|
if (isJidGroup(jid)) {
|
||||||
|
throw new BadRequestException('Mentions must be a number');
|
||||||
|
}
|
||||||
|
return jid;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
throw new NotFoundException('Group not found');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1357,7 +1508,6 @@ export class WAStartupService {
|
|||||||
|
|
||||||
if (sender.includes('@broadcast')) {
|
if (sender.includes('@broadcast')) {
|
||||||
this.logger.verbose('Sending message');
|
this.logger.verbose('Sending message');
|
||||||
console.log(message['status']);
|
|
||||||
return await this.client.sendMessage(
|
return await this.client.sendMessage(
|
||||||
sender,
|
sender,
|
||||||
message['status'].content as unknown as AnyMessageContent,
|
message['status'].content as unknown as AnyMessageContent,
|
||||||
@ -1392,6 +1542,14 @@ export class WAStartupService {
|
|||||||
this.logger.verbose('Sending data to webhook in event SEND_MESSAGE');
|
this.logger.verbose('Sending data to webhook in event SEND_MESSAGE');
|
||||||
await this.sendDataWebhook(Events.SEND_MESSAGE, messageRaw);
|
await this.sendDataWebhook(Events.SEND_MESSAGE, messageRaw);
|
||||||
|
|
||||||
|
// if (this.localChatwoot.enabled) {
|
||||||
|
// this.chatwootService.eventWhatsapp(
|
||||||
|
// Events.SEND_MESSAGE,
|
||||||
|
// { instanceName: this.instance.name },
|
||||||
|
// messageRaw,
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
|
||||||
this.logger.verbose('Inserting message in database');
|
this.logger.verbose('Inserting message in database');
|
||||||
await this.repository.message.insert(
|
await this.repository.message.insert(
|
||||||
[messageRaw],
|
[messageRaw],
|
||||||
@ -1954,13 +2112,19 @@ export class WAStartupService {
|
|||||||
const jid = this.createJid(number);
|
const jid = this.createJid(number);
|
||||||
if (isJidGroup(jid)) {
|
if (isJidGroup(jid)) {
|
||||||
const group = await this.findGroup({ groupJid: jid }, 'inner');
|
const group = await this.findGroup({ groupJid: jid }, 'inner');
|
||||||
|
|
||||||
|
if (!group) throw new BadRequestException('Group not found');
|
||||||
|
|
||||||
onWhatsapp.push(new OnWhatsAppDto(group.id, !!group?.id, group?.subject));
|
onWhatsapp.push(new OnWhatsAppDto(group.id, !!group?.id, group?.subject));
|
||||||
} else {
|
} else {
|
||||||
try {
|
const verify = await this.client.onWhatsApp(jid);
|
||||||
const result = (await this.client.onWhatsApp(jid))[0];
|
|
||||||
|
const result = verify[0];
|
||||||
|
|
||||||
|
if (!result) {
|
||||||
|
onWhatsapp.push(new OnWhatsAppDto(jid, false));
|
||||||
|
} else {
|
||||||
onWhatsapp.push(new OnWhatsAppDto(result.jid, result.exists));
|
onWhatsapp.push(new OnWhatsAppDto(result.jid, result.exists));
|
||||||
} catch (error) {
|
|
||||||
onWhatsapp.push(new OnWhatsAppDto(number, false));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2414,10 +2578,34 @@ export class WAStartupService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async fetchAllGroups() {
|
public async fetchAllGroups(getParticipants: GetParticipant) {
|
||||||
this.logger.verbose('Fetching all groups');
|
this.logger.verbose('Fetching all groups');
|
||||||
try {
|
try {
|
||||||
return await this.client.groupFetchAllParticipating();
|
const fetch = Object.values(await this.client.groupFetchAllParticipating());
|
||||||
|
|
||||||
|
const groups = fetch.map((group) => {
|
||||||
|
const result = {
|
||||||
|
id: group.id,
|
||||||
|
subject: group.subject,
|
||||||
|
subjectOwner: group.subjectOwner,
|
||||||
|
subjectTime: group.subjectTime,
|
||||||
|
size: group.size,
|
||||||
|
creation: group.creation,
|
||||||
|
owner: group.owner,
|
||||||
|
desc: group.desc,
|
||||||
|
descId: group.descId,
|
||||||
|
restrict: group.restrict,
|
||||||
|
announce: group.announce,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (getParticipants.getParticipants == 'true') {
|
||||||
|
result['participants'] = group.participants;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
});
|
||||||
|
|
||||||
|
return groups;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new NotFoundException('Error fetching group', error.toString());
|
throw new NotFoundException('Error fetching group', error.toString());
|
||||||
}
|
}
|
||||||
|
@ -6,10 +6,10 @@ export enum Events {
|
|||||||
QRCODE_UPDATED = 'qrcode.updated',
|
QRCODE_UPDATED = 'qrcode.updated',
|
||||||
CONNECTION_UPDATE = 'connection.update',
|
CONNECTION_UPDATE = 'connection.update',
|
||||||
STATUS_INSTANCE = 'status.instance',
|
STATUS_INSTANCE = 'status.instance',
|
||||||
SEND_MESSAGE = 'send.message',
|
|
||||||
MESSAGES_SET = 'messages.set',
|
MESSAGES_SET = 'messages.set',
|
||||||
MESSAGES_UPSERT = 'messages.upsert',
|
MESSAGES_UPSERT = 'messages.upsert',
|
||||||
MESSAGES_UPDATE = 'messages.update',
|
MESSAGES_UPDATE = 'messages.update',
|
||||||
|
SEND_MESSAGE = 'send.message',
|
||||||
CONTACTS_SET = 'contacts.set',
|
CONTACTS_SET = 'contacts.set',
|
||||||
CONTACTS_UPSERT = 'contacts.upsert',
|
CONTACTS_UPSERT = 'contacts.upsert',
|
||||||
CONTACTS_UPDATE = 'contacts.update',
|
CONTACTS_UPDATE = 'contacts.update',
|
||||||
@ -41,6 +41,15 @@ export declare namespace wa {
|
|||||||
webhook_by_events?: boolean;
|
webhook_by_events?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type LocalChatwoot = {
|
||||||
|
enabled?: boolean;
|
||||||
|
account_id?: string;
|
||||||
|
token?: string;
|
||||||
|
url?: string;
|
||||||
|
name_inbox?: string;
|
||||||
|
sign_msg?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
export type StateConnection = {
|
export type StateConnection = {
|
||||||
instance?: string;
|
instance?: string;
|
||||||
state?: WAConnectionState | 'refused';
|
state?: WAConnectionState | 'refused';
|
||||||
|
@ -14,6 +14,8 @@ import { GroupController } from './controllers/group.controller';
|
|||||||
import { ViewsController } from './controllers/views.controller';
|
import { ViewsController } from './controllers/views.controller';
|
||||||
import { WebhookService } from './services/webhook.service';
|
import { WebhookService } from './services/webhook.service';
|
||||||
import { WebhookController } from './controllers/webhook.controller';
|
import { WebhookController } from './controllers/webhook.controller';
|
||||||
|
import { ChatwootService } from './services/chatwoot.service';
|
||||||
|
import { ChatwootController } from './controllers/chatwoot.controller';
|
||||||
import { RepositoryBroker } from './repository/repository.manager';
|
import { RepositoryBroker } from './repository/repository.manager';
|
||||||
import {
|
import {
|
||||||
AuthModel,
|
AuthModel,
|
||||||
@ -21,10 +23,12 @@ import {
|
|||||||
ContactModel,
|
ContactModel,
|
||||||
MessageModel,
|
MessageModel,
|
||||||
MessageUpModel,
|
MessageUpModel,
|
||||||
|
ChatwootModel,
|
||||||
|
WebhookModel,
|
||||||
} from './models';
|
} from './models';
|
||||||
import { dbserver } from '../db/db.connect';
|
import { dbserver } from '../db/db.connect';
|
||||||
import { WebhookRepository } from './repository/webhook.repository';
|
import { WebhookRepository } from './repository/webhook.repository';
|
||||||
import { WebhookModel } from './models/webhook.model';
|
import { ChatwootRepository } from './repository/chatwoot.repository';
|
||||||
import { AuthRepository } from './repository/auth.repository';
|
import { AuthRepository } from './repository/auth.repository';
|
||||||
import { WAStartupService } from './services/whatsapp.service';
|
import { WAStartupService } from './services/whatsapp.service';
|
||||||
import { delay } from '@whiskeysockets/baileys';
|
import { delay } from '@whiskeysockets/baileys';
|
||||||
@ -38,6 +42,7 @@ const chatRepository = new ChatRepository(ChatModel, configService);
|
|||||||
const contactRepository = new ContactRepository(ContactModel, configService);
|
const contactRepository = new ContactRepository(ContactModel, configService);
|
||||||
const messageUpdateRepository = new MessageUpRepository(MessageUpModel, configService);
|
const messageUpdateRepository = new MessageUpRepository(MessageUpModel, configService);
|
||||||
const webhookRepository = new WebhookRepository(WebhookModel, configService);
|
const webhookRepository = new WebhookRepository(WebhookModel, configService);
|
||||||
|
const chatwootRepository = new ChatwootRepository(ChatwootModel, configService);
|
||||||
const authRepository = new AuthRepository(AuthModel, configService);
|
const authRepository = new AuthRepository(AuthModel, configService);
|
||||||
|
|
||||||
export const repository = new RepositoryBroker(
|
export const repository = new RepositoryBroker(
|
||||||
@ -46,6 +51,7 @@ export const repository = new RepositoryBroker(
|
|||||||
contactRepository,
|
contactRepository,
|
||||||
messageUpdateRepository,
|
messageUpdateRepository,
|
||||||
webhookRepository,
|
webhookRepository,
|
||||||
|
chatwootRepository,
|
||||||
authRepository,
|
authRepository,
|
||||||
configService,
|
configService,
|
||||||
dbserver?.getClient(),
|
dbserver?.getClient(),
|
||||||
@ -66,6 +72,10 @@ const webhookService = new WebhookService(waMonitor);
|
|||||||
|
|
||||||
export const webhookController = new WebhookController(webhookService);
|
export const webhookController = new WebhookController(webhookService);
|
||||||
|
|
||||||
|
const chatwootService = new ChatwootService(waMonitor);
|
||||||
|
|
||||||
|
export const chatwootController = new ChatwootController(chatwootService, configService);
|
||||||
|
|
||||||
export const instanceController = new InstanceController(
|
export const instanceController = new InstanceController(
|
||||||
waMonitor,
|
waMonitor,
|
||||||
configService,
|
configService,
|
||||||
@ -73,6 +83,7 @@ export const instanceController = new InstanceController(
|
|||||||
eventEmitter,
|
eventEmitter,
|
||||||
authService,
|
authService,
|
||||||
webhookService,
|
webhookService,
|
||||||
|
chatwootService,
|
||||||
cache,
|
cache,
|
||||||
);
|
);
|
||||||
export const viewsController = new ViewsController(waMonitor, configService);
|
export const viewsController = new ViewsController(waMonitor, configService);
|
||||||
@ -105,6 +116,17 @@ export async function initInstance() {
|
|||||||
configService.get<Auth>('AUTHENTICATION').INSTANCE.WEBHOOK_URL;
|
configService.get<Auth>('AUTHENTICATION').INSTANCE.WEBHOOK_URL;
|
||||||
logger.verbose('Instance webhook: ' + instanceWebhook);
|
logger.verbose('Instance webhook: ' + instanceWebhook);
|
||||||
|
|
||||||
|
const chatwootAccountId =
|
||||||
|
configService.get<Auth>('AUTHENTICATION').INSTANCE.CHATWOOT_ACCOUNT_ID;
|
||||||
|
logger.verbose('Chatwoot account id: ' + chatwootAccountId);
|
||||||
|
|
||||||
|
const chatwootToken =
|
||||||
|
configService.get<Auth>('AUTHENTICATION').INSTANCE.CHATWOOT_TOKEN;
|
||||||
|
logger.verbose('Chatwoot token: ' + chatwootToken);
|
||||||
|
|
||||||
|
const chatwootUrl = configService.get<Auth>('AUTHENTICATION').INSTANCE.CHATWOOT_URL;
|
||||||
|
logger.verbose('Chatwoot url: ' + chatwootUrl);
|
||||||
|
|
||||||
instance.instanceName = instanceName;
|
instance.instanceName = instanceName;
|
||||||
|
|
||||||
waMonitor.waInstances[instance.instanceName] = instance;
|
waMonitor.waInstances[instance.instanceName] = instance;
|
||||||
@ -126,6 +148,22 @@ export async function initInstance() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (chatwootUrl && chatwootToken && chatwootAccountId) {
|
||||||
|
logger.verbose('Creating chatwoot for instance: ' + instanceName);
|
||||||
|
try {
|
||||||
|
chatwootService.create(instance, {
|
||||||
|
enabled: true,
|
||||||
|
url: chatwootUrl,
|
||||||
|
token: chatwootToken,
|
||||||
|
account_id: chatwootAccountId,
|
||||||
|
sign_msg: false,
|
||||||
|
});
|
||||||
|
logger.verbose('Chatwoot created');
|
||||||
|
} catch (error) {
|
||||||
|
logger.log(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const state = instance.connectionStatus?.state;
|
const state = instance.connectionStatus?.state;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user