Initial commit: Port Nimara CRM (Layers 0-4)
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>
2026-03-26 11:52:51 +01:00
|
|
|
|
import { buildStoragePath } from '@/lib/minio';
|
|
|
|
|
|
import { MIME_TO_EXT } from '@/lib/constants/file-validation';
|
|
|
|
|
|
|
|
|
|
|
|
export function generateStorageKey(
|
|
|
|
|
|
portSlug: string,
|
|
|
|
|
|
entity: string,
|
|
|
|
|
|
entityId: string,
|
|
|
|
|
|
mimeType: string,
|
|
|
|
|
|
): string {
|
|
|
|
|
|
const fileId = crypto.randomUUID();
|
|
|
|
|
|
const extension = MIME_TO_EXT[mimeType] ?? 'bin';
|
|
|
|
|
|
return buildStoragePath(portSlug, entity, entityId, fileId, extension);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function sanitizeFilename(name: string): string {
|
2026-05-13 12:58:58 +02:00
|
|
|
|
return (
|
|
|
|
|
|
name
|
|
|
|
|
|
.normalize('NFC')
|
|
|
|
|
|
.replace(/[/\\:]/g, '') // strip path chars
|
|
|
|
|
|
.replace(/\x00/g, '') // strip null bytes
|
|
|
|
|
|
.replace(/[\x01-\x1f\x7f]/g, '') // strip control chars
|
|
|
|
|
|
// asset-auditor M2: strip Unicode bidi-control + zero-width
|
|
|
|
|
|
// characters that enable the classic Windows-icon spoof
|
|
|
|
|
|
// (`invoice_fdp.exe` renders as `invoice_exe.pdf`) and folder-
|
|
|
|
|
|
// listing collision spoofs (U+200B/U+FEFF).
|
|
|
|
|
|
|
|
|
|
|
|
.replace(/[---]/g, '')
|
|
|
|
|
|
.trim()
|
|
|
|
|
|
.slice(0, 255)
|
|
|
|
|
|
);
|
Initial commit: Port Nimara CRM (Layers 0-4)
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>
2026-03-26 11:52:51 +01:00
|
|
|
|
}
|