fix(api): modifica fetchChats para trazer mensagens de contatos não salvos

- Muda tabela base da consulta de Contact para Message
- Altera INNER JOIN para LEFT JOIN entre Message e Contact
- Usa COALESCE para campos que podem estar vazios
- Adiciona flag isSaved para identificar contatos salvos/não salvos
- Preserva toda funcionalidade de filtros existente

Resolve issue #1376
This commit is contained in:
Leandro Rocha 2025-04-21 00:30:48 -03:00
parent 2ded19752f
commit b94b452597

View File

@ -682,13 +682,6 @@ export class ChannelStartupService {
: createJid(query.where?.remoteJid) : createJid(query.where?.remoteJid)
: null; : null;
const limit =
query.offset && !query.page
? Prisma.sql` LIMIT ${query.offset}`
: query.offset && query.page
? Prisma.sql` LIMIT ${query.offset} OFFSET ${((query.page as number) - 1) * query.offset}`
: Prisma.sql``;
const where = { const where = {
instanceId: this.instanceId, instanceId: this.instanceId,
}; };
@ -700,91 +693,88 @@ export class ChannelStartupService {
const timestampFilter = const timestampFilter =
query?.where?.messageTimestamp?.gte && query?.where?.messageTimestamp?.lte query?.where?.messageTimestamp?.gte && query?.where?.messageTimestamp?.lte
? Prisma.sql` ? Prisma.sql`
AND "Message"."messageTimestamp" >= ${Math.floor(new Date(query.where.messageTimestamp.gte).getTime() / 1000)} AND "Message"."messageTimestamp" >= ${Math.floor(new Date(query.where.messageTimestamp.gte).getTime() / 1000)}
AND "Message"."messageTimestamp" <= ${Math.floor(new Date(query.where.messageTimestamp.lte).getTime() / 1000)}` AND "Message"."messageTimestamp" <= ${Math.floor(new Date(query.where.messageTimestamp.lte).getTime() / 1000)}`
: Prisma.sql``; : Prisma.sql``;
const results = await this.prismaRepository.$queryRaw` const results = await this.prismaRepository.$queryRaw`
WITH rankedMessages AS ( WITH rankedMessages AS (
SELECT DISTINCT ON ("Contact"."remoteJid") SELECT DISTINCT ON ("Message"."key"->>'remoteJid')
"Contact"."id", "Contact"."id" as "contactId",
"Contact"."remoteJid", "Message"."key"->>'remoteJid' as "remoteJid",
"Contact"."pushName", COALESCE("Contact"."pushName", "Message"."pushName") as "pushName",
"Contact"."profilePicUrl", "Contact"."profilePicUrl",
COALESCE( COALESCE(
to_timestamp("Message"."messageTimestamp"::double precision), to_timestamp("Message"."messageTimestamp"::double precision),
"Contact"."updatedAt" "Contact"."updatedAt"
) as "updatedAt", ) as "updatedAt",
"Chat"."name" as "chatName", "Chat"."createdAt" as "windowStart",
"Chat"."createdAt" as "windowStart", "Chat"."createdAt" + INTERVAL '24 hours' as "windowExpires",
"Chat"."createdAt" + INTERVAL '24 hours' as "windowExpires", CASE WHEN "Chat"."createdAt" + INTERVAL '24 hours' > NOW() THEN true ELSE false END as "windowActive",
CASE "Message"."id" AS lastMessageId,
WHEN "Chat"."createdAt" + INTERVAL '24 hours' > NOW() THEN true "Message"."key" AS lastMessage_key,
ELSE false "Message"."pushName" AS lastMessagePushName,
END as "windowActive", "Message"."participant" AS lastMessageParticipant,
"Message"."id" AS lastMessageId, "Message"."messageType" AS lastMessageMessageType,
"Message"."key" AS lastMessage_key, "Message"."message" AS lastMessageMessage,
"Message"."pushName" AS lastMessagePushName, "Message"."contextInfo" AS lastMessageContextInfo,
"Message"."participant" AS lastMessageParticipant, "Message"."source" AS lastMessageSource,
"Message"."messageType" AS lastMessageMessageType, "Message"."messageTimestamp" AS lastMessageMessageTimestamp,
"Message"."message" AS lastMessageMessage, "Message"."instanceId" AS lastMessageInstanceId,
"Message"."contextInfo" AS lastMessageContextInfo, "Message"."sessionId" AS lastMessageSessionId,
"Message"."source" AS lastMessageSource, "Message"."status" AS lastMessageStatus
"Message"."messageTimestamp" AS lastMessageMessageTimestamp, FROM "Message"
"Message"."instanceId" AS lastMessageInstanceId, LEFT JOIN "Contact" ON "Contact"."remoteJid" = "Message"."key"->>'remoteJid' AND "Contact"."instanceId" = "Message"."instanceId"
"Message"."sessionId" AS lastMessageSessionId, LEFT JOIN "Chat" ON "Chat"."remoteJid" = "Message"."key"->>'remoteJid' AND "Chat"."instanceId" = "Message"."instanceId"
"Message"."status" AS lastMessageStatus WHERE "Message"."instanceId" = ${this.instanceId}
FROM "Contact" ${remoteJid ? Prisma.sql`AND "Message"."key"->>'remoteJid' = ${remoteJid}` : Prisma.sql``}
INNER JOIN "Message" ON "Message"."key"->>'remoteJid' = "Contact"."remoteJid" ${timestampFilter}
LEFT JOIN "Chat" ON "Chat"."remoteJid" = "Contact"."remoteJid" ORDER BY "Message"."key"->>'remoteJid', "Message"."messageTimestamp" DESC
AND "Chat"."instanceId" = "Contact"."instanceId" )
WHERE SELECT * FROM rankedMessages
"Contact"."instanceId" = ${this.instanceId} ORDER BY "updatedAt" DESC NULLS LAST;
AND "Message"."instanceId" = ${this.instanceId}
${remoteJid ? Prisma.sql`AND "Contact"."remoteJid" = ${remoteJid}` : Prisma.sql``}
${timestampFilter}
ORDER BY
"Contact"."remoteJid",
"Message"."messageTimestamp" DESC
${limit}
)
SELECT * FROM rankedMessages
ORDER BY "updatedAt" DESC NULLS LAST;
`; `;
if (results && isArray(results) && results.length > 0) { if (results && Array.isArray(results) && results.length > 0) {
const mappedResults = results.map((contact) => { const mappedResults = results.map((item) => {
const lastMessage = contact.lastMessageId const lastMessage = item.lastMessageId
? { ? {
id: contact.lastMessageId, id: item.lastMessageId,
key: contact.lastMessageKey, key: item.lastMessage_key,
pushName: contact.lastMessagePushName, pushName: item.lastMessagePushName,
participant: contact.lastMessageParticipant, participant: item.lastMessageParticipant,
messageType: contact.lastMessageMessageType, messageType: item.lastMessageMessageType,
message: contact.lastMessageMessage, message: item.lastMessageMessage,
contextInfo: contact.lastMessageContextInfo, contextInfo: item.lastMessageContextInfo,
source: contact.lastMessageSource, source: item.lastMessageSource,
messageTimestamp: contact.lastMessageMessageTimestamp, messageTimestamp: item.lastMessageMessageTimestamp,
instanceId: contact.lastMessageInstanceId, instanceId: item.lastMessageInstanceId,
sessionId: contact.lastMessageSessionId, sessionId: item.lastMessageSessionId,
status: contact.lastMessageStatus, status: item.lastMessageStatus,
} }
: undefined; : undefined;
return { return {
id: contact.id, id: item.contactId || null,
remoteJid: contact.remoteJid, remoteJid: item.remoteJid,
pushName: contact.pushName, pushName: item.pushName,
chatName: contact.chatName, profilePicUrl: item.profilePicUrl,
profilePicUrl: contact.profilePicUrl, updatedAt: item.updatedAt,
updatedAt: contact.updatedAt, windowStart: item.windowStart,
windowStart: contact.windowStart, windowExpires: item.windowExpires,
windowExpires: contact.windowExpires, windowActive: item.windowActive,
windowActive: contact.windowActive,
lastMessage: lastMessage ? this.cleanMessageData(lastMessage) : undefined, lastMessage: lastMessage ? this.cleanMessageData(lastMessage) : undefined,
unreadCount: 0,
isSaved: !!item.contactId,
}; };
}); });
if (query?.take && query?.skip) {
const skip = query.skip || 0;
const take = query.take || 20;
return mappedResults.slice(skip, skip + take);
}
return mappedResults; return mappedResults;
} }