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:
@@ -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()});
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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()
|
||||
|
||||
@@ -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(),
|
||||
|
||||
@@ -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())
|
||||
});
|
||||
@@ -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())
|
||||
}));
|
||||
@@ -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())
|
||||
}));
|
||||
@@ -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),
|
||||
|
||||
@@ -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())
|
||||
}));
|
||||
@@ -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(),
|
||||
|
||||
@@ -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())
|
||||
}));
|
||||
@@ -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())
|
||||
});
|
||||
Reference in New Issue
Block a user