feat: dispositivos e ajustes de csat e relatórios
This commit is contained in:
parent
25d2a9b062
commit
e0ef66555d
86 changed files with 5811 additions and 992 deletions
263
tests/tickets.lifecycle.test.ts
Normal file
263
tests/tickets.lifecycle.test.ts
Normal file
|
|
@ -0,0 +1,263 @@
|
|||
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest"
|
||||
|
||||
import type { Doc, Id } from "../convex/_generated/dataModel"
|
||||
import { resolveTicketHandler, reopenTicketHandler } from "../convex/tickets"
|
||||
import { requireStaff, requireUser } from "../convex/rbac"
|
||||
|
||||
type ResolveCtx = Parameters<typeof resolveTicketHandler>[0]
|
||||
type ReopenCtx = Parameters<typeof reopenTicketHandler>[0]
|
||||
|
||||
type TicketDoc = Doc<"tickets">
|
||||
|
||||
type MockedTicket = TicketDoc & { _id: Id<"tickets"> }
|
||||
|
||||
defineMocks()
|
||||
|
||||
function defineMocks() {
|
||||
vi.mock("../convex/rbac", () => ({
|
||||
requireStaff: vi.fn(),
|
||||
requireUser: vi.fn(),
|
||||
requireAdmin: vi.fn(),
|
||||
}))
|
||||
}
|
||||
|
||||
const mockedRequireStaff = vi.mocked(requireStaff)
|
||||
const mockedRequireUser = vi.mocked(requireUser)
|
||||
|
||||
const NOW = 1_706_700_000_000
|
||||
|
||||
beforeEach(() => {
|
||||
vi.setSystemTime(NOW)
|
||||
})
|
||||
|
||||
afterEach(() => {
|
||||
vi.clearAllMocks()
|
||||
})
|
||||
|
||||
function buildTicket(overrides: Partial<TicketDoc> = {}): MockedTicket {
|
||||
const base: TicketDoc = {
|
||||
_id: "ticket_main" as Id<"tickets">,
|
||||
_creationTime: NOW - 10_000,
|
||||
tenantId: "tenant-1",
|
||||
reference: 41_000,
|
||||
subject: "Computador não liga",
|
||||
summary: null,
|
||||
status: "AWAITING_ATTENDANCE",
|
||||
priority: "MEDIUM",
|
||||
channel: "EMAIL",
|
||||
queueId: null,
|
||||
queueSnapshot: null,
|
||||
requesterId: "user_requester" as Id<"users">,
|
||||
requesterSnapshot: { name: "Cliente", email: "cliente@example.com" },
|
||||
assigneeId: "user_agent" as Id<"users">,
|
||||
assigneeSnapshot: { name: "Agente", email: "agente@example.com" },
|
||||
companyId: null,
|
||||
companySnapshot: null,
|
||||
machineId: null,
|
||||
machineSnapshot: null,
|
||||
slaPolicyId: null,
|
||||
dueAt: null,
|
||||
firstResponseAt: null,
|
||||
resolvedAt: null,
|
||||
createdAt: NOW - 50_000,
|
||||
updatedAt: NOW - 1_000,
|
||||
tags: [],
|
||||
customFields: [],
|
||||
activeSessionId: null,
|
||||
totalWorkedMs: 0,
|
||||
internalWorkedMs: 0,
|
||||
externalWorkedMs: 0,
|
||||
csatScore: undefined,
|
||||
csatMaxScore: undefined,
|
||||
csatComment: undefined,
|
||||
csatRatedAt: undefined,
|
||||
csatRatedBy: undefined,
|
||||
csatAssigneeId: undefined,
|
||||
csatAssigneeSnapshot: undefined,
|
||||
workSummary: undefined,
|
||||
reopenDeadline: undefined,
|
||||
reopenedAt: undefined,
|
||||
formTemplate: undefined,
|
||||
chatEnabled: true,
|
||||
relatedTicketIds: undefined,
|
||||
resolvedWithTicketId: undefined,
|
||||
reopenWindowDays: undefined,
|
||||
reopenedBy: undefined,
|
||||
}
|
||||
return { ...(base as TicketDoc), ...overrides }
|
||||
}
|
||||
|
||||
function createResolveCtx(tickets: Record<string, TicketDoc>, events: Doc<"ticketEvents">[] = []) {
|
||||
const patch = vi.fn(async () => undefined)
|
||||
const insert = vi.fn(async () => undefined)
|
||||
const collect = vi.fn(async () => events)
|
||||
const query = vi.fn(() => ({
|
||||
withIndex: vi.fn((_index: string, cb: (builder: { eq: (field: string, value: unknown) => { collect: typeof collect } }) => { collect: typeof collect }) => {
|
||||
const cursor = {
|
||||
eq: vi.fn(() => ({ collect })),
|
||||
}
|
||||
const result = cb(cursor as { eq: (field: string, value: unknown) => { collect: typeof collect } })
|
||||
return result ?? { collect }
|
||||
}),
|
||||
}))
|
||||
|
||||
const ctx: ResolveCtx = {
|
||||
db: {
|
||||
get: vi.fn(async (id: Id<"tickets"> | Id<"users">) => {
|
||||
return tickets[String(id)] ?? null
|
||||
}),
|
||||
patch,
|
||||
insert,
|
||||
query,
|
||||
delete: vi.fn(),
|
||||
},
|
||||
} as unknown as ResolveCtx
|
||||
|
||||
return { ctx, patch, insert }
|
||||
}
|
||||
|
||||
function createReopenCtx(ticket: TicketDoc) {
|
||||
const patch = vi.fn(async () => undefined)
|
||||
const insert = vi.fn(async () => undefined)
|
||||
const ctx: ReopenCtx = {
|
||||
db: {
|
||||
get: vi.fn(async () => ticket),
|
||||
patch,
|
||||
insert,
|
||||
query: vi.fn(),
|
||||
delete: vi.fn(),
|
||||
},
|
||||
} as unknown as ReopenCtx
|
||||
|
||||
return { ctx, patch, insert }
|
||||
}
|
||||
|
||||
describe("convex.tickets.resolveTicketHandler", () => {
|
||||
it("marca o ticket como resolvido, vincula outro ticket e define prazo de reabertura", async () => {
|
||||
const ticketMain = buildTicket()
|
||||
const ticketLinked = buildTicket({
|
||||
_id: "ticket_related" as Id<"tickets">,
|
||||
reference: 41_001,
|
||||
status: "AWAITING_ATTENDANCE",
|
||||
})
|
||||
|
||||
const { ctx, patch, insert } = createResolveCtx({
|
||||
[String(ticketMain._id)]: ticketMain,
|
||||
[String(ticketLinked._id)]: ticketLinked,
|
||||
})
|
||||
|
||||
mockedRequireStaff.mockResolvedValue({
|
||||
user: {
|
||||
_id: "user_agent" as Id<"users">,
|
||||
name: "Agente",
|
||||
email: "agente@example.com",
|
||||
},
|
||||
role: "ADMIN",
|
||||
})
|
||||
|
||||
const result = await resolveTicketHandler(ctx, {
|
||||
ticketId: ticketMain._id,
|
||||
actorId: "user_agent" as Id<"users">,
|
||||
resolvedWithTicketId: ticketLinked._id,
|
||||
reopenWindowDays: 14,
|
||||
})
|
||||
|
||||
expect(result.ok).toBe(true)
|
||||
expect(result.reopenWindowDays).toBe(14)
|
||||
expect(result.reopenDeadline).toBe(NOW + 14 * 24 * 60 * 60 * 1000)
|
||||
|
||||
expect(patch).toHaveBeenCalledWith(
|
||||
ticketMain._id,
|
||||
expect.objectContaining({
|
||||
status: "RESOLVED",
|
||||
reopenDeadline: result.reopenDeadline,
|
||||
resolvedWithTicketId: ticketLinked._id,
|
||||
})
|
||||
)
|
||||
|
||||
expect(patch).toHaveBeenCalledWith(
|
||||
ticketLinked._id,
|
||||
expect.objectContaining({
|
||||
relatedTicketIds: expect.arrayContaining([ticketMain._id]),
|
||||
})
|
||||
)
|
||||
|
||||
expect(insert).toHaveBeenCalledWith(
|
||||
"ticketEvents",
|
||||
expect.objectContaining({
|
||||
ticketId: ticketMain._id,
|
||||
type: "STATUS_CHANGED",
|
||||
payload: expect.objectContaining({ to: "RESOLVED" }),
|
||||
})
|
||||
)
|
||||
expect(insert).toHaveBeenCalledWith(
|
||||
"ticketEvents",
|
||||
expect.objectContaining({
|
||||
ticketId: ticketLinked._id,
|
||||
type: "TICKET_LINKED",
|
||||
payload: expect.objectContaining({ linkedTicketId: ticketMain._id }),
|
||||
})
|
||||
)
|
||||
})
|
||||
})
|
||||
|
||||
describe("convex.tickets.reopenTicketHandler", () => {
|
||||
it("reabre o ticket quando dentro do prazo e permissões válidas", async () => {
|
||||
const ticket = buildTicket({
|
||||
status: "RESOLVED",
|
||||
reopenDeadline: NOW + 3 * 24 * 60 * 60 * 1000,
|
||||
resolvedAt: NOW - 60_000,
|
||||
})
|
||||
|
||||
const { ctx, patch, insert } = createReopenCtx(ticket)
|
||||
|
||||
mockedRequireUser.mockResolvedValue({
|
||||
user: {
|
||||
_id: ticket.requesterId,
|
||||
email: "cliente@example.com",
|
||||
companyId: null,
|
||||
},
|
||||
role: "COLLABORATOR",
|
||||
})
|
||||
|
||||
const result = await reopenTicketHandler(ctx, {
|
||||
ticketId: ticket._id,
|
||||
actorId: ticket.requesterId,
|
||||
})
|
||||
|
||||
expect(result.ok).toBe(true)
|
||||
expect(result.reopenedAt).toBe(NOW)
|
||||
expect(patch).toHaveBeenCalledWith(
|
||||
ticket._id,
|
||||
expect.objectContaining({ status: "AWAITING_ATTENDANCE", reopenedAt: NOW, resolvedAt: undefined })
|
||||
)
|
||||
expect(insert).toHaveBeenCalledWith(
|
||||
"ticketEvents",
|
||||
expect.objectContaining({ ticketId: ticket._id, type: "TICKET_REOPENED" })
|
||||
)
|
||||
})
|
||||
|
||||
it("falha quando o prazo para reabrir expirou", async () => {
|
||||
const ticket = buildTicket({
|
||||
status: "RESOLVED",
|
||||
reopenDeadline: NOW - 1,
|
||||
})
|
||||
|
||||
const { ctx } = createReopenCtx(ticket)
|
||||
mockedRequireUser.mockResolvedValue({
|
||||
user: {
|
||||
_id: ticket.requesterId,
|
||||
email: "cliente@example.com",
|
||||
companyId: null,
|
||||
},
|
||||
role: "COLLABORATOR",
|
||||
})
|
||||
|
||||
await expect(
|
||||
reopenTicketHandler(ctx, {
|
||||
ticketId: ticket._id,
|
||||
actorId: ticket.requesterId,
|
||||
})
|
||||
).rejects.toThrow("O prazo para reabrir este chamado expirou")
|
||||
})
|
||||
})
|
||||
Loading…
Add table
Add a link
Reference in a new issue