Backend: - Add MCP Serializer to centralize entity-to-array conversion (~300 lines deduped) - Fix race condition in task/ticket number generation (SELECT FOR UPDATE + transaction) - Add unique constraint on task (project_id, number) with migration - Fix MIME type validation: use server-detected finfo instead of client-supplied type - Add allowlist of permitted MIME types for uploads - Fix TaskDocumentDownloadController: allow ROLE_CLIENT access, add priority:1 - Fix notification sent even when ticket status unchanged - Remove redundant exception constructors - Simplify services (BookStackApi double fetch, TokenEncryptor, GiteaApi) - Consolidate duplicate checks in processors Frontend: - Fix useApi isHandlingUnauthorized scope (module-level to prevent double 401 redirect) - Fix client-tickets toast key copy-paste bug - Merge duplicated tasks service methods (getByProject + getByProjectArchived) - Extract shared uploadWithRelation helper in task-documents service - Extract formatFileSize utility from duplicated component code - Extract status transition logic into useClientTicketHelpers composable - Remove dead code (unused router, handleLogout, empty script blocks) - Merge duplicate watchers and onMounted calls - Normalize arrow functions to function declarations per convention Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
57 lines
2.0 KiB
TypeScript
57 lines
2.0 KiB
TypeScript
import type { TaskDocument } from './dto/task-document'
|
|
import type { HydraCollection } from '~/utils/api'
|
|
import { extractHydraMembers } from '~/utils/api'
|
|
import { $fetch } from 'ofetch'
|
|
|
|
export function useTaskDocumentService() {
|
|
const api = useApi()
|
|
const config = useRuntimeConfig()
|
|
const baseURL = config.public.apiBase || '/api'
|
|
|
|
async function getByTask(taskId: number): Promise<TaskDocument[]> {
|
|
const data = await api.get<HydraCollection<TaskDocument>>('/task_documents', {
|
|
task: `/api/tasks/${taskId}`,
|
|
})
|
|
return extractHydraMembers(data)
|
|
}
|
|
|
|
async function uploadWithRelation(relationField: string, relationIri: string, file: File): Promise<TaskDocument> {
|
|
const formData = new FormData()
|
|
formData.append('file', file)
|
|
formData.append(relationField, relationIri)
|
|
|
|
return $fetch<TaskDocument>(`${baseURL}/task_documents`, {
|
|
method: 'POST',
|
|
body: formData,
|
|
credentials: 'include',
|
|
})
|
|
}
|
|
|
|
async function upload(taskId: number, file: File): Promise<TaskDocument> {
|
|
return uploadWithRelation('task', `/api/tasks/${taskId}`, file)
|
|
}
|
|
|
|
async function uploadForTicket(clientTicketId: number, file: File): Promise<TaskDocument> {
|
|
return uploadWithRelation('clientTicket', `/api/client_tickets/${clientTicketId}`, file)
|
|
}
|
|
|
|
async function getByTicket(clientTicketId: number): Promise<TaskDocument[]> {
|
|
const data = await api.get<HydraCollection<TaskDocument>>('/task_documents', {
|
|
clientTicket: `/api/client_tickets/${clientTicketId}`,
|
|
})
|
|
return extractHydraMembers(data)
|
|
}
|
|
|
|
async function remove(id: number): Promise<void> {
|
|
await api.delete(`/task_documents/${id}`, {}, {
|
|
toastSuccessKey: 'taskDocuments.deleted',
|
|
})
|
|
}
|
|
|
|
function getDownloadUrl(id: number): string {
|
|
return `${baseURL}/task_documents/${id}/download`
|
|
}
|
|
|
|
return { getByTask, upload, uploadForTicket, getByTicket, remove, getDownloadUrl }
|
|
}
|