feat: añadir /api/me/tasks/overview y adaptar /app para consumirla
Co-authored-by: aider (openrouter/openai/gpt-5) <aider@aider.chat>webui
parent
0f95e633d6
commit
b5d91d518f
@ -0,0 +1,113 @@
|
||||
import type { RequestHandler } from './$types';
|
||||
import { getDb } from '$lib/server/db';
|
||||
|
||||
export const GET: RequestHandler = async (event) => {
|
||||
// Requiere sesión
|
||||
const userId = event.locals.userId ?? null;
|
||||
if (!userId) {
|
||||
return new Response('Unauthorized', { status: 401 });
|
||||
}
|
||||
|
||||
const url = new URL(event.request.url);
|
||||
const orderParam = (url.searchParams.get('order') || 'due').trim().toLowerCase();
|
||||
const order = orderParam === 'group_then_due' ? 'group_then_due' : 'due';
|
||||
|
||||
const db = await getDb();
|
||||
|
||||
// Orden para "assigned"
|
||||
const assignedOrder =
|
||||
order === 'group_then_due'
|
||||
? `(t.group_id IS NULL) ASC, g.name COLLATE NOCASE ASC, CASE WHEN t.due_date IS NULL THEN 1 ELSE 0 END, t.due_date ASC, t.id ASC`
|
||||
: `CASE WHEN t.due_date IS NULL THEN 1 ELSE 0 END, t.due_date ASC, t.id ASC`;
|
||||
|
||||
// Tareas asignadas al usuario (abiertas)
|
||||
const assignedRows = db
|
||||
.prepare(
|
||||
`SELECT t.id, t.description, t.due_date, t.group_id, t.display_code, g.name AS group_name
|
||||
FROM tasks t
|
||||
LEFT JOIN groups g ON g.id = t.group_id
|
||||
INNER JOIN task_assignments a ON a.task_id = t.id
|
||||
WHERE a.user_id = ?
|
||||
AND COALESCE(t.completed, 0) = 0
|
||||
AND t.completed_at IS NULL
|
||||
AND (
|
||||
t.group_id IS NULL OR (
|
||||
EXISTS (SELECT 1 FROM allowed_groups ag WHERE ag.group_id = t.group_id AND ag.status = 'allowed')
|
||||
AND EXISTS (SELECT 1 FROM group_members gm WHERE gm.group_id = t.group_id AND gm.user_id = ? AND gm.is_active = 1)
|
||||
)
|
||||
)
|
||||
ORDER BY ${assignedOrder}`
|
||||
)
|
||||
.all(userId, userId) as any[];
|
||||
|
||||
const assigned = assignedRows.map((r) => ({
|
||||
id: Number(r.id),
|
||||
description: String(r.description || ''),
|
||||
due_date: r.due_date ? String(r.due_date) : null,
|
||||
group_id: r.group_id ? String(r.group_id) : null,
|
||||
group_name: r.group_name != null ? String(r.group_name) : null, // personales => null
|
||||
display_code: r.display_code != null ? Number(r.display_code) : null,
|
||||
assignees: [] as string[]
|
||||
}));
|
||||
|
||||
// Cargar asignados completos para "assigned"
|
||||
if (assigned.length > 0) {
|
||||
const ids = assigned.map((it) => it.id);
|
||||
const placeholders = ids.map(() => '?').join(',');
|
||||
const assignRows = db
|
||||
.prepare(
|
||||
`SELECT task_id, user_id
|
||||
FROM task_assignments
|
||||
WHERE task_id IN (${placeholders})
|
||||
ORDER BY assigned_at ASC`
|
||||
)
|
||||
.all(...ids) as any[];
|
||||
const map = new Map<number, string[]>();
|
||||
for (const row of assignRows) {
|
||||
const tid = Number(row.task_id);
|
||||
const uid = String(row.user_id);
|
||||
if (!map.has(tid)) map.set(tid, []);
|
||||
map.get(tid)!.push(uid);
|
||||
}
|
||||
for (const it of assigned) {
|
||||
it.assignees = map.get(it.id) || [];
|
||||
}
|
||||
}
|
||||
|
||||
// Orden para "unassigned"
|
||||
const unassignedOrder =
|
||||
order === 'group_then_due'
|
||||
? `(t.group_id IS NULL) ASC, g.name COLLATE NOCASE ASC, CASE WHEN t.due_date IS NULL THEN 1 ELSE 0 END, t.due_date ASC, t.id ASC`
|
||||
: `CASE WHEN t.due_date IS NULL THEN 1 ELSE 0 END, t.due_date ASC, t.id ASC`;
|
||||
|
||||
// Tareas sin responsable (solo de grupos permitidos donde soy miembro activo)
|
||||
const unassignedRows = db
|
||||
.prepare(
|
||||
`SELECT t.id, t.description, t.due_date, t.group_id, t.display_code, g.name AS group_name
|
||||
FROM tasks t
|
||||
LEFT JOIN groups g ON g.id = t.group_id
|
||||
WHERE t.group_id IS NOT NULL
|
||||
AND COALESCE(t.completed, 0) = 0
|
||||
AND t.completed_at IS NULL
|
||||
AND NOT EXISTS (SELECT 1 FROM task_assignments a WHERE a.task_id = t.id)
|
||||
AND EXISTS (SELECT 1 FROM allowed_groups ag WHERE ag.group_id = t.group_id AND ag.status = 'allowed')
|
||||
AND EXISTS (SELECT 1 FROM group_members gm WHERE gm.group_id = t.group_id AND gm.user_id = ? AND gm.is_active = 1)
|
||||
ORDER BY ${unassignedOrder}`
|
||||
)
|
||||
.all(userId) as any[];
|
||||
|
||||
const unassigned = unassignedRows.map((r) => ({
|
||||
id: Number(r.id),
|
||||
description: String(r.description || ''),
|
||||
due_date: r.due_date ? String(r.due_date) : null,
|
||||
group_id: r.group_id ? String(r.group_id) : null,
|
||||
group_name: r.group_name != null ? String(r.group_name) : null,
|
||||
display_code: r.display_code != null ? Number(r.display_code) : null,
|
||||
assignees: [] as string[] // por definición, vacío
|
||||
}));
|
||||
|
||||
return new Response(JSON.stringify({ assigned, unassigned }), {
|
||||
status: 200,
|
||||
headers: { 'content-type': 'application/json; charset=utf-8', 'cache-control': 'no-store' }
|
||||
});
|
||||
};
|
||||
Loading…
Reference in New Issue