717 lines
27 KiB
TypeScript
717 lines
27 KiB
TypeScript
import { Router, type Request, type Response } from "express";
|
|
import { db } from "../../db/index";
|
|
import { sql } from "drizzle-orm";
|
|
|
|
const router = Router();
|
|
|
|
function requireAuth(req: any, res: any, next: any) {
|
|
if (!req.isAuthenticated()) {
|
|
return res.status(401).json({ error: "Authentication required" });
|
|
}
|
|
next();
|
|
}
|
|
|
|
router.use(requireAuth);
|
|
|
|
// ========== CONTACTS ==========
|
|
|
|
router.get("/contacts", async (req: Request, res: Response) => {
|
|
try {
|
|
const { type, status, search, limit = 50, offset = 0 } = req.query;
|
|
|
|
let query = sql`SELECT * FROM xos_contacts WHERE 1=1`;
|
|
|
|
if (type) query = sql`${query} AND type = ${type}`;
|
|
if (status) query = sql`${query} AND lead_status = ${status}`;
|
|
if (search) query = sql`${query} AND (name ILIKE ${'%' + search + '%'} OR email ILIKE ${'%' + search + '%'} OR company ILIKE ${'%' + search + '%'})`;
|
|
|
|
query = sql`${query} ORDER BY created_at DESC LIMIT ${parseInt(limit as string)} OFFSET ${parseInt(offset as string)}`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching contacts:", error);
|
|
res.status(500).json({ error: "Failed to fetch contacts" });
|
|
}
|
|
});
|
|
|
|
router.get("/contacts/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const result = await db.execute(sql`SELECT * FROM xos_contacts WHERE id = ${id}`);
|
|
const contact = (result.rows || result)[0];
|
|
|
|
if (!contact) return res.status(404).json({ error: "Contact not found" });
|
|
|
|
// Get related activities
|
|
const activities = await db.execute(sql`
|
|
SELECT * FROM xos_activities WHERE contact_id = ${id} ORDER BY created_at DESC LIMIT 10
|
|
`);
|
|
|
|
// Get related deals
|
|
const deals = await db.execute(sql`
|
|
SELECT d.*, s.name as stage_name FROM xos_deals d
|
|
LEFT JOIN xos_pipeline_stages s ON d.stage_id = s.id
|
|
WHERE d.contact_id = ${id} ORDER BY d.created_at DESC
|
|
`);
|
|
|
|
// Get related conversations
|
|
const conversations = await db.execute(sql`
|
|
SELECT * FROM xos_conversations WHERE contact_id = ${id} ORDER BY created_at DESC LIMIT 5
|
|
`);
|
|
|
|
res.json({
|
|
...contact,
|
|
activities: activities.rows || activities,
|
|
deals: deals.rows || deals,
|
|
conversations: conversations.rows || conversations,
|
|
});
|
|
} catch (error) {
|
|
console.error("Error fetching contact:", error);
|
|
res.status(500).json({ error: "Failed to fetch contact" });
|
|
}
|
|
});
|
|
|
|
router.post("/contacts", async (req: Request, res: Response) => {
|
|
try {
|
|
const { name, email, phone, whatsapp, type, company, position, source, tags, notes } = req.body;
|
|
|
|
if (!name) return res.status(400).json({ error: "Name is required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_contacts (name, email, phone, whatsapp, type, company, position, source, tags, notes)
|
|
VALUES (${name}, ${email || null}, ${phone || null}, ${whatsapp || null}, ${type || 'lead'},
|
|
${company || null}, ${position || null}, ${source || 'manual'}, ${tags || null}, ${notes || null})
|
|
RETURNING *
|
|
`);
|
|
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating contact:", error);
|
|
res.status(500).json({ error: "Failed to create contact" });
|
|
}
|
|
});
|
|
|
|
router.put("/contacts/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const { name, email, phone, whatsapp, type, company, position, lead_status, lead_score, tags, notes, assigned_to } = req.body;
|
|
|
|
const result = await db.execute(sql`
|
|
UPDATE xos_contacts SET
|
|
name = COALESCE(${name}, name),
|
|
email = COALESCE(${email}, email),
|
|
phone = COALESCE(${phone}, phone),
|
|
whatsapp = COALESCE(${whatsapp}, whatsapp),
|
|
type = COALESCE(${type}, type),
|
|
company = COALESCE(${company}, company),
|
|
position = COALESCE(${position}, position),
|
|
lead_status = COALESCE(${lead_status}, lead_status),
|
|
lead_score = COALESCE(${lead_score ? parseInt(lead_score) : null}, lead_score),
|
|
tags = COALESCE(${tags}, tags),
|
|
notes = COALESCE(${notes}, notes),
|
|
assigned_to = COALESCE(${assigned_to}, assigned_to),
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE id = ${id}
|
|
RETURNING *
|
|
`);
|
|
|
|
res.json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error updating contact:", error);
|
|
res.status(500).json({ error: "Failed to update contact" });
|
|
}
|
|
});
|
|
|
|
router.delete("/contacts/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
await db.execute(sql`DELETE FROM xos_contacts WHERE id = ${id}`);
|
|
res.json({ success: true });
|
|
} catch (error) {
|
|
console.error("Error deleting contact:", error);
|
|
res.status(500).json({ error: "Failed to delete contact" });
|
|
}
|
|
});
|
|
|
|
// ========== COMPANIES ==========
|
|
|
|
router.get("/companies", async (req: Request, res: Response) => {
|
|
try {
|
|
const { search, industry, limit = 50 } = req.query;
|
|
|
|
let query = sql`SELECT * FROM xos_companies WHERE 1=1`;
|
|
if (search) query = sql`${query} AND (name ILIKE ${'%' + search + '%'} OR domain ILIKE ${'%' + search + '%'})`;
|
|
if (industry) query = sql`${query} AND industry = ${industry}`;
|
|
query = sql`${query} ORDER BY created_at DESC LIMIT ${parseInt(limit as string)}`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching companies:", error);
|
|
res.status(500).json({ error: "Failed to fetch companies" });
|
|
}
|
|
});
|
|
|
|
router.post("/companies", async (req: Request, res: Response) => {
|
|
try {
|
|
const { name, trade_name, document, domain, industry, size, phone, email, website, address, city, state } = req.body;
|
|
|
|
if (!name) return res.status(400).json({ error: "Name is required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_companies (name, trade_name, document, domain, industry, size, phone, email, website, address, city, state)
|
|
VALUES (${name}, ${trade_name || null}, ${document || null}, ${domain || null}, ${industry || null},
|
|
${size || null}, ${phone || null}, ${email || null}, ${website || null}, ${address || null}, ${city || null}, ${state || null})
|
|
RETURNING *
|
|
`);
|
|
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating company:", error);
|
|
res.status(500).json({ error: "Failed to create company" });
|
|
}
|
|
});
|
|
|
|
// ========== PIPELINES & DEALS ==========
|
|
|
|
router.get("/pipelines", async (req: Request, res: Response) => {
|
|
try {
|
|
const pipelines = await db.execute(sql`
|
|
SELECT p.*,
|
|
(SELECT json_agg(s ORDER BY s.sort_order) FROM xos_pipeline_stages s WHERE s.pipeline_id = p.id) as stages
|
|
FROM xos_pipelines p
|
|
WHERE p.is_active = true
|
|
ORDER BY p.is_default DESC, p.name
|
|
`);
|
|
|
|
res.json(pipelines.rows || pipelines);
|
|
} catch (error) {
|
|
console.error("Error fetching pipelines:", error);
|
|
res.status(500).json({ error: "Failed to fetch pipelines" });
|
|
}
|
|
});
|
|
|
|
router.get("/deals", async (req: Request, res: Response) => {
|
|
try {
|
|
const { pipeline_id, stage_id, status, assigned_to } = req.query;
|
|
|
|
let query = sql`
|
|
SELECT d.*,
|
|
c.name as contact_name, c.email as contact_email, c.avatar_url as contact_avatar,
|
|
co.name as company_name,
|
|
s.name as stage_name, s.color as stage_color
|
|
FROM xos_deals d
|
|
LEFT JOIN xos_contacts c ON d.contact_id = c.id
|
|
LEFT JOIN xos_companies co ON d.company_id = co.id
|
|
LEFT JOIN xos_pipeline_stages s ON d.stage_id = s.id
|
|
WHERE 1=1
|
|
`;
|
|
|
|
if (pipeline_id) query = sql`${query} AND d.pipeline_id = ${parseInt(pipeline_id as string)}`;
|
|
if (stage_id) query = sql`${query} AND d.stage_id = ${parseInt(stage_id as string)}`;
|
|
if (status) query = sql`${query} AND d.status = ${status}`;
|
|
if (assigned_to) query = sql`${query} AND d.assigned_to = ${assigned_to}`;
|
|
|
|
query = sql`${query} ORDER BY d.created_at DESC`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching deals:", error);
|
|
res.status(500).json({ error: "Failed to fetch deals" });
|
|
}
|
|
});
|
|
|
|
router.post("/deals", async (req: Request, res: Response) => {
|
|
try {
|
|
const { pipeline_id, stage_id, contact_id, company_id, title, value, expected_close_date, assigned_to, notes } = req.body;
|
|
|
|
if (!title || !pipeline_id || !stage_id) {
|
|
return res.status(400).json({ error: "Title, pipeline_id and stage_id are required" });
|
|
}
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_deals (pipeline_id, stage_id, contact_id, company_id, title, value, expected_close_date, assigned_to, notes)
|
|
VALUES (${parseInt(pipeline_id)}, ${parseInt(stage_id)}, ${contact_id ? parseInt(contact_id) : null},
|
|
${company_id ? parseInt(company_id) : null}, ${title}, ${parseFloat(value) || 0},
|
|
${expected_close_date || null}, ${assigned_to || null}, ${notes || null})
|
|
RETURNING *
|
|
`);
|
|
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating deal:", error);
|
|
res.status(500).json({ error: "Failed to create deal" });
|
|
}
|
|
});
|
|
|
|
router.put("/deals/:id/stage", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const { stage_id } = req.body;
|
|
if (!stage_id) return res.status(400).json({ error: "stage_id is required" });
|
|
|
|
// Check if stage is won or lost
|
|
const stageResult = await db.execute(sql`SELECT is_won, is_lost FROM xos_pipeline_stages WHERE id = ${parseInt(stage_id)}`);
|
|
const stage = (stageResult.rows || stageResult)[0];
|
|
|
|
let status = 'open';
|
|
let closedAt = null;
|
|
if (stage?.is_won) {
|
|
status = 'won';
|
|
closedAt = sql`CURRENT_TIMESTAMP`;
|
|
} else if (stage?.is_lost) {
|
|
status = 'lost';
|
|
closedAt = sql`CURRENT_TIMESTAMP`;
|
|
}
|
|
|
|
const result = await db.execute(sql`
|
|
UPDATE xos_deals SET
|
|
stage_id = ${parseInt(stage_id)},
|
|
status = ${status},
|
|
closed_at = ${closedAt},
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE id = ${id}
|
|
RETURNING *
|
|
`);
|
|
|
|
res.json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error updating deal stage:", error);
|
|
res.status(500).json({ error: "Failed to update deal stage" });
|
|
}
|
|
});
|
|
|
|
router.put("/deals/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const { title, value, notes, expected_close_date } = req.body;
|
|
|
|
const result = await db.execute(sql`
|
|
UPDATE xos_deals SET
|
|
title = COALESCE(${title}, title),
|
|
value = COALESCE(${value ? parseFloat(value) : null}, value),
|
|
notes = COALESCE(${notes}, notes),
|
|
expected_close_date = ${expected_close_date || null},
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE id = ${id}
|
|
RETURNING *
|
|
`);
|
|
|
|
res.json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error updating deal:", error);
|
|
res.status(500).json({ error: "Failed to update deal" });
|
|
}
|
|
});
|
|
|
|
router.delete("/deals/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
await db.execute(sql`DELETE FROM xos_deals WHERE id = ${id}`);
|
|
|
|
res.json({ success: true });
|
|
} catch (error) {
|
|
console.error("Error deleting deal:", error);
|
|
res.status(500).json({ error: "Failed to delete deal" });
|
|
}
|
|
});
|
|
|
|
// ========== CONVERSATIONS & INBOX ==========
|
|
|
|
router.get("/conversations", async (req: Request, res: Response) => {
|
|
try {
|
|
const { status, channel, assigned_to, queue_id, limit = 50, offset = 0 } = req.query;
|
|
|
|
let query = sql`
|
|
SELECT cv.*,
|
|
c.name as contact_name, c.email as contact_email, c.avatar_url as contact_avatar, c.whatsapp as contact_whatsapp,
|
|
q.name as queue_name, q.color as queue_color,
|
|
(SELECT content FROM xos_messages WHERE conversation_id = cv.id ORDER BY created_at DESC LIMIT 1) as last_message
|
|
FROM xos_conversations cv
|
|
LEFT JOIN xos_contacts c ON cv.contact_id = c.id
|
|
LEFT JOIN xos_queues q ON cv.queue_id = q.id
|
|
WHERE 1=1
|
|
`;
|
|
|
|
if (status) query = sql`${query} AND cv.status = ${status}`;
|
|
if (channel) query = sql`${query} AND cv.channel = ${channel}`;
|
|
if (assigned_to) query = sql`${query} AND cv.assigned_to = ${assigned_to}`;
|
|
if (queue_id) query = sql`${query} AND cv.queue_id = ${parseInt(queue_id as string)}`;
|
|
|
|
query = sql`${query} ORDER BY cv.updated_at DESC LIMIT ${parseInt(limit as string)} OFFSET ${parseInt(offset as string)}`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching conversations:", error);
|
|
res.status(500).json({ error: "Failed to fetch conversations" });
|
|
}
|
|
});
|
|
|
|
router.get("/conversations/:id/messages", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const messages = await db.execute(sql`
|
|
SELECT * FROM xos_messages WHERE conversation_id = ${id} ORDER BY created_at ASC
|
|
`);
|
|
|
|
res.json(messages.rows || messages);
|
|
} catch (error) {
|
|
console.error("Error fetching messages:", error);
|
|
res.status(500).json({ error: "Failed to fetch messages" });
|
|
}
|
|
});
|
|
|
|
// ========== TICKETS ==========
|
|
|
|
router.get("/tickets", async (req: Request, res: Response) => {
|
|
try {
|
|
const { status, priority, assigned_to, limit = 50, offset = 0 } = req.query;
|
|
|
|
let query = sql`
|
|
SELECT t.*,
|
|
c.name as contact_name, c.email as contact_email
|
|
FROM xos_tickets t
|
|
LEFT JOIN xos_contacts c ON t.contact_id = c.id
|
|
WHERE 1=1
|
|
`;
|
|
|
|
if (status) query = sql`${query} AND t.status = ${status}`;
|
|
if (priority) query = sql`${query} AND t.priority = ${priority}`;
|
|
if (assigned_to) query = sql`${query} AND t.assigned_to = ${assigned_to}`;
|
|
|
|
query = sql`${query} ORDER BY t.created_at DESC LIMIT ${parseInt(limit as string)} OFFSET ${parseInt(offset as string)}`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching tickets:", error);
|
|
res.status(500).json({ error: "Failed to fetch tickets" });
|
|
}
|
|
});
|
|
|
|
router.post("/tickets", async (req: Request, res: Response) => {
|
|
try {
|
|
const { contact_id, subject, description, category, priority } = req.body;
|
|
|
|
if (!subject) return res.status(400).json({ error: "Subject is required" });
|
|
|
|
// Generate ticket number
|
|
const ticketNumber = `TK-${Date.now().toString(36).toUpperCase()}`;
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_tickets (ticket_number, contact_id, subject, description, category, priority)
|
|
VALUES (${ticketNumber}, ${contact_id ? parseInt(contact_id) : null}, ${subject}, ${description || null}, ${category || null}, ${priority || 'normal'})
|
|
RETURNING *
|
|
`);
|
|
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating ticket:", error);
|
|
res.status(500).json({ error: "Failed to create ticket" });
|
|
}
|
|
});
|
|
|
|
// ========== ACTIVITIES ==========
|
|
|
|
router.get("/activities", async (req: Request, res: Response) => {
|
|
try {
|
|
const { contact_id, deal_id, type, status, limit = 50, offset = 0 } = req.query;
|
|
|
|
let query = sql`
|
|
SELECT a.*,
|
|
c.name as contact_name,
|
|
d.title as deal_title
|
|
FROM xos_activities a
|
|
LEFT JOIN xos_contacts c ON a.contact_id = c.id
|
|
LEFT JOIN xos_deals d ON a.deal_id = d.id
|
|
WHERE 1=1
|
|
`;
|
|
|
|
if (contact_id) query = sql`${query} AND a.contact_id = ${parseInt(contact_id as string)}`;
|
|
if (deal_id) query = sql`${query} AND a.deal_id = ${parseInt(deal_id as string)}`;
|
|
if (type) query = sql`${query} AND a.type = ${type}`;
|
|
if (status) query = sql`${query} AND a.status = ${status}`;
|
|
|
|
query = sql`${query} ORDER BY a.due_at ASC NULLS LAST, a.created_at DESC LIMIT ${parseInt(limit as string)} OFFSET ${parseInt(offset as string)}`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching activities:", error);
|
|
res.status(500).json({ error: "Failed to fetch activities" });
|
|
}
|
|
});
|
|
|
|
router.post("/activities", async (req: Request, res: Response) => {
|
|
try {
|
|
const { type, title, description, due_at, contact_id, company_id, deal_id, assigned_to, priority } = req.body;
|
|
|
|
if (!type || !title) return res.status(400).json({ error: "Type and title are required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_activities (type, title, description, due_at, contact_id, company_id, deal_id, assigned_to, priority)
|
|
VALUES (${type}, ${title}, ${description || null}, ${due_at || null},
|
|
${contact_id ? parseInt(contact_id) : null}, ${company_id ? parseInt(company_id) : null},
|
|
${deal_id ? parseInt(deal_id) : null}, ${assigned_to || null}, ${priority || 'normal'})
|
|
RETURNING *
|
|
`);
|
|
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating activity:", error);
|
|
res.status(500).json({ error: "Failed to create activity" });
|
|
}
|
|
});
|
|
|
|
// ========== DASHBOARD STATS ==========
|
|
|
|
router.get("/stats", async (req: Request, res: Response) => {
|
|
try {
|
|
const stats = await db.execute(sql`
|
|
SELECT
|
|
(SELECT COUNT(*) FROM xos_contacts) as total_contacts,
|
|
(SELECT COUNT(*) FROM xos_contacts WHERE type = 'lead') as total_leads,
|
|
(SELECT COUNT(*) FROM xos_contacts WHERE type = 'customer') as total_customers,
|
|
(SELECT COUNT(*) FROM xos_companies) as total_companies,
|
|
(SELECT COUNT(*) FROM xos_deals WHERE status = 'open') as open_deals,
|
|
(SELECT COUNT(*) FROM xos_deals WHERE status = 'won') as won_deals,
|
|
(SELECT COALESCE(SUM(value), 0) FROM xos_deals WHERE status = 'open') as pipeline_value,
|
|
(SELECT COALESCE(SUM(value), 0) FROM xos_deals WHERE status = 'won') as won_value,
|
|
(SELECT COUNT(*) FROM xos_conversations WHERE status = 'open') as open_conversations,
|
|
(SELECT COUNT(*) FROM xos_tickets WHERE status NOT IN ('resolved', 'closed')) as open_tickets,
|
|
(SELECT COUNT(*) FROM xos_activities WHERE status = 'pending' AND due_at <= CURRENT_TIMESTAMP) as overdue_activities
|
|
`);
|
|
|
|
res.json((stats.rows || stats)[0]);
|
|
} catch (error) {
|
|
console.error("Error fetching stats:", error);
|
|
res.status(500).json({ error: "Failed to fetch stats" });
|
|
}
|
|
});
|
|
|
|
// ========== QUEUES (FILAS) ==========
|
|
|
|
router.get("/queues", async (req: Request, res: Response) => {
|
|
try {
|
|
const result = await db.execute(sql`
|
|
SELECT q.*,
|
|
(SELECT COUNT(*) FROM xos_queue_users qu WHERE qu.queue_id = q.id AND qu.is_active = true) as users_count,
|
|
(SELECT COUNT(*) FROM xos_conversations c WHERE c.queue_id = q.id AND c.status = 'open') as open_conversations
|
|
FROM xos_queues q
|
|
WHERE q.is_active = true
|
|
ORDER BY q.order_priority
|
|
`);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching queues:", error);
|
|
res.status(500).json({ error: "Failed to fetch queues" });
|
|
}
|
|
});
|
|
|
|
router.post("/queues", async (req: Request, res: Response) => {
|
|
try {
|
|
const { name, color, greetingMessage, outOfHoursMessage, schedules, orderPriority } = req.body;
|
|
if (!name) return res.status(400).json({ error: "Name is required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_queues (name, color, greeting_message, out_of_hours_message, schedules, order_priority)
|
|
VALUES (${name}, ${color || 'blue'}, ${greetingMessage || null}, ${outOfHoursMessage || null},
|
|
${schedules ? JSON.stringify(schedules) : null}, ${orderPriority || 0})
|
|
RETURNING *
|
|
`);
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating queue:", error);
|
|
res.status(500).json({ error: "Failed to create queue" });
|
|
}
|
|
});
|
|
|
|
// ========== INTERNAL NOTES (NOTAS INTERNAS) ==========
|
|
|
|
router.get("/notes", async (req: Request, res: Response) => {
|
|
try {
|
|
const { conversationId, contactId, ticketId } = req.query;
|
|
|
|
let query = sql`SELECT * FROM xos_internal_notes WHERE 1=1`;
|
|
if (conversationId) query = sql`${query} AND conversation_id = ${parseInt(conversationId as string)}`;
|
|
if (contactId) query = sql`${query} AND contact_id = ${parseInt(contactId as string)}`;
|
|
if (ticketId) query = sql`${query} AND ticket_id = ${parseInt(ticketId as string)}`;
|
|
query = sql`${query} ORDER BY is_pinned DESC, created_at DESC`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching notes:", error);
|
|
res.status(500).json({ error: "Failed to fetch notes" });
|
|
}
|
|
});
|
|
|
|
router.post("/notes", async (req: Request, res: Response) => {
|
|
try {
|
|
const { conversationId, contactId, ticketId, content, userName } = req.body;
|
|
if (!content) return res.status(400).json({ error: "Content is required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_internal_notes (conversation_id, contact_id, ticket_id, content, user_name)
|
|
VALUES (${conversationId || null}, ${contactId || null}, ${ticketId || null}, ${content}, ${userName || 'Usuário'})
|
|
RETURNING *
|
|
`);
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating note:", error);
|
|
res.status(500).json({ error: "Failed to create note" });
|
|
}
|
|
});
|
|
|
|
router.delete("/notes/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
await db.execute(sql`DELETE FROM xos_internal_notes WHERE id = ${id}`);
|
|
res.json({ success: true });
|
|
} catch (error) {
|
|
console.error("Error deleting note:", error);
|
|
res.status(500).json({ error: "Failed to delete note" });
|
|
}
|
|
});
|
|
|
|
// ========== QUICK MESSAGES (MENSAGENS RÁPIDAS) ==========
|
|
|
|
router.get("/quick-messages", async (req: Request, res: Response) => {
|
|
try {
|
|
const { search } = req.query;
|
|
let query = sql`SELECT * FROM xos_quick_messages WHERE 1=1`;
|
|
if (search) query = sql`${query} AND (shortcode ILIKE ${'%' + search + '%'} OR title ILIKE ${'%' + search + '%'} OR content ILIKE ${'%' + search + '%'})`;
|
|
query = sql`${query} ORDER BY usage_count DESC, shortcode`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching quick messages:", error);
|
|
res.status(500).json({ error: "Failed to fetch quick messages" });
|
|
}
|
|
});
|
|
|
|
router.post("/quick-messages", async (req: Request, res: Response) => {
|
|
try {
|
|
const { shortcode, title, content, mediaUrl, mediaType, scope } = req.body;
|
|
if (!shortcode || !content) return res.status(400).json({ error: "Shortcode and content required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_quick_messages (shortcode, title, content, media_url, media_type, scope)
|
|
VALUES (${shortcode}, ${title || null}, ${content}, ${mediaUrl || null}, ${mediaType || null}, ${scope || 'company'})
|
|
RETURNING *
|
|
`);
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating quick message:", error);
|
|
res.status(500).json({ error: "Failed to create quick message" });
|
|
}
|
|
});
|
|
|
|
router.put("/quick-messages/:id/use", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const result = await db.execute(sql`
|
|
UPDATE xos_quick_messages SET usage_count = usage_count + 1 WHERE id = ${id} RETURNING *
|
|
`);
|
|
res.json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error updating quick message:", error);
|
|
res.status(500).json({ error: "Failed to update quick message" });
|
|
}
|
|
});
|
|
|
|
// ========== CONVERSATION TRANSFER ==========
|
|
|
|
router.put("/conversations/:id/transfer", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
|
|
const { queueId, assignedTo } = req.body;
|
|
|
|
const result = await db.execute(sql`
|
|
UPDATE xos_conversations SET
|
|
queue_id = COALESCE(${queueId ? parseInt(queueId) : null}, queue_id),
|
|
assigned_to = COALESCE(${assignedTo || null}, assigned_to),
|
|
updated_at = CURRENT_TIMESTAMP
|
|
WHERE id = ${id}
|
|
RETURNING *
|
|
`);
|
|
res.json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error transferring conversation:", error);
|
|
res.status(500).json({ error: "Failed to transfer conversation" });
|
|
}
|
|
});
|
|
|
|
// ========== SCHEDULED MESSAGES ==========
|
|
|
|
router.get("/scheduled-messages", async (req: Request, res: Response) => {
|
|
try {
|
|
const { contactId, status } = req.query;
|
|
let query = sql`SELECT sm.*, c.name as contact_name FROM xos_scheduled_messages sm
|
|
LEFT JOIN xos_contacts c ON sm.contact_id = c.id WHERE 1=1`;
|
|
if (contactId) query = sql`${query} AND sm.contact_id = ${parseInt(contactId as string)}`;
|
|
if (status) query = sql`${query} AND sm.status = ${status}`;
|
|
query = sql`${query} ORDER BY sm.scheduled_at`;
|
|
|
|
const result = await db.execute(query);
|
|
res.json(result.rows || result);
|
|
} catch (error) {
|
|
console.error("Error fetching scheduled messages:", error);
|
|
res.status(500).json({ error: "Failed to fetch scheduled messages" });
|
|
}
|
|
});
|
|
|
|
router.post("/scheduled-messages", async (req: Request, res: Response) => {
|
|
try {
|
|
const { contactId, conversationId, content, scheduledAt, mediaUrl, mediaType } = req.body;
|
|
if (!content || !scheduledAt) return res.status(400).json({ error: "Content and scheduledAt required" });
|
|
|
|
const result = await db.execute(sql`
|
|
INSERT INTO xos_scheduled_messages (contact_id, conversation_id, content, scheduled_at, media_url, media_type)
|
|
VALUES (${contactId || null}, ${conversationId || null}, ${content}, ${scheduledAt}, ${mediaUrl || null}, ${mediaType || null})
|
|
RETURNING *
|
|
`);
|
|
res.status(201).json((result.rows || result)[0]);
|
|
} catch (error) {
|
|
console.error("Error creating scheduled message:", error);
|
|
res.status(500).json({ error: "Failed to create scheduled message" });
|
|
}
|
|
});
|
|
|
|
router.delete("/scheduled-messages/:id", async (req: Request, res: Response) => {
|
|
try {
|
|
const id = parseInt(req.params.id);
|
|
if (isNaN(id)) return res.status(400).json({ error: "Invalid ID" });
|
|
await db.execute(sql`UPDATE xos_scheduled_messages SET status = 'cancelled' WHERE id = ${id}`);
|
|
res.json({ success: true });
|
|
} catch (error) {
|
|
console.error("Error cancelling scheduled message:", error);
|
|
res.status(500).json({ error: "Failed to cancel scheduled message" });
|
|
}
|
|
});
|
|
|
|
export default router;
|