feat: add per-patient Local folder in Documents page backed by Cloud Storage

- Documents page shows a "Local Folder" card for each selected patient
  with an "Open in Cloud Storage" button that deep-links to their folder
- Cloud Storage page reads ?folderId URL param on mount and auto-opens
  the folder panel for seamless navigation from Documents
- Backend: GET /api/cloud-storage/patient-folder/:patientId endpoint
  that idempotently gets or creates a top-level CloudFolder per patient
- CloudFolder schema gains optional patientId field linked to Patient
- Disk directories for cloud storage folders now use the folder's name
  (e.g. "Xiaohui Wang/") instead of the opaque "folder-{id}/" path

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
ff
2026-06-05 23:01:56 -04:00
parent 2457e12b5c
commit d5bc96ff39
158 changed files with 1539 additions and 130 deletions

View File

@@ -4,9 +4,11 @@ export const CloudFolderAggregateResultSchema = z.object({ _count: z.object({
userId: z.number(),
name: z.number(),
parentId: z.number(),
patientId: z.number(),
parent: z.number(),
children: z.number(),
user: z.number(),
patient: z.number(),
files: z.number(),
createdAt: z.number(),
updatedAt: z.number()
@@ -14,18 +16,21 @@ export const CloudFolderAggregateResultSchema = z.object({ _count: z.object({
_sum: z.object({
id: z.number().nullable(),
userId: z.number().nullable(),
parentId: z.number().nullable()
parentId: z.number().nullable(),
patientId: z.number().nullable()
}).nullable().optional(),
_avg: z.object({
id: z.number().nullable(),
userId: z.number().nullable(),
parentId: z.number().nullable()
parentId: z.number().nullable(),
patientId: z.number().nullable()
}).nullable().optional(),
_min: z.object({
id: z.number().int().nullable(),
userId: z.number().int().nullable(),
name: z.string().nullable(),
parentId: z.number().int().nullable(),
patientId: z.number().int().nullable(),
createdAt: z.date().nullable(),
updatedAt: z.date().nullable()
}).nullable().optional(),
@@ -34,6 +39,7 @@ export const CloudFolderAggregateResultSchema = z.object({ _count: z.object({
userId: z.number().int().nullable(),
name: z.string().nullable(),
parentId: z.number().int().nullable(),
patientId: z.number().int().nullable(),
createdAt: z.date().nullable(),
updatedAt: z.date().nullable()
}).nullable().optional()});

View File

@@ -4,9 +4,11 @@ export const CloudFolderCreateResultSchema = z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -4,9 +4,11 @@ export const CloudFolderDeleteResultSchema = z.nullable(z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -4,9 +4,11 @@ export const CloudFolderFindFirstResultSchema = z.nullable(z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -5,9 +5,11 @@ export const CloudFolderFindManyResultSchema = z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -4,9 +4,11 @@ export const CloudFolderFindUniqueResultSchema = z.nullable(z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -4,6 +4,7 @@ export const CloudFolderGroupByResultSchema = z.array(z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int(),
patientId: z.number().int(),
createdAt: z.date(),
updatedAt: z.date(),
_count: z.object({
@@ -11,9 +12,11 @@ export const CloudFolderGroupByResultSchema = z.array(z.object({
userId: z.number(),
name: z.number(),
parentId: z.number(),
patientId: z.number(),
parent: z.number(),
children: z.number(),
user: z.number(),
patient: z.number(),
files: z.number(),
createdAt: z.number(),
updatedAt: z.number()
@@ -21,18 +24,21 @@ export const CloudFolderGroupByResultSchema = z.array(z.object({
_sum: z.object({
id: z.number().nullable(),
userId: z.number().nullable(),
parentId: z.number().nullable()
parentId: z.number().nullable(),
patientId: z.number().nullable()
}).nullable().optional(),
_avg: z.object({
id: z.number().nullable(),
userId: z.number().nullable(),
parentId: z.number().nullable()
parentId: z.number().nullable(),
patientId: z.number().nullable()
}).nullable().optional(),
_min: z.object({
id: z.number().int().nullable(),
userId: z.number().int().nullable(),
name: z.string().nullable(),
parentId: z.number().int().nullable(),
patientId: z.number().int().nullable(),
createdAt: z.date().nullable(),
updatedAt: z.date().nullable()
}).nullable().optional(),
@@ -41,6 +47,7 @@ export const CloudFolderGroupByResultSchema = z.array(z.object({
userId: z.number().int().nullable(),
name: z.string().nullable(),
parentId: z.number().int().nullable(),
patientId: z.number().int().nullable(),
createdAt: z.date().nullable(),
updatedAt: z.date().nullable()
}).nullable().optional()

View File

@@ -4,9 +4,11 @@ export const CloudFolderUpdateResultSchema = z.nullable(z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -4,9 +4,11 @@ export const CloudFolderUpsertResultSchema = z.object({
userId: z.number().int(),
name: z.string(),
parentId: z.number().int().optional(),
patientId: z.number().int().optional(),
parent: z.unknown().optional(),
children: z.array(z.unknown()),
user: z.unknown(),
patient: z.unknown().optional(),
files: z.array(z.unknown()),
createdAt: z.date(),
updatedAt: z.date()

View File

@@ -29,7 +29,8 @@ export const PatientAggregateResultSchema = z.object({ _count: z.object({
payment: z.number(),
communications: z.number(),
documents: z.number(),
conversation: z.number()
conversation: z.number(),
cloudFolders: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable(),

View File

@@ -29,5 +29,6 @@ export const PatientCreateResultSchema = z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
});

View File

@@ -29,5 +29,6 @@ export const PatientDeleteResultSchema = z.nullable(z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
}));

View File

@@ -29,5 +29,6 @@ export const PatientFindFirstResultSchema = z.nullable(z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
}));

View File

@@ -30,7 +30,8 @@ export const PatientFindManyResultSchema = z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
})),
pagination: z.object({
page: z.number().int().min(1),

View File

@@ -29,5 +29,6 @@ export const PatientFindUniqueResultSchema = z.nullable(z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
}));

View File

@@ -50,7 +50,8 @@ export const PatientGroupByResultSchema = z.array(z.object({
payment: z.number(),
communications: z.number(),
documents: z.number(),
conversation: z.number()
conversation: z.number(),
cloudFolders: z.number()
}).optional(),
_sum: z.object({
id: z.number().nullable(),

View File

@@ -29,5 +29,6 @@ export const PatientUpdateResultSchema = z.nullable(z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
}));

View File

@@ -29,5 +29,6 @@ export const PatientUpsertResultSchema = z.object({
payment: z.array(z.unknown()),
communications: z.array(z.unknown()),
documents: z.array(z.unknown()),
conversation: z.unknown().optional()
conversation: z.unknown().optional(),
cloudFolders: z.array(z.unknown())
});