mirror of
https://github.com/EvolutionAPI/evolution-api.git
synced 2025-07-14 01:41:24 -06:00
Merge branch 'develop' into chatwoot-format
This commit is contained in:
commit
fb24b7eaa7
11
CHANGELOG.md
11
CHANGELOG.md
@ -1,17 +1,16 @@
|
|||||||
# 1.6.1 (develop)
|
# 1.6.1 (develop)
|
||||||
|
|
||||||
### Feature
|
|
||||||
|
|
||||||
* New env `TYPEBOT_KEEP_OPEN` to keep the session open after the end of the flow
|
|
||||||
* Add `sign_delimiter` to chatwoot set to change the delimiter of the signature
|
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
* Fixed Lid Messages
|
* Fixed Lid Messages
|
||||||
* Fixed sending variables to typebot
|
* Fixed sending variables to typebot
|
||||||
* Fixed sending variables from typebot
|
* Fixed sending variables from typebot
|
||||||
* Correction sending s3/minio media to chatwoot and typebot
|
* Correction sending s3/minio media to chatwoot and typebot
|
||||||
* Fixed message formatting in chatwoot (bold, italic, underline, monospace)
|
* Fixed the problem with typebot closing at the end of the flow, now this is optional with the TYPEBOT_KEEP_OPEN variable
|
||||||
|
* Fixed chatwoot Bold, Italic and Underline formatting using Regex
|
||||||
|
* Added the sign_delimiter property to the Chatwoot configuration, allowing you to set a different delimiter for the signature. Default when not defined \n
|
||||||
|
* Include instance Id field in the instance configuration
|
||||||
|
|
||||||
|
|
||||||
# 1.6.0 (2023-12-12 17:24)
|
# 1.6.0 (2023-12-12 17:24)
|
||||||
|
|
||||||
|
@ -3,7 +3,7 @@ version: '3.3'
|
|||||||
services:
|
services:
|
||||||
api:
|
api:
|
||||||
container_name: evolution_api
|
container_name: evolution_api
|
||||||
image: davidsongomes/evolution-api:latest
|
image: atendai/evolution-api:latest
|
||||||
restart: always
|
restart: always
|
||||||
ports:
|
ports:
|
||||||
- 8080:8080
|
- 8080:8080
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { delay } from '@whiskeysockets/baileys';
|
import { delay } from '@whiskeysockets/baileys';
|
||||||
import { isURL } from 'class-validator';
|
import { isURL } from 'class-validator';
|
||||||
import EventEmitter2 from 'eventemitter2';
|
import EventEmitter2 from 'eventemitter2';
|
||||||
|
import { v4 } from 'uuid';
|
||||||
|
|
||||||
import { ConfigService, HttpServer } from '../../config/env.config';
|
import { ConfigService, HttpServer } from '../../config/env.config';
|
||||||
import { Logger } from '../../config/logger.config';
|
import { Logger } from '../../config/logger.config';
|
||||||
@ -87,8 +88,11 @@ export class InstanceController {
|
|||||||
const instance = new WAStartupService(this.configService, this.eventEmitter, this.repository, this.cache);
|
const instance = new WAStartupService(this.configService, this.eventEmitter, this.repository, this.cache);
|
||||||
instance.instanceName = instanceName;
|
instance.instanceName = instanceName;
|
||||||
|
|
||||||
|
const instanceId = v4();
|
||||||
|
|
||||||
instance.sendDataWebhook(Events.INSTANCE_CREATE, {
|
instance.sendDataWebhook(Events.INSTANCE_CREATE, {
|
||||||
instanceName,
|
instanceName,
|
||||||
|
instanceId: instanceId,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.logger.verbose('instance: ' + instance.instanceName + ' created');
|
this.logger.verbose('instance: ' + instance.instanceName + ' created');
|
||||||
@ -100,6 +104,7 @@ export class InstanceController {
|
|||||||
const hash = await this.authService.generateHash(
|
const hash = await this.authService.generateHash(
|
||||||
{
|
{
|
||||||
instanceName: instance.instanceName,
|
instanceName: instance.instanceName,
|
||||||
|
instanceId: instanceId,
|
||||||
},
|
},
|
||||||
token,
|
token,
|
||||||
);
|
);
|
||||||
@ -367,6 +372,7 @@ export class InstanceController {
|
|||||||
const result = {
|
const result = {
|
||||||
instance: {
|
instance: {
|
||||||
instanceName: instance.instanceName,
|
instanceName: instance.instanceName,
|
||||||
|
instanceId: instanceId,
|
||||||
status: 'created',
|
status: 'created',
|
||||||
},
|
},
|
||||||
hash,
|
hash,
|
||||||
@ -459,6 +465,7 @@ export class InstanceController {
|
|||||||
return {
|
return {
|
||||||
instance: {
|
instance: {
|
||||||
instanceName: instance.instanceName,
|
instanceName: instance.instanceName,
|
||||||
|
instanceId: instanceId,
|
||||||
status: 'created',
|
status: 'created',
|
||||||
},
|
},
|
||||||
hash,
|
hash,
|
||||||
@ -584,11 +591,13 @@ export class InstanceController {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public async fetchInstances({ instanceName }: InstanceDto) {
|
public async fetchInstances({ instanceName, instanceId }: InstanceDto) {
|
||||||
if (instanceName) {
|
if (instanceName) {
|
||||||
this.logger.verbose('requested fetchInstances from ' + instanceName + ' instance');
|
this.logger.verbose('requested fetchInstances from ' + instanceName + ' instance');
|
||||||
this.logger.verbose('instanceName: ' + instanceName);
|
this.logger.verbose('instanceName: ' + instanceName);
|
||||||
return this.waMonitor.instanceInfo(instanceName);
|
return this.waMonitor.instanceInfo(instanceName);
|
||||||
|
} else if (instanceId) {
|
||||||
|
return this.waMonitor.instanceInfoById(instanceId);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.logger.verbose('requested fetchInstances (all instances)');
|
this.logger.verbose('requested fetchInstances (all instances)');
|
||||||
@ -636,6 +645,7 @@ export class InstanceController {
|
|||||||
|
|
||||||
this.waMonitor.waInstances[instanceName].sendDataWebhook(Events.INSTANCE_DELETE, {
|
this.waMonitor.waInstances[instanceName].sendDataWebhook(Events.INSTANCE_DELETE, {
|
||||||
instanceName,
|
instanceName,
|
||||||
|
instanceId: (await this.repository.auth.find(instanceName))?.instanceId,
|
||||||
});
|
});
|
||||||
delete this.waMonitor.waInstances[instanceName];
|
delete this.waMonitor.waInstances[instanceName];
|
||||||
this.eventEmitter.emit('remove.instance', instanceName, 'inner');
|
this.eventEmitter.emit('remove.instance', instanceName, 'inner');
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
export class InstanceDto {
|
export class InstanceDto {
|
||||||
instanceName: string;
|
instanceName: string;
|
||||||
|
instanceId?: string;
|
||||||
qrcode?: boolean;
|
qrcode?: boolean;
|
||||||
number?: string;
|
number?: string;
|
||||||
token?: string;
|
token?: string;
|
||||||
|
@ -6,12 +6,14 @@ export class AuthRaw {
|
|||||||
_id?: string;
|
_id?: string;
|
||||||
jwt?: string;
|
jwt?: string;
|
||||||
apikey?: string;
|
apikey?: string;
|
||||||
|
instanceId?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const authSchema = new Schema<AuthRaw>({
|
const authSchema = new Schema<AuthRaw>({
|
||||||
_id: { type: String, _id: true },
|
_id: { type: String, _id: true },
|
||||||
jwt: { type: String, minlength: 1 },
|
jwt: { type: String, minlength: 1 },
|
||||||
apikey: { type: String, minlength: 1 },
|
apikey: { type: String, minlength: 1 },
|
||||||
|
instanceId: { type: String, minlength: 1 },
|
||||||
});
|
});
|
||||||
|
|
||||||
export const AuthModel = dbserver?.model(AuthRaw.name, authSchema, 'authentication');
|
export const AuthModel = dbserver?.model(AuthRaw.name, authSchema, 'authentication');
|
||||||
|
@ -19,6 +19,7 @@ export class AuthRepository extends Repository {
|
|||||||
public async create(data: AuthRaw, instance: string): Promise<IInsert> {
|
public async create(data: AuthRaw, instance: string): Promise<IInsert> {
|
||||||
try {
|
try {
|
||||||
this.logger.verbose('creating auth');
|
this.logger.verbose('creating auth');
|
||||||
|
|
||||||
if (this.dbSettings.ENABLED) {
|
if (this.dbSettings.ENABLED) {
|
||||||
this.logger.verbose('saving auth to db');
|
this.logger.verbose('saving auth to db');
|
||||||
const insert = await this.authModel.replaceOne({ _id: instance }, { ...data }, { upsert: true });
|
const insert = await this.authModel.replaceOne({ _id: instance }, { ...data }, { upsert: true });
|
||||||
@ -62,4 +63,20 @@ export class AuthRepository extends Repository {
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async findInstanceNameById(instanceId: string): Promise<string | null> {
|
||||||
|
try {
|
||||||
|
this.logger.verbose('finding auth by instanceId');
|
||||||
|
if (this.dbSettings.ENABLED) {
|
||||||
|
this.logger.verbose('finding auth in db');
|
||||||
|
const response = await this.authModel.findOne({ instanceId });
|
||||||
|
|
||||||
|
return response._id;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('finding auth in store is not supported');
|
||||||
|
} catch (error) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -46,7 +46,10 @@ export class AuthService {
|
|||||||
|
|
||||||
this.logger.verbose('JWT token created: ' + token);
|
this.logger.verbose('JWT token created: ' + token);
|
||||||
|
|
||||||
const auth = await this.repository.auth.create({ jwt: token }, instance.instanceName);
|
const auth = await this.repository.auth.create(
|
||||||
|
{ jwt: token, instanceId: instance.instanceId },
|
||||||
|
instance.instanceName,
|
||||||
|
);
|
||||||
|
|
||||||
this.logger.verbose('JWT token saved in database');
|
this.logger.verbose('JWT token saved in database');
|
||||||
|
|
||||||
@ -66,7 +69,7 @@ export class AuthService {
|
|||||||
|
|
||||||
this.logger.verbose(token ? 'APIKEY defined: ' + apikey : 'APIKEY created: ' + apikey);
|
this.logger.verbose(token ? 'APIKEY defined: ' + apikey : 'APIKEY created: ' + apikey);
|
||||||
|
|
||||||
const auth = await this.repository.auth.create({ apikey }, instance.instanceName);
|
const auth = await this.repository.auth.create({ apikey, instanceId: instance.instanceId }, instance.instanceName);
|
||||||
|
|
||||||
this.logger.verbose('APIKEY saved in database');
|
this.logger.verbose('APIKEY saved in database');
|
||||||
|
|
||||||
|
@ -112,6 +112,7 @@ export class WAMonitoringService {
|
|||||||
const instanceData = {
|
const instanceData = {
|
||||||
instance: {
|
instance: {
|
||||||
instanceName: key,
|
instanceName: key,
|
||||||
|
instanceId: (await this.repository.auth.find(key))?.instanceId,
|
||||||
owner: value.wuid,
|
owner: value.wuid,
|
||||||
profileName: (await value.getProfileName()) || 'not loaded',
|
profileName: (await value.getProfileName()) || 'not loaded',
|
||||||
profilePictureUrl: value.profilePictureUrl,
|
profilePictureUrl: value.profilePictureUrl,
|
||||||
@ -135,6 +136,89 @@ export class WAMonitoringService {
|
|||||||
const instanceData = {
|
const instanceData = {
|
||||||
instance: {
|
instance: {
|
||||||
instanceName: key,
|
instanceName: key,
|
||||||
|
instanceId: (await this.repository.auth.find(key))?.instanceId,
|
||||||
|
status: value.connectionStatus.state,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
if (this.configService.get<Auth>('AUTHENTICATION').EXPOSE_IN_FETCH_INSTANCES) {
|
||||||
|
instanceData.instance['serverUrl'] = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
instanceData.instance['apikey'] = (await this.repository.auth.find(key))?.apikey;
|
||||||
|
|
||||||
|
instanceData.instance['chatwoot'] = chatwoot;
|
||||||
|
}
|
||||||
|
|
||||||
|
instances.push(instanceData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.verbose('return instance info: ' + instances.length);
|
||||||
|
|
||||||
|
return instances.find((i) => i.instance.instanceName === instanceName) ?? instances;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async instanceInfoById(instanceId?: string) {
|
||||||
|
this.logger.verbose('get instance info');
|
||||||
|
const instanceName = await this.repository.auth.findInstanceNameById(instanceId);
|
||||||
|
if (!instanceName) {
|
||||||
|
throw new NotFoundException(`Instance "${instanceId}" not found`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (instanceName && !this.waInstances[instanceName]) {
|
||||||
|
throw new NotFoundException(`Instance "${instanceName}" not found`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const instances: any[] = [];
|
||||||
|
|
||||||
|
for await (const [key, value] of Object.entries(this.waInstances)) {
|
||||||
|
if (value) {
|
||||||
|
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 && findChatwoot.enabled) {
|
||||||
|
chatwoot = {
|
||||||
|
...findChatwoot,
|
||||||
|
webhook_url: `${urlServer}/chatwoot/webhook/${encodeURIComponent(key)}`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value.connectionStatus.state === 'open') {
|
||||||
|
this.logger.verbose('instance: ' + key + ' - connectionStatus: open');
|
||||||
|
|
||||||
|
const instanceData = {
|
||||||
|
instance: {
|
||||||
|
instanceName: key,
|
||||||
|
instanceId: (await this.repository.auth.find(key))?.instanceId,
|
||||||
|
owner: value.wuid,
|
||||||
|
profileName: (await value.getProfileName()) || 'not loaded',
|
||||||
|
profilePictureUrl: value.profilePictureUrl,
|
||||||
|
profileStatus: (await value.getProfileStatus()) || '',
|
||||||
|
status: value.connectionStatus.state,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
if (this.configService.get<Auth>('AUTHENTICATION').EXPOSE_IN_FETCH_INSTANCES) {
|
||||||
|
instanceData.instance['serverUrl'] = this.configService.get<HttpServer>('SERVER').URL;
|
||||||
|
|
||||||
|
instanceData.instance['apikey'] = (await this.repository.auth.find(key))?.apikey;
|
||||||
|
|
||||||
|
instanceData.instance['chatwoot'] = chatwoot;
|
||||||
|
}
|
||||||
|
|
||||||
|
instances.push(instanceData);
|
||||||
|
} else {
|
||||||
|
this.logger.verbose('instance: ' + key + ' - connectionStatus: ' + value.connectionStatus.state);
|
||||||
|
|
||||||
|
const instanceData = {
|
||||||
|
instance: {
|
||||||
|
instanceName: key,
|
||||||
|
instanceId: (await this.repository.auth.find(key))?.instanceId,
|
||||||
status: value.connectionStatus.state,
|
status: value.connectionStatus.state,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user