mirror of
https://github.com/EvolutionAPI/evolution-api.git
synced 2025-09-02 04:28:24 -06:00
Added Typebot integration
This commit is contained in:
parent
84386847e2
commit
3b3118d764
@ -974,3 +974,16 @@ export const rabbitmqSchema: JSONSchema7 = {
|
|||||||
required: ['enabled'],
|
required: ['enabled'],
|
||||||
...isNotEmpty('enabled'),
|
...isNotEmpty('enabled'),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const typebotSchema: JSONSchema7 = {
|
||||||
|
$id: v4(),
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
enabled: { type: 'boolean', enum: [true, false] },
|
||||||
|
url: { type: 'string' },
|
||||||
|
typebot: { type: 'string' },
|
||||||
|
expire: { type: 'integer' },
|
||||||
|
},
|
||||||
|
required: ['enabled', 'url', 'typebot', 'expire'],
|
||||||
|
...isNotEmpty('enabled', 'url', 'typebot', 'expire'),
|
||||||
|
};
|
||||||
|
36
src/whatsapp/controllers/typebot.controller.ts
Normal file
36
src/whatsapp/controllers/typebot.controller.ts
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
import { InstanceDto } from '../dto/instance.dto';
|
||||||
|
import { TypebotDto } from '../dto/typebot.dto';
|
||||||
|
import { TypebotService } from '../services/typebot.service';
|
||||||
|
|
||||||
|
const logger = new Logger('TypebotController');
|
||||||
|
|
||||||
|
export class TypebotController {
|
||||||
|
constructor(private readonly typebotService: TypebotService) {}
|
||||||
|
|
||||||
|
public async createTypebot(instance: InstanceDto, data: TypebotDto) {
|
||||||
|
logger.verbose('requested createTypebot from ' + instance.instanceName + ' instance');
|
||||||
|
|
||||||
|
if (!data.enabled) {
|
||||||
|
logger.verbose('typebot disabled');
|
||||||
|
data.url = '';
|
||||||
|
data.typebot = '';
|
||||||
|
data.expire = 0;
|
||||||
|
data.sessions = [];
|
||||||
|
} else {
|
||||||
|
const saveData = await this.typebotService.find(instance);
|
||||||
|
|
||||||
|
if (saveData.enabled) {
|
||||||
|
logger.verbose('typebot enabled');
|
||||||
|
data.sessions = saveData.sessions;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.typebotService.create(instance, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async findTypebot(instance: InstanceDto) {
|
||||||
|
logger.verbose('requested findTypebot from ' + instance.instanceName + ' instance');
|
||||||
|
return this.typebotService.find(instance);
|
||||||
|
}
|
||||||
|
}
|
14
src/whatsapp/dto/typebot.dto.ts
Normal file
14
src/whatsapp/dto/typebot.dto.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
export class Session {
|
||||||
|
remoteJid?: string;
|
||||||
|
sessionId?: string;
|
||||||
|
createdAt?: number;
|
||||||
|
updateAt?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class TypebotDto {
|
||||||
|
enabled?: boolean;
|
||||||
|
url: string;
|
||||||
|
typebot?: string;
|
||||||
|
expire?: number;
|
||||||
|
sessions?: Session[];
|
||||||
|
}
|
@ -5,5 +5,6 @@ export * from './contact.model';
|
|||||||
export * from './message.model';
|
export * from './message.model';
|
||||||
export * from './rabbitmq.model';
|
export * from './rabbitmq.model';
|
||||||
export * from './settings.model';
|
export * from './settings.model';
|
||||||
|
export * from './typebot.model';
|
||||||
export * from './webhook.model';
|
export * from './webhook.model';
|
||||||
export * from './websocket.model';
|
export * from './websocket.model';
|
||||||
|
38
src/whatsapp/models/typebot.model.ts
Normal file
38
src/whatsapp/models/typebot.model.ts
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
import { Schema } from 'mongoose';
|
||||||
|
|
||||||
|
import { dbserver } from '../../libs/db.connect';
|
||||||
|
|
||||||
|
class Session {
|
||||||
|
remoteJid?: string;
|
||||||
|
sessionId?: string;
|
||||||
|
createdAt?: number;
|
||||||
|
updateAt?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class TypebotRaw {
|
||||||
|
_id?: string;
|
||||||
|
enabled?: boolean;
|
||||||
|
url: string;
|
||||||
|
typebot?: string;
|
||||||
|
expire?: number;
|
||||||
|
sessions?: Session[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const typebotSchema = new Schema<TypebotRaw>({
|
||||||
|
_id: { type: String, _id: true },
|
||||||
|
enabled: { type: Boolean, required: true },
|
||||||
|
url: { type: String, required: true },
|
||||||
|
typebot: { type: String, required: true },
|
||||||
|
expire: { type: Number, required: true },
|
||||||
|
sessions: [
|
||||||
|
{
|
||||||
|
remoteJid: { type: String, required: true },
|
||||||
|
sessionId: { type: String, required: true },
|
||||||
|
createdAt: { type: Number, required: true },
|
||||||
|
updateAt: { type: Number, required: true },
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
export const TypebotModel = dbserver?.model(TypebotRaw.name, typebotSchema, 'typebot');
|
||||||
|
export type ITypebotModel = typeof TypebotModel;
|
@ -12,6 +12,7 @@ import { MessageRepository } from './message.repository';
|
|||||||
import { MessageUpRepository } from './messageUp.repository';
|
import { MessageUpRepository } from './messageUp.repository';
|
||||||
import { RabbitmqRepository } from './rabbitmq.repository';
|
import { RabbitmqRepository } from './rabbitmq.repository';
|
||||||
import { SettingsRepository } from './settings.repository';
|
import { SettingsRepository } from './settings.repository';
|
||||||
|
import { TypebotRepository } from './typebot.repository';
|
||||||
import { WebhookRepository } from './webhook.repository';
|
import { WebhookRepository } from './webhook.repository';
|
||||||
import { WebsocketRepository } from './websocket.repository';
|
import { WebsocketRepository } from './websocket.repository';
|
||||||
export class RepositoryBroker {
|
export class RepositoryBroker {
|
||||||
@ -25,6 +26,7 @@ export class RepositoryBroker {
|
|||||||
public readonly settings: SettingsRepository,
|
public readonly settings: SettingsRepository,
|
||||||
public readonly websocket: WebsocketRepository,
|
public readonly websocket: WebsocketRepository,
|
||||||
public readonly rabbitmq: RabbitmqRepository,
|
public readonly rabbitmq: RabbitmqRepository,
|
||||||
|
public readonly typebot: TypebotRepository,
|
||||||
public readonly auth: AuthRepository,
|
public readonly auth: AuthRepository,
|
||||||
private configService: ConfigService,
|
private configService: ConfigService,
|
||||||
dbServer?: MongoClient,
|
dbServer?: MongoClient,
|
||||||
@ -57,6 +59,7 @@ export class RepositoryBroker {
|
|||||||
const settingsDir = join(storePath, 'settings');
|
const settingsDir = join(storePath, 'settings');
|
||||||
const websocketDir = join(storePath, 'websocket');
|
const websocketDir = join(storePath, 'websocket');
|
||||||
const rabbitmqDir = join(storePath, 'rabbitmq');
|
const rabbitmqDir = join(storePath, 'rabbitmq');
|
||||||
|
const typebotDir = join(storePath, 'typebot');
|
||||||
const tempDir = join(storePath, 'temp');
|
const tempDir = join(storePath, 'temp');
|
||||||
|
|
||||||
if (!fs.existsSync(authDir)) {
|
if (!fs.existsSync(authDir)) {
|
||||||
@ -99,6 +102,10 @@ export class RepositoryBroker {
|
|||||||
this.logger.verbose('creating rabbitmq dir: ' + rabbitmqDir);
|
this.logger.verbose('creating rabbitmq dir: ' + rabbitmqDir);
|
||||||
fs.mkdirSync(rabbitmqDir, { recursive: true });
|
fs.mkdirSync(rabbitmqDir, { recursive: true });
|
||||||
}
|
}
|
||||||
|
if (!fs.existsSync(typebotDir)) {
|
||||||
|
this.logger.verbose('creating typebot dir: ' + typebotDir);
|
||||||
|
fs.mkdirSync(typebotDir, { recursive: true });
|
||||||
|
}
|
||||||
if (!fs.existsSync(tempDir)) {
|
if (!fs.existsSync(tempDir)) {
|
||||||
this.logger.verbose('creating temp dir: ' + tempDir);
|
this.logger.verbose('creating temp dir: ' + tempDir);
|
||||||
fs.mkdirSync(tempDir, { recursive: true });
|
fs.mkdirSync(tempDir, { recursive: true });
|
||||||
|
68
src/whatsapp/repository/typebot.repository.ts
Normal file
68
src/whatsapp/repository/typebot.repository.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import { readFileSync } from 'fs';
|
||||||
|
import { join } from 'path';
|
||||||
|
|
||||||
|
import { ConfigService } from '../../config/env.config';
|
||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
import { IInsert, Repository } from '../abstract/abstract.repository';
|
||||||
|
import { ITypebotModel, TypebotRaw } from '../models';
|
||||||
|
|
||||||
|
export class TypebotRepository extends Repository {
|
||||||
|
constructor(private readonly typebotModel: ITypebotModel, private readonly configService: ConfigService) {
|
||||||
|
super(configService);
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly logger = new Logger('TypebotRepository');
|
||||||
|
|
||||||
|
public async create(data: TypebotRaw, instance: string): Promise<IInsert> {
|
||||||
|
try {
|
||||||
|
this.logger.verbose('creating typebot');
|
||||||
|
if (this.dbSettings.ENABLED) {
|
||||||
|
this.logger.verbose('saving typebot to db');
|
||||||
|
const insert = await this.typebotModel.replaceOne({ _id: instance }, { ...data }, { upsert: true });
|
||||||
|
|
||||||
|
this.logger.verbose('typebot saved to db: ' + insert.modifiedCount + ' typebot');
|
||||||
|
return { insertCount: insert.modifiedCount };
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('saving typebot to store');
|
||||||
|
|
||||||
|
this.writeStore<TypebotRaw>({
|
||||||
|
path: join(this.storePath, 'typebot'),
|
||||||
|
fileName: instance,
|
||||||
|
data,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.logger.verbose('typebot saved to store in path: ' + join(this.storePath, 'typebot') + '/' + instance);
|
||||||
|
|
||||||
|
this.logger.verbose('typebot created');
|
||||||
|
return { insertCount: 1 };
|
||||||
|
} catch (error) {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async find(instance: string): Promise<TypebotRaw> {
|
||||||
|
try {
|
||||||
|
this.logger.verbose('finding typebot');
|
||||||
|
if (this.dbSettings.ENABLED) {
|
||||||
|
this.logger.verbose('finding typebot in db');
|
||||||
|
return await this.typebotModel.findOne({ _id: instance });
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('finding typebot in store');
|
||||||
|
return JSON.parse(
|
||||||
|
readFileSync(join(this.storePath, 'typebot', instance + '.json'), {
|
||||||
|
encoding: 'utf-8',
|
||||||
|
}),
|
||||||
|
) as TypebotRaw;
|
||||||
|
} catch (error) {
|
||||||
|
return {
|
||||||
|
enabled: false,
|
||||||
|
url: '',
|
||||||
|
typebot: '',
|
||||||
|
expire: 0,
|
||||||
|
sessions: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -11,6 +11,7 @@ import { InstanceRouter } from './instance.router';
|
|||||||
import { RabbitmqRouter } from './rabbitmq.router';
|
import { RabbitmqRouter } from './rabbitmq.router';
|
||||||
import { MessageRouter } from './sendMessage.router';
|
import { MessageRouter } from './sendMessage.router';
|
||||||
import { SettingsRouter } from './settings.router';
|
import { SettingsRouter } from './settings.router';
|
||||||
|
import { TypebotRouter } from './typebot.router';
|
||||||
import { ViewsRouter } from './view.router';
|
import { ViewsRouter } from './view.router';
|
||||||
import { WebhookRouter } from './webhook.router';
|
import { WebhookRouter } from './webhook.router';
|
||||||
import { WebsocketRouter } from './websocket.router';
|
import { WebsocketRouter } from './websocket.router';
|
||||||
@ -48,6 +49,7 @@ router
|
|||||||
.use('/chatwoot', new ChatwootRouter(...guards).router)
|
.use('/chatwoot', new ChatwootRouter(...guards).router)
|
||||||
.use('/settings', new SettingsRouter(...guards).router)
|
.use('/settings', new SettingsRouter(...guards).router)
|
||||||
.use('/websocket', new WebsocketRouter(...guards).router)
|
.use('/websocket', new WebsocketRouter(...guards).router)
|
||||||
.use('/rabbitmq', new RabbitmqRouter(...guards).router);
|
.use('/rabbitmq', new RabbitmqRouter(...guards).router)
|
||||||
|
.use('/typebot', new TypebotRouter(...guards).router);
|
||||||
|
|
||||||
export { HttpStatus, router };
|
export { HttpStatus, router };
|
||||||
|
52
src/whatsapp/routers/typebot.router.ts
Normal file
52
src/whatsapp/routers/typebot.router.ts
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import { RequestHandler, Router } from 'express';
|
||||||
|
|
||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
import { instanceNameSchema, typebotSchema } from '../../validate/validate.schema';
|
||||||
|
import { RouterBroker } from '../abstract/abstract.router';
|
||||||
|
import { InstanceDto } from '../dto/instance.dto';
|
||||||
|
import { TypebotDto } from '../dto/typebot.dto';
|
||||||
|
import { typebotController } from '../whatsapp.module';
|
||||||
|
import { HttpStatus } from './index.router';
|
||||||
|
|
||||||
|
const logger = new Logger('TypebotRouter');
|
||||||
|
|
||||||
|
export class TypebotRouter extends RouterBroker {
|
||||||
|
constructor(...guards: RequestHandler[]) {
|
||||||
|
super();
|
||||||
|
this.router
|
||||||
|
.post(this.routerPath('set'), ...guards, async (req, res) => {
|
||||||
|
logger.verbose('request received in setTypebot');
|
||||||
|
logger.verbose('request body: ');
|
||||||
|
logger.verbose(req.body);
|
||||||
|
|
||||||
|
logger.verbose('request query: ');
|
||||||
|
logger.verbose(req.query);
|
||||||
|
const response = await this.dataValidate<TypebotDto>({
|
||||||
|
request: req,
|
||||||
|
schema: typebotSchema,
|
||||||
|
ClassRef: TypebotDto,
|
||||||
|
execute: (instance, data) => typebotController.createTypebot(instance, data),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(HttpStatus.CREATED).json(response);
|
||||||
|
})
|
||||||
|
.get(this.routerPath('find'), ...guards, async (req, res) => {
|
||||||
|
logger.verbose('request received in findTypebot');
|
||||||
|
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) => typebotController.findTypebot(instance),
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(HttpStatus.OK).json(response);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public readonly router = Router();
|
||||||
|
}
|
336
src/whatsapp/services/typebot.service.ts
Normal file
336
src/whatsapp/services/typebot.service.ts
Normal file
@ -0,0 +1,336 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
import { Logger } from '../../config/logger.config';
|
||||||
|
import { InstanceDto } from '../dto/instance.dto';
|
||||||
|
import { Session, TypebotDto } from '../dto/typebot.dto';
|
||||||
|
import { MessageRaw } from '../models';
|
||||||
|
import { WAMonitoringService } from './monitor.service';
|
||||||
|
|
||||||
|
export class TypebotService {
|
||||||
|
constructor(private readonly waMonitor: WAMonitoringService) {}
|
||||||
|
|
||||||
|
private readonly logger = new Logger(TypebotService.name);
|
||||||
|
|
||||||
|
public create(instance: InstanceDto, data: TypebotDto) {
|
||||||
|
this.logger.verbose('create typebot: ' + instance.instanceName);
|
||||||
|
this.waMonitor.waInstances[instance.instanceName].setTypebot(data);
|
||||||
|
|
||||||
|
return { typebot: { ...instance, typebot: data } };
|
||||||
|
}
|
||||||
|
|
||||||
|
public async find(instance: InstanceDto): Promise<TypebotDto> {
|
||||||
|
try {
|
||||||
|
this.logger.verbose('find typebot: ' + instance.instanceName);
|
||||||
|
const result = await this.waMonitor.waInstances[instance.instanceName].findTypebot();
|
||||||
|
|
||||||
|
if (Object.keys(result).length === 0) {
|
||||||
|
throw new Error('Typebot not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
} catch (error) {
|
||||||
|
return { enabled: false, url: '', typebot: '', expire: 0, sessions: [] };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private getTypeMessage(msg: any) {
|
||||||
|
this.logger.verbose('get type message');
|
||||||
|
|
||||||
|
const types = {
|
||||||
|
conversation: msg.conversation,
|
||||||
|
imageMessage: msg.imageMessage?.caption,
|
||||||
|
videoMessage: msg.videoMessage?.caption,
|
||||||
|
extendedTextMessage: msg.extendedTextMessage?.text,
|
||||||
|
messageContextInfo: msg.messageContextInfo?.stanzaId,
|
||||||
|
stickerMessage: undefined,
|
||||||
|
documentMessage: msg.documentMessage?.caption,
|
||||||
|
documentWithCaptionMessage: msg.documentWithCaptionMessage?.message?.documentMessage?.caption,
|
||||||
|
audioMessage: msg.audioMessage?.caption,
|
||||||
|
contactMessage: msg.contactMessage?.vcard,
|
||||||
|
contactsArrayMessage: msg.contactsArrayMessage,
|
||||||
|
locationMessage: msg.locationMessage,
|
||||||
|
liveLocationMessage: msg.liveLocationMessage,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.logger.verbose('type message: ' + types);
|
||||||
|
|
||||||
|
return types;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getMessageContent(types: any) {
|
||||||
|
this.logger.verbose('get message content');
|
||||||
|
const typeKey = Object.keys(types).find((key) => types[key] !== undefined);
|
||||||
|
|
||||||
|
const result = typeKey ? types[typeKey] : undefined;
|
||||||
|
|
||||||
|
this.logger.verbose('message content: ' + result);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getConversationMessage(msg: any) {
|
||||||
|
this.logger.verbose('get conversation message');
|
||||||
|
|
||||||
|
const types = this.getTypeMessage(msg);
|
||||||
|
|
||||||
|
const messageContent = this.getMessageContent(types);
|
||||||
|
|
||||||
|
this.logger.verbose('conversation message: ' + messageContent);
|
||||||
|
|
||||||
|
return messageContent;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async createNewSession(instance: InstanceDto, data: any) {
|
||||||
|
const id = Math.floor(Math.random() * 10000000000).toString();
|
||||||
|
const reqData = {
|
||||||
|
sessionId: id,
|
||||||
|
startParams: {
|
||||||
|
typebot: data.typebot,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const request = await axios.post(data.url + '/api/v1/sendMessage', reqData);
|
||||||
|
|
||||||
|
if (request.data.sessionId) {
|
||||||
|
data.sessions.push({
|
||||||
|
remoteJid: data.remoteJid,
|
||||||
|
sessionId: `${id}-${request.data.sessionId}`,
|
||||||
|
createdAt: Date.now(),
|
||||||
|
updateAt: Date.now(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const typebotData = {
|
||||||
|
enabled: true,
|
||||||
|
url: data.url,
|
||||||
|
typebot: data.typebot,
|
||||||
|
expire: data.expire,
|
||||||
|
sessions: data.sessions,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.create(instance, typebotData);
|
||||||
|
}
|
||||||
|
|
||||||
|
return request.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async sendWAMessage(instance: InstanceDto, remoteJid: string, messages: any[], closeSession?: boolean) {
|
||||||
|
processMessages(this.waMonitor.waInstances[instance.instanceName], messages)
|
||||||
|
.then(async () => {
|
||||||
|
if (closeSession) {
|
||||||
|
const typebotData = await this.find(instance);
|
||||||
|
|
||||||
|
const session = typebotData.sessions.find((session) => session.remoteJid === remoteJid);
|
||||||
|
|
||||||
|
if (session) {
|
||||||
|
typebotData.sessions.splice(typebotData.sessions.indexOf(session), 1);
|
||||||
|
|
||||||
|
this.create(instance, typebotData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
console.error('Erro ao processar mensagens:', err);
|
||||||
|
});
|
||||||
|
|
||||||
|
async function processMessages(instance, messages) {
|
||||||
|
for (const message of messages) {
|
||||||
|
if (message.type === 'text') {
|
||||||
|
let formattedText = '';
|
||||||
|
|
||||||
|
for (const richText of message.content.richText) {
|
||||||
|
for (const element of richText.children) {
|
||||||
|
let text = '';
|
||||||
|
if (element.text) {
|
||||||
|
text = element.text;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (element.bold) {
|
||||||
|
text = `*${text}*`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (element.italic) {
|
||||||
|
text = `_${text}_`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (element.underline) {
|
||||||
|
text = `~${text}~`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (element.url) {
|
||||||
|
const linkText = element.children[0].text;
|
||||||
|
text = `[${linkText}](${element.url})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
formattedText += text;
|
||||||
|
}
|
||||||
|
formattedText += '\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
formattedText = formattedText.replace(/\n$/, '');
|
||||||
|
|
||||||
|
await instance.textMessage({
|
||||||
|
number: remoteJid.split('@')[0],
|
||||||
|
options: {
|
||||||
|
delay: 1200,
|
||||||
|
presence: 'composing',
|
||||||
|
linkPreview: false,
|
||||||
|
},
|
||||||
|
textMessage: {
|
||||||
|
text: formattedText,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (message.type === 'image') {
|
||||||
|
await instance.mediaMessage({
|
||||||
|
number: remoteJid.split('@')[0],
|
||||||
|
options: {
|
||||||
|
delay: 1200,
|
||||||
|
presence: 'composing',
|
||||||
|
},
|
||||||
|
mediaMessage: {
|
||||||
|
mediatype: 'image',
|
||||||
|
media: message.content.url,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (message.type === 'video') {
|
||||||
|
await instance.mediaMessage({
|
||||||
|
number: remoteJid.split('@')[0],
|
||||||
|
options: {
|
||||||
|
delay: 1200,
|
||||||
|
presence: 'composing',
|
||||||
|
},
|
||||||
|
mediaMessage: {
|
||||||
|
mediatype: 'video',
|
||||||
|
media: message.content.url,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (message.type === 'audio') {
|
||||||
|
await instance.audioWhatsapp({
|
||||||
|
number: remoteJid.split('@')[0],
|
||||||
|
options: {
|
||||||
|
delay: 1200,
|
||||||
|
presence: 'recording',
|
||||||
|
encoding: true,
|
||||||
|
},
|
||||||
|
audioMessage: {
|
||||||
|
audio: message.content.url,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async sendTypebot(instance: InstanceDto, remoteJid: string, msg: MessageRaw) {
|
||||||
|
const url = (await this.find(instance)).url;
|
||||||
|
const typebot = (await this.find(instance)).typebot;
|
||||||
|
const sessions = ((await this.find(instance)).sessions as Session[]) ?? [];
|
||||||
|
const expire = (await this.find(instance)).expire;
|
||||||
|
|
||||||
|
const session = sessions.find((session) => session.remoteJid === remoteJid);
|
||||||
|
|
||||||
|
if (session && expire && expire > 0) {
|
||||||
|
const now = Date.now();
|
||||||
|
|
||||||
|
const diff = now - session.updateAt;
|
||||||
|
|
||||||
|
const diffInMinutes = Math.floor(diff / 1000 / 60);
|
||||||
|
|
||||||
|
if (diffInMinutes > expire) {
|
||||||
|
sessions.splice(sessions.indexOf(session), 1);
|
||||||
|
|
||||||
|
const data = await this.createNewSession(instance, {
|
||||||
|
url: url,
|
||||||
|
typebot: typebot,
|
||||||
|
expire: expire,
|
||||||
|
sessions: sessions,
|
||||||
|
remoteJid: remoteJid,
|
||||||
|
});
|
||||||
|
|
||||||
|
await this.sendWAMessage(instance, remoteJid, data.messages);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!session) {
|
||||||
|
const data = await this.createNewSession(instance, {
|
||||||
|
url: url,
|
||||||
|
typebot: typebot,
|
||||||
|
expire: expire,
|
||||||
|
sessions: sessions,
|
||||||
|
remoteJid: remoteJid,
|
||||||
|
});
|
||||||
|
|
||||||
|
await this.sendWAMessage(instance, remoteJid, data.messages);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
sessions.map((session) => {
|
||||||
|
if (session.remoteJid === remoteJid) {
|
||||||
|
session.updateAt = Date.now();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const typebotData = {
|
||||||
|
enabled: true,
|
||||||
|
url: url,
|
||||||
|
typebot: typebot,
|
||||||
|
expire: expire,
|
||||||
|
sessions,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.create(instance, typebotData);
|
||||||
|
|
||||||
|
const content = this.getConversationMessage(msg.message);
|
||||||
|
|
||||||
|
if (!content) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (content.toLowerCase() === 'sair') {
|
||||||
|
sessions.splice(sessions.indexOf(session), 1);
|
||||||
|
|
||||||
|
const typebotData = {
|
||||||
|
enabled: true,
|
||||||
|
url: url,
|
||||||
|
typebot: typebot,
|
||||||
|
expire: expire,
|
||||||
|
sessions,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.create(instance, typebotData);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const reqData = {
|
||||||
|
message: content,
|
||||||
|
sessionId: session.sessionId.split('-')[1],
|
||||||
|
};
|
||||||
|
|
||||||
|
const request = await axios.post(url + '/api/v1/sendMessage', reqData);
|
||||||
|
|
||||||
|
if (!request.data.input) {
|
||||||
|
sessions.splice(sessions.indexOf(session), 1);
|
||||||
|
|
||||||
|
await this.createNewSession(instance, {
|
||||||
|
url: url,
|
||||||
|
typebot: typebot,
|
||||||
|
expire: expire,
|
||||||
|
sessions: sessions,
|
||||||
|
remoteJid: remoteJid,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.sendWAMessage(instance, remoteJid, request.data.messages, !request.data.input);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
@ -111,7 +111,7 @@ import {
|
|||||||
SendTextDto,
|
SendTextDto,
|
||||||
StatusMessage,
|
StatusMessage,
|
||||||
} from '../dto/sendMessage.dto';
|
} from '../dto/sendMessage.dto';
|
||||||
import { RabbitmqRaw, SettingsRaw } from '../models';
|
import { RabbitmqRaw, SettingsRaw, TypebotRaw } from '../models';
|
||||||
import { ChatRaw } from '../models/chat.model';
|
import { ChatRaw } from '../models/chat.model';
|
||||||
import { ChatwootRaw } from '../models/chatwoot.model';
|
import { ChatwootRaw } from '../models/chatwoot.model';
|
||||||
import { ContactRaw } from '../models/contact.model';
|
import { ContactRaw } from '../models/contact.model';
|
||||||
@ -125,6 +125,7 @@ import { RepositoryBroker } from '../repository/repository.manager';
|
|||||||
import { Events, MessageSubtype, TypeMediaMessage, wa } from '../types/wa.types';
|
import { Events, MessageSubtype, TypeMediaMessage, wa } from '../types/wa.types';
|
||||||
import { waMonitor } from '../whatsapp.module';
|
import { waMonitor } from '../whatsapp.module';
|
||||||
import { ChatwootService } from './chatwoot.service';
|
import { ChatwootService } from './chatwoot.service';
|
||||||
|
import { TypebotService } from './typebot.service';
|
||||||
|
|
||||||
export class WAStartupService {
|
export class WAStartupService {
|
||||||
constructor(
|
constructor(
|
||||||
@ -146,6 +147,7 @@ export class WAStartupService {
|
|||||||
private readonly localSettings: wa.LocalSettings = {};
|
private readonly localSettings: wa.LocalSettings = {};
|
||||||
private readonly localWebsocket: wa.LocalWebsocket = {};
|
private readonly localWebsocket: wa.LocalWebsocket = {};
|
||||||
private readonly localRabbitmq: wa.LocalRabbitmq = {};
|
private readonly localRabbitmq: wa.LocalRabbitmq = {};
|
||||||
|
private readonly localTypebot: wa.LocalTypebot = {};
|
||||||
public stateConnection: wa.StateConnection = { state: 'close' };
|
public stateConnection: wa.StateConnection = { state: 'close' };
|
||||||
public 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();
|
||||||
@ -157,6 +159,8 @@ export class WAStartupService {
|
|||||||
|
|
||||||
private chatwootService = new ChatwootService(waMonitor, this.configService);
|
private chatwootService = new ChatwootService(waMonitor, this.configService);
|
||||||
|
|
||||||
|
private typebotService = new TypebotService(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) {
|
||||||
@ -483,6 +487,48 @@ export class WAStartupService {
|
|||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async loadTypebot() {
|
||||||
|
this.logger.verbose('Loading typebot');
|
||||||
|
const data = await this.repository.typebot.find(this.instanceName);
|
||||||
|
|
||||||
|
this.localTypebot.enabled = data?.enabled;
|
||||||
|
this.logger.verbose(`Typebot enabled: ${this.localTypebot.enabled}`);
|
||||||
|
|
||||||
|
this.localTypebot.url = data?.url;
|
||||||
|
this.logger.verbose(`Typebot url: ${this.localTypebot.url}`);
|
||||||
|
|
||||||
|
this.localTypebot.typebot = data?.typebot;
|
||||||
|
this.logger.verbose(`Typebot typebot: ${this.localTypebot.typebot}`);
|
||||||
|
|
||||||
|
this.localTypebot.expire = data?.expire;
|
||||||
|
this.logger.verbose(`Typebot expire: ${this.localTypebot.expire}`);
|
||||||
|
|
||||||
|
this.localTypebot.sessions = data?.sessions;
|
||||||
|
|
||||||
|
this.logger.verbose('Typebot loaded');
|
||||||
|
}
|
||||||
|
|
||||||
|
public async setTypebot(data: TypebotRaw) {
|
||||||
|
this.logger.verbose('Setting typebot');
|
||||||
|
await this.repository.typebot.create(data, this.instanceName);
|
||||||
|
this.logger.verbose(`Typebot typebot: ${data.typebot}`);
|
||||||
|
this.logger.verbose(`Typebot expire: ${data.expire}`);
|
||||||
|
Object.assign(this.localTypebot, data);
|
||||||
|
this.logger.verbose('Typebot set');
|
||||||
|
}
|
||||||
|
|
||||||
|
public async findTypebot() {
|
||||||
|
this.logger.verbose('Finding typebot');
|
||||||
|
const data = await this.repository.typebot.find(this.instanceName);
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
this.logger.verbose('Typebot not found');
|
||||||
|
throw new NotFoundException('Typebot not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
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 webhookLocal = this.localWebhook.events;
|
const webhookLocal = this.localWebhook.events;
|
||||||
@ -943,6 +989,7 @@ export class WAStartupService {
|
|||||||
this.loadSettings();
|
this.loadSettings();
|
||||||
this.loadWebsocket();
|
this.loadWebsocket();
|
||||||
this.loadRabbitmq();
|
this.loadRabbitmq();
|
||||||
|
this.loadTypebot();
|
||||||
|
|
||||||
this.instance.authState = await this.defineAuthState();
|
this.instance.authState = await this.defineAuthState();
|
||||||
|
|
||||||
@ -1250,6 +1297,14 @@ export class WAStartupService {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.localTypebot.enabled) {
|
||||||
|
await this.typebotService.sendTypebot(
|
||||||
|
{ instanceName: this.instance.name },
|
||||||
|
messageRaw.key.remoteJid,
|
||||||
|
messageRaw,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.verbose('Inserting message in database');
|
this.logger.verbose('Inserting message in database');
|
||||||
await this.repository.message.insert([messageRaw], this.instance.name, database.SAVE_DATA.NEW_MESSAGE);
|
await this.repository.message.insert([messageRaw], this.instance.name, database.SAVE_DATA.NEW_MESSAGE);
|
||||||
|
|
||||||
|
@ -80,6 +80,20 @@ export declare namespace wa {
|
|||||||
events?: string[];
|
events?: string[];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
type Session = {
|
||||||
|
remoteJid?: string;
|
||||||
|
sessionId?: string;
|
||||||
|
createdAt?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type LocalTypebot = {
|
||||||
|
enabled?: boolean;
|
||||||
|
url?: string;
|
||||||
|
typebot?: string;
|
||||||
|
expire?: number;
|
||||||
|
sessions?: Session[];
|
||||||
|
};
|
||||||
|
|
||||||
export type StateConnection = {
|
export type StateConnection = {
|
||||||
instance?: string;
|
instance?: string;
|
||||||
state?: WAConnectionState | 'refused';
|
state?: WAConnectionState | 'refused';
|
||||||
|
@ -10,6 +10,7 @@ import { InstanceController } from './controllers/instance.controller';
|
|||||||
import { RabbitmqController } from './controllers/rabbitmq.controller';
|
import { RabbitmqController } from './controllers/rabbitmq.controller';
|
||||||
import { SendMessageController } from './controllers/sendMessage.controller';
|
import { SendMessageController } from './controllers/sendMessage.controller';
|
||||||
import { SettingsController } from './controllers/settings.controller';
|
import { SettingsController } from './controllers/settings.controller';
|
||||||
|
import { TypebotController } from './controllers/typebot.controller';
|
||||||
import { ViewsController } from './controllers/views.controller';
|
import { ViewsController } from './controllers/views.controller';
|
||||||
import { WebhookController } from './controllers/webhook.controller';
|
import { WebhookController } from './controllers/webhook.controller';
|
||||||
import { WebsocketController } from './controllers/websocket.controller';
|
import { WebsocketController } from './controllers/websocket.controller';
|
||||||
@ -22,6 +23,7 @@ import {
|
|||||||
MessageUpModel,
|
MessageUpModel,
|
||||||
RabbitmqModel,
|
RabbitmqModel,
|
||||||
SettingsModel,
|
SettingsModel,
|
||||||
|
TypebotModel,
|
||||||
WebhookModel,
|
WebhookModel,
|
||||||
WebsocketModel,
|
WebsocketModel,
|
||||||
} from './models';
|
} from './models';
|
||||||
@ -34,6 +36,7 @@ import { MessageUpRepository } from './repository/messageUp.repository';
|
|||||||
import { RabbitmqRepository } from './repository/rabbitmq.repository';
|
import { RabbitmqRepository } from './repository/rabbitmq.repository';
|
||||||
import { RepositoryBroker } from './repository/repository.manager';
|
import { RepositoryBroker } from './repository/repository.manager';
|
||||||
import { SettingsRepository } from './repository/settings.repository';
|
import { SettingsRepository } from './repository/settings.repository';
|
||||||
|
import { TypebotRepository } from './repository/typebot.repository';
|
||||||
import { WebhookRepository } from './repository/webhook.repository';
|
import { WebhookRepository } from './repository/webhook.repository';
|
||||||
import { WebsocketRepository } from './repository/websocket.repository';
|
import { WebsocketRepository } from './repository/websocket.repository';
|
||||||
import { AuthService } from './services/auth.service';
|
import { AuthService } from './services/auth.service';
|
||||||
@ -41,6 +44,7 @@ import { ChatwootService } from './services/chatwoot.service';
|
|||||||
import { WAMonitoringService } from './services/monitor.service';
|
import { WAMonitoringService } from './services/monitor.service';
|
||||||
import { RabbitmqService } from './services/rabbitmq.service';
|
import { RabbitmqService } from './services/rabbitmq.service';
|
||||||
import { SettingsService } from './services/settings.service';
|
import { SettingsService } from './services/settings.service';
|
||||||
|
import { TypebotService } from './services/typebot.service';
|
||||||
import { WebhookService } from './services/webhook.service';
|
import { WebhookService } from './services/webhook.service';
|
||||||
import { WebsocketService } from './services/websocket.service';
|
import { WebsocketService } from './services/websocket.service';
|
||||||
|
|
||||||
@ -50,6 +54,7 @@ const messageRepository = new MessageRepository(MessageModel, configService);
|
|||||||
const chatRepository = new ChatRepository(ChatModel, configService);
|
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 typebotRepository = new TypebotRepository(TypebotModel, configService);
|
||||||
const webhookRepository = new WebhookRepository(WebhookModel, configService);
|
const webhookRepository = new WebhookRepository(WebhookModel, configService);
|
||||||
const websocketRepository = new WebsocketRepository(WebsocketModel, configService);
|
const websocketRepository = new WebsocketRepository(WebsocketModel, configService);
|
||||||
const rabbitmqRepository = new RabbitmqRepository(RabbitmqModel, configService);
|
const rabbitmqRepository = new RabbitmqRepository(RabbitmqModel, configService);
|
||||||
@ -67,6 +72,7 @@ export const repository = new RepositoryBroker(
|
|||||||
settingsRepository,
|
settingsRepository,
|
||||||
websocketRepository,
|
websocketRepository,
|
||||||
rabbitmqRepository,
|
rabbitmqRepository,
|
||||||
|
typebotRepository,
|
||||||
authRepository,
|
authRepository,
|
||||||
configService,
|
configService,
|
||||||
dbserver?.getClient(),
|
dbserver?.getClient(),
|
||||||
@ -78,6 +84,10 @@ export const waMonitor = new WAMonitoringService(eventEmitter, configService, re
|
|||||||
|
|
||||||
const authService = new AuthService(configService, waMonitor, repository);
|
const authService = new AuthService(configService, waMonitor, repository);
|
||||||
|
|
||||||
|
const typebotService = new TypebotService(waMonitor);
|
||||||
|
|
||||||
|
export const typebotController = new TypebotController(typebotService);
|
||||||
|
|
||||||
const webhookService = new WebhookService(waMonitor);
|
const webhookService = new WebhookService(waMonitor);
|
||||||
|
|
||||||
export const webhookController = new WebhookController(webhookService);
|
export const webhookController = new WebhookController(webhookService);
|
||||||
|
Loading…
Reference in New Issue
Block a user