Initial commit: Port Nimara CRM (Layers 0-4)
Some checks failed
Build & Push Docker Images / build-and-push (push) Has been cancelled
Build & Push Docker Images / deploy (push) Has been cancelled
Build & Push Docker Images / lint (push) Has been cancelled

Full CRM rebuild with Next.js 15, TypeScript, Tailwind, Drizzle ORM,
PostgreSQL, Redis, BullMQ, MinIO, and Socket.io. Includes 461 source
files covering clients, berths, interests/pipeline, documents/EOI,
expenses/invoices, email, notifications, dashboard, admin, and
client portal. CI/CD via Gitea Actions with Docker builds.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-03-26 11:52:51 +01:00
commit 67d7e6e3d5
572 changed files with 86496 additions and 0 deletions

View File

@@ -0,0 +1,113 @@
import { NextResponse } from 'next/server';
import { and, eq, desc, inArray } from 'drizzle-orm';
import { withAuth, withPermission } from '@/lib/api/helpers';
import { errorResponse, NotFoundError } from '@/lib/errors';
import { db } from '@/lib/db';
import { interests } from '@/lib/db/schema/interests';
import { auditLogs } from '@/lib/db/schema/system';
import { documents, documentEvents } from '@/lib/db/schema/documents';
interface TimelineEvent {
id: string;
type: 'audit' | 'document_event';
action: string;
description: string;
userId: string | null;
createdAt: Date;
metadata: Record<string, unknown>;
}
// GET /api/v1/interests/[id]/timeline
export const GET = withAuth(
withPermission('interests', 'view', async (req, ctx, params) => {
try {
const interestId = params.id!;
const interest = await db.query.interests.findFirst({
where: and(eq(interests.id, interestId), eq(interests.portId, ctx.portId)),
});
if (!interest) throw new NotFoundError('Interest');
// Fetch audit logs for this interest
const auditRows = await db
.select()
.from(auditLogs)
.where(
and(
eq(auditLogs.entityType, 'interest'),
eq(auditLogs.entityId, interestId),
),
)
.orderBy(desc(auditLogs.createdAt))
.limit(50);
// Fetch document events for documents linked to this interest
const interestDocs = await db
.select({ id: documents.id, title: documents.title })
.from(documents)
.where(eq(documents.interestId, interestId));
const docIds = interestDocs.map((d) => d.id);
const docEventRows =
docIds.length > 0
? await db
.select({
id: documentEvents.id,
documentId: documentEvents.documentId,
eventType: documentEvents.eventType,
eventData: documentEvents.eventData,
createdAt: documentEvents.createdAt,
})
.from(documentEvents)
.where(inArray(documentEvents.documentId, docIds))
.orderBy(desc(documentEvents.createdAt))
.limit(50)
: [];
const docTitles = Object.fromEntries(interestDocs.map((d) => [d.id, d.title]));
// Union and sort
const auditEvents: TimelineEvent[] = auditRows.map((row) => ({
id: row.id,
type: 'audit',
action: row.action,
description: buildAuditDescription(row.action, row.newValue as Record<string, unknown> | null),
userId: row.userId,
createdAt: row.createdAt,
metadata: (row.metadata as Record<string, unknown>) ?? {},
}));
const docEvents: TimelineEvent[] = docEventRows.map((row) => ({
id: row.id,
type: 'document_event',
action: row.eventType,
description: `Document "${docTitles[row.documentId] ?? row.documentId}": ${row.eventType}`,
userId: null,
createdAt: row.createdAt,
metadata: (row.eventData as Record<string, unknown>) ?? {},
}));
const allEvents = [...auditEvents, ...docEvents];
allEvents.sort((a, b) => new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime());
return NextResponse.json({ data: allEvents.slice(0, 50) });
} catch (error) {
return errorResponse(error);
}
}),
);
function buildAuditDescription(
action: string,
newValue: Record<string, unknown> | null,
): string {
if (action === 'create') return 'Interest created';
if (action === 'archive') return 'Interest archived';
if (action === 'restore') return 'Interest restored';
if (action === 'update' && newValue?.pipelineStage) {
return `Stage changed to "${newValue.pipelineStage}"`;
}
if (action === 'update') return 'Interest updated';
return action;
}