feat: granular file access audit logging (viewed/opened/downloaded)
Some checks failed
Build and Push Docker Image / build (push) Has been cancelled
Some checks failed
Build and Push Docker Image / build (push) Has been cancelled
Replace single FILE_DOWNLOADED action with three granular actions: - FILE_VIEWED: inline preview loaded in the UI - FILE_OPENED: file opened in a new browser tab - FILE_DOWNLOADED: explicit download button clicked Add 'purpose' field to getDownloadUrl input (preview/open/download). All client callers updated to pass the appropriate purpose. Audit page updated with new filter options and color mappings. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -77,6 +77,8 @@ const ACTION_TYPES = [
|
|||||||
'ROUND_ARCHIVED',
|
'ROUND_ARCHIVED',
|
||||||
'UPLOAD_FILE',
|
'UPLOAD_FILE',
|
||||||
'DELETE_FILE',
|
'DELETE_FILE',
|
||||||
|
'FILE_VIEWED',
|
||||||
|
'FILE_OPENED',
|
||||||
'FILE_DOWNLOADED',
|
'FILE_DOWNLOADED',
|
||||||
'BULK_CREATE',
|
'BULK_CREATE',
|
||||||
'BULK_UPDATE_STATUS',
|
'BULK_UPDATE_STATUS',
|
||||||
@@ -171,6 +173,8 @@ const actionColors: Record<string, 'default' | 'destructive' | 'secondary' | 'ou
|
|||||||
ROUND_ACTIVATED: 'default',
|
ROUND_ACTIVATED: 'default',
|
||||||
ROUND_CLOSED: 'secondary',
|
ROUND_CLOSED: 'secondary',
|
||||||
ROUND_ARCHIVED: 'secondary',
|
ROUND_ARCHIVED: 'secondary',
|
||||||
|
FILE_VIEWED: 'outline',
|
||||||
|
FILE_OPENED: 'outline',
|
||||||
FILE_DOWNLOADED: 'outline',
|
FILE_DOWNLOADED: 'outline',
|
||||||
ROLE_CHANGED: 'secondary',
|
ROLE_CHANGED: 'secondary',
|
||||||
PASSWORD_SET: 'outline',
|
PASSWORD_SET: 'outline',
|
||||||
|
|||||||
@@ -146,7 +146,7 @@ export default function BulkUploadPage() {
|
|||||||
const handleViewFile = useCallback(
|
const handleViewFile = useCallback(
|
||||||
async (bucket: string, objectKey: string) => {
|
async (bucket: string, objectKey: string) => {
|
||||||
try {
|
try {
|
||||||
const { url } = await utils.file.getDownloadUrl.fetch({ bucket, objectKey })
|
const { url } = await utils.file.getDownloadUrl.fetch({ bucket, objectKey, purpose: 'open' as const })
|
||||||
window.open(url, '_blank')
|
window.open(url, '_blank')
|
||||||
} catch {
|
} catch {
|
||||||
toast.error('Failed to open file. It may have been deleted from storage.')
|
toast.error('Failed to open file. It may have been deleted from storage.')
|
||||||
|
|||||||
@@ -46,11 +46,11 @@ const fileTypeLabels: Record<string, string> = {
|
|||||||
|
|
||||||
function FileActionButtons({ bucket, objectKey, fileName }: { bucket: string; objectKey: string; fileName: string }) {
|
function FileActionButtons({ bucket, objectKey, fileName }: { bucket: string; objectKey: string; fileName: string }) {
|
||||||
const { data: viewData } = trpc.file.getDownloadUrl.useQuery(
|
const { data: viewData } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket, objectKey, forDownload: false },
|
{ bucket, objectKey, forDownload: false, purpose: 'open' as const },
|
||||||
{ staleTime: 10 * 60 * 1000 }
|
{ staleTime: 10 * 60 * 1000 }
|
||||||
)
|
)
|
||||||
const { data: dlData } = trpc.file.getDownloadUrl.useQuery(
|
const { data: dlData } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket, objectKey, forDownload: true, fileName },
|
{ bucket, objectKey, forDownload: true, fileName, purpose: 'download' as const },
|
||||||
{ staleTime: 10 * 60 * 1000 }
|
{ staleTime: 10 * 60 * 1000 }
|
||||||
)
|
)
|
||||||
const viewUrl = typeof viewData === 'string' ? viewData : viewData?.url
|
const viewUrl = typeof viewData === 'string' ? viewData : viewData?.url
|
||||||
|
|||||||
@@ -240,7 +240,7 @@ function FileItem({ file }: { file: ProjectFile }) {
|
|||||||
const Icon = getFileIcon(file.fileType, file.mimeType)
|
const Icon = getFileIcon(file.fileType, file.mimeType)
|
||||||
|
|
||||||
const { data: urlData, isLoading: isLoadingUrl } = trpc.file.getDownloadUrl.useQuery(
|
const { data: urlData, isLoading: isLoadingUrl } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket: file.bucket, objectKey: file.objectKey },
|
{ bucket: file.bucket, objectKey: file.objectKey, purpose: 'preview' as const },
|
||||||
{ enabled: showPreview }
|
{ enabled: showPreview }
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -440,7 +440,7 @@ function VersionDownloadButton({ bucket, objectKey }: { bucket: string; objectKe
|
|||||||
const [downloading, setDownloading] = useState(false)
|
const [downloading, setDownloading] = useState(false)
|
||||||
|
|
||||||
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket, objectKey },
|
{ bucket, objectKey, purpose: 'open' as const },
|
||||||
{ enabled: false }
|
{ enabled: false }
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -537,7 +537,7 @@ function FileOpenButton({ file, className, label }: { file: ProjectFile; classNa
|
|||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
|
|
||||||
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket: file.bucket, objectKey: file.objectKey },
|
{ bucket: file.bucket, objectKey: file.objectKey, purpose: 'open' as const },
|
||||||
{ enabled: false }
|
{ enabled: false }
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -590,7 +590,7 @@ function FileDownloadButton({ file, className, label }: { file: ProjectFile; cla
|
|||||||
const [downloading, setDownloading] = useState(false)
|
const [downloading, setDownloading] = useState(false)
|
||||||
|
|
||||||
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket: file.bucket, objectKey: file.objectKey, forDownload: true, fileName: file.fileName },
|
{ bucket: file.bucket, objectKey: file.objectKey, forDownload: true, fileName: file.fileName, purpose: 'download' as const },
|
||||||
{ enabled: false }
|
{ enabled: false }
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -746,7 +746,7 @@ function CompactFileItem({ file }: { file: ProjectFile }) {
|
|||||||
const Icon = getFileIcon(file.fileType, file.mimeType)
|
const Icon = getFileIcon(file.fileType, file.mimeType)
|
||||||
|
|
||||||
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
const { refetch } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket: file.bucket, objectKey: file.objectKey },
|
{ bucket: file.bucket, objectKey: file.objectKey, purpose: 'open' as const },
|
||||||
{ enabled: false }
|
{ enabled: false }
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -72,7 +72,7 @@ interface RequirementUploadSlotProps {
|
|||||||
|
|
||||||
function ViewFileButton({ bucket, objectKey }: { bucket: string; objectKey: string }) {
|
function ViewFileButton({ bucket, objectKey }: { bucket: string; objectKey: string }) {
|
||||||
const { data } = trpc.file.getDownloadUrl.useQuery(
|
const { data } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket, objectKey, forDownload: false },
|
{ bucket, objectKey, forDownload: false, purpose: 'open' as const },
|
||||||
{ staleTime: 10 * 60 * 1000 }
|
{ staleTime: 10 * 60 * 1000 }
|
||||||
)
|
)
|
||||||
const href = typeof data === 'string' ? data : data?.url
|
const href = typeof data === 'string' ? data : data?.url
|
||||||
@@ -87,7 +87,7 @@ function ViewFileButton({ bucket, objectKey }: { bucket: string; objectKey: stri
|
|||||||
|
|
||||||
function DownloadFileButton({ bucket, objectKey, fileName }: { bucket: string; objectKey: string; fileName: string }) {
|
function DownloadFileButton({ bucket, objectKey, fileName }: { bucket: string; objectKey: string; fileName: string }) {
|
||||||
const { data } = trpc.file.getDownloadUrl.useQuery(
|
const { data } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket, objectKey, forDownload: true, fileName },
|
{ bucket, objectKey, forDownload: true, fileName, purpose: 'download' as const },
|
||||||
{ staleTime: 10 * 60 * 1000 }
|
{ staleTime: 10 * 60 * 1000 }
|
||||||
)
|
)
|
||||||
const href = typeof data === 'string' ? data : data?.url
|
const href = typeof data === 'string' ? data : data?.url
|
||||||
@@ -229,7 +229,7 @@ export function RequirementUploadSlot({
|
|||||||
|
|
||||||
// Fetch preview URL only when preview is toggled on
|
// Fetch preview URL only when preview is toggled on
|
||||||
const { data: previewUrlData, isLoading: isLoadingPreview } = trpc.file.getDownloadUrl.useQuery(
|
const { data: previewUrlData, isLoading: isLoadingPreview } = trpc.file.getDownloadUrl.useQuery(
|
||||||
{ bucket: existingFile?.bucket || '', objectKey: existingFile?.objectKey || '', forDownload: false },
|
{ bucket: existingFile?.bucket || '', objectKey: existingFile?.objectKey || '', forDownload: false, purpose: 'preview' as const },
|
||||||
{ enabled: showPreview && !!existingFile?.bucket && !!existingFile?.objectKey, staleTime: 10 * 60 * 1000 }
|
{ enabled: showPreview && !!existingFile?.bucket && !!existingFile?.objectKey, staleTime: 10 * 60 * 1000 }
|
||||||
)
|
)
|
||||||
const previewUrl = typeof previewUrlData === 'string' ? previewUrlData : previewUrlData?.url
|
const previewUrl = typeof previewUrlData === 'string' ? previewUrlData : previewUrlData?.url
|
||||||
|
|||||||
@@ -17,6 +17,8 @@ export const fileRouter = router({
|
|||||||
objectKey: z.string(),
|
objectKey: z.string(),
|
||||||
forDownload: z.boolean().optional(),
|
forDownload: z.boolean().optional(),
|
||||||
fileName: z.string().optional(),
|
fileName: z.string().optional(),
|
||||||
|
/** Why the URL is being requested — drives audit log granularity. */
|
||||||
|
purpose: z.enum(['preview', 'open', 'download']).optional(),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
.query(async ({ ctx, input }) => {
|
.query(async ({ ctx, input }) => {
|
||||||
@@ -124,14 +126,25 @@ export const fileRouter = router({
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only log actual downloads, not preview/view URL requests
|
// Log file access with granular action based on purpose
|
||||||
if (input.forDownload) {
|
const purpose = input.purpose ?? (input.forDownload ? 'download' : undefined)
|
||||||
|
if (purpose) {
|
||||||
|
const actionMap = {
|
||||||
|
preview: 'FILE_VIEWED',
|
||||||
|
open: 'FILE_OPENED',
|
||||||
|
download: 'FILE_DOWNLOADED',
|
||||||
|
} as const
|
||||||
await logAudit({
|
await logAudit({
|
||||||
prisma: ctx.prisma,
|
prisma: ctx.prisma,
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
action: 'FILE_DOWNLOADED',
|
action: actionMap[purpose],
|
||||||
entityType: 'ProjectFile',
|
entityType: 'ProjectFile',
|
||||||
detailsJson: { bucket: input.bucket, objectKey: input.objectKey, fileName: input.fileName },
|
detailsJson: {
|
||||||
|
bucket: input.bucket,
|
||||||
|
objectKey: input.objectKey,
|
||||||
|
fileName: input.fileName,
|
||||||
|
purpose,
|
||||||
|
},
|
||||||
ipAddress: ctx.ip,
|
ipAddress: ctx.ip,
|
||||||
userAgent: ctx.userAgent,
|
userAgent: ctx.userAgent,
|
||||||
})
|
})
|
||||||
|
|||||||
Reference in New Issue
Block a user