2025-06-15 16:32:34 +02:00
|
|
|
import { requireAuth } from '~/server/utils/auth';
|
2025-06-10 15:27:57 +02:00
|
|
|
import { deleteFile, deleteFolder, getMinioClient } from '~/server/utils/minio';
|
2025-06-04 16:32:50 +02:00
|
|
|
|
|
|
|
|
export default defineEventHandler(async (event) => {
|
2025-06-15 16:32:34 +02:00
|
|
|
// Check authentication (x-tag header OR Keycloak session)
|
|
|
|
|
await requireAuth(event);
|
|
|
|
|
|
2025-06-04 16:32:50 +02:00
|
|
|
try {
|
|
|
|
|
const body = await readBody(event);
|
2025-06-10 15:27:57 +02:00
|
|
|
const { fileName, isFolder, bucket } = body;
|
|
|
|
|
const targetBucket = bucket || 'client-portal';
|
2025-06-04 16:32:50 +02:00
|
|
|
|
2025-06-10 15:42:00 +02:00
|
|
|
console.log('[delete] Delete request:', {
|
|
|
|
|
fileName,
|
|
|
|
|
isFolder,
|
|
|
|
|
bucket,
|
|
|
|
|
targetBucket
|
|
|
|
|
});
|
|
|
|
|
|
2025-06-04 16:32:50 +02:00
|
|
|
if (!fileName) {
|
|
|
|
|
throw createError({
|
|
|
|
|
statusCode: 400,
|
|
|
|
|
statusMessage: 'File name is required',
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 13:59:09 +02:00
|
|
|
// Protect EOIs folder from deletion
|
|
|
|
|
if (fileName === 'EOIs/' || fileName === 'EOIs') {
|
|
|
|
|
throw createError({
|
|
|
|
|
statusCode: 403,
|
|
|
|
|
statusMessage: 'The EOIs folder is protected and cannot be deleted',
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-04 16:32:50 +02:00
|
|
|
// Delete folder or file based on type
|
2025-06-10 15:27:57 +02:00
|
|
|
if (targetBucket === 'client-portal') {
|
|
|
|
|
// Use existing functions for default bucket
|
|
|
|
|
if (isFolder) {
|
|
|
|
|
await deleteFolder(fileName);
|
|
|
|
|
} else {
|
|
|
|
|
await deleteFile(fileName);
|
|
|
|
|
}
|
2025-06-04 16:32:50 +02:00
|
|
|
} else {
|
2025-06-10 15:27:57 +02:00
|
|
|
// For other buckets, use MinIO client directly
|
|
|
|
|
const client = getMinioClient();
|
|
|
|
|
|
|
|
|
|
if (isFolder) {
|
|
|
|
|
// List all objects in the folder and delete them
|
|
|
|
|
const objectsList: string[] = [];
|
|
|
|
|
|
|
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
|
const stream = client.listObjectsV2(targetBucket, fileName, true);
|
|
|
|
|
|
|
|
|
|
stream.on('data', (obj) => {
|
|
|
|
|
if (obj && obj.name) {
|
|
|
|
|
objectsList.push(obj.name);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
stream.on('error', reject);
|
|
|
|
|
|
|
|
|
|
stream.on('end', async () => {
|
|
|
|
|
try {
|
|
|
|
|
if (objectsList.length > 0) {
|
|
|
|
|
await client.removeObjects(targetBucket, objectsList);
|
|
|
|
|
}
|
|
|
|
|
resolve(true);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
reject(error);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
// Delete single file
|
2025-06-10 15:42:00 +02:00
|
|
|
console.log(`[delete] Attempting to delete file '${fileName}' from bucket '${targetBucket}'`);
|
|
|
|
|
|
|
|
|
|
// First check if the file exists
|
|
|
|
|
try {
|
|
|
|
|
const stat = await client.statObject(targetBucket, fileName);
|
|
|
|
|
console.log('[delete] File exists in bucket:', stat);
|
|
|
|
|
} catch (err: any) {
|
|
|
|
|
console.error(`[delete] File '${fileName}' not found in bucket '${targetBucket}':`, err.message);
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 15:27:57 +02:00
|
|
|
await client.removeObject(targetBucket, fileName);
|
2025-06-10 15:42:00 +02:00
|
|
|
console.log(`[delete] Successfully deleted '${fileName}' from '${targetBucket}'`);
|
2025-06-10 15:27:57 +02:00
|
|
|
}
|
2025-06-04 16:32:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Log audit event
|
|
|
|
|
await logAuditEvent(event, 'delete', fileName);
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
success: true,
|
|
|
|
|
message: isFolder ? 'Folder deleted successfully' : 'File deleted successfully',
|
|
|
|
|
};
|
|
|
|
|
} catch (error: any) {
|
|
|
|
|
console.error('Failed to delete:', error);
|
|
|
|
|
throw createError({
|
|
|
|
|
statusCode: 500,
|
|
|
|
|
statusMessage: error.message || 'Failed to delete',
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Audit logging helper
|
|
|
|
|
async function logAuditEvent(event: any, action: string, filePath: string) {
|
|
|
|
|
try {
|
|
|
|
|
const user = event.context.user || { email: 'anonymous' };
|
|
|
|
|
const auditLog = {
|
|
|
|
|
user_email: user.email,
|
|
|
|
|
action,
|
|
|
|
|
file_path: filePath,
|
|
|
|
|
timestamp: new Date().toISOString(),
|
|
|
|
|
ip_address: getClientIP(event),
|
|
|
|
|
success: true,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// You can store this in your database or logging system
|
|
|
|
|
console.log('Audit log:', auditLog);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error('Failed to log audit event:', error);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getClientIP(event: any): string {
|
|
|
|
|
return event.node.req.headers['x-forwarded-for'] ||
|
|
|
|
|
event.node.req.connection.remoteAddress ||
|
|
|
|
|
'unknown';
|
|
|
|
|
}
|