From b87081dd2cfd48c37641ea653c25b61a7ac9c395 Mon Sep 17 00:00:00 2001 From: BOHA Date: Mon, 23 Mar 2026 10:19:15 +0100 Subject: [PATCH] feat: integrate NAS file operations with project CRUD Co-Authored-By: Claude Opus 4.6 (1M context) --- src/routes/admin/projects.ts | 4 +++- src/services/projects.service.ts | 25 +++++++++++++++++++++++-- 2 files changed, 26 insertions(+), 3 deletions(-) diff --git a/src/routes/admin/projects.ts b/src/routes/admin/projects.ts index edba8db..69ebe15 100644 --- a/src/routes/admin/projects.ts +++ b/src/routes/admin/projects.ts @@ -97,7 +97,9 @@ export default async function projectsRoutes(fastify: FastifyInstance): Promise< const id = parseId(request.params.id, reply); if (id === null) return; - const result = await deleteProject(id); + const body = request.body as Record; + const deleteFiles = !!body?.delete_files; + const result = await deleteProject(id, deleteFiles); if (result && 'error' in result) { if (result.error === 'not_found') return error(reply, 'Projekt nenalezen', 404); if (result.error === 'has_order') return error(reply, 'Nelze smazat projekt propojený s objednávkou. Nejdříve smažte objednávku.', 400); diff --git a/src/services/projects.service.ts b/src/services/projects.service.ts index 92791b7..9b46b8b 100644 --- a/src/services/projects.service.ts +++ b/src/services/projects.service.ts @@ -1,5 +1,8 @@ import prisma from '../config/database'; import { generateSharedNumber } from './numbering.service'; +import { NasFileManager } from './nas-file-manager'; + +const nasFileManager = new NasFileManager(); const ALLOWED_SORT_FIELDS = ['id', 'project_number', 'name', 'status', 'created_at']; @@ -50,7 +53,11 @@ export async function getProject(id: number) { where: { id }, include: { customers: true, users: true, quotations: true, orders: true, project_notes: { orderBy: { created_at: 'desc' } } }, }); - return project || null; + if (!project) return null; + return { + ...project, + has_nas_folder: project.project_number ? nasFileManager.projectFolderExists(project.project_number) : false, + }; } export async function createProject(body: Record) { @@ -68,6 +75,11 @@ export async function createProject(body: Record) { notes: body.notes ? String(body.notes) : null, }, }); + + if (project.project_number && nasFileManager.isConfigured()) { + nasFileManager.createProjectFolder(project.project_number, project.name || ''); + } + return project; } @@ -86,14 +98,23 @@ export async function updateProject(id: number, body: Record) { if (body.end_date !== undefined) data.end_date = body.end_date ? new Date(String(body.end_date)) : null; await prisma.projects.update({ where: { id }, data }); + + if (existing.name !== data.name && existing.project_number && nasFileManager.isConfigured()) { + nasFileManager.renameProjectFolder(existing.project_number, String(data.name || '')); + } + return existing; } -export async function deleteProject(id: number) { +export async function deleteProject(id: number, deleteFiles: boolean = false) { const existing = await prisma.projects.findUnique({ where: { id } }); if (!existing) return { error: 'not_found' as const }; if (existing.order_id) return { error: 'has_order' as const }; + if (deleteFiles && existing.project_number && nasFileManager.isConfigured()) { + await nasFileManager.deleteProjectFolder(existing.project_number); + } + await prisma.projects.delete({ where: { id } }); return existing; }