@rpcbase/server 0.548.0 → 0.549.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{handler-V5AVyt5y.js → handler-Cn5I8j8k.js} +90 -18
- package/dist/handler-Cn5I8j8k.js.map +1 -0
- package/dist/{handler-BPtmV7Gp.js → handler-Da2KGCRq.js} +68 -61
- package/dist/handler-Da2KGCRq.js.map +1 -0
- package/dist/uploads/api/file-uploads/handlers/completeUpload.d.ts.map +1 -1
- package/dist/uploads/api/file-uploads/imageVariantsPostProcessor.d.ts +2 -0
- package/dist/uploads/api/file-uploads/imageVariantsPostProcessor.d.ts.map +1 -0
- package/dist/uploads/api/file-uploads/processors/convertHeifToWebp.d.ts +31 -0
- package/dist/uploads/api/file-uploads/processors/convertHeifToWebp.d.ts.map +1 -0
- package/dist/uploads/api/file-uploads/processors/index.d.ts +4 -0
- package/dist/uploads/api/file-uploads/processors/index.d.ts.map +1 -1
- package/dist/uploads/api/files/handlers/deleteFile.d.ts.map +1 -1
- package/dist/uploads/api/files/handlers/getFile.d.ts.map +1 -1
- package/dist/uploads-BAxHzidK.js +573 -0
- package/dist/uploads-BAxHzidK.js.map +1 -0
- package/dist/uploads.d.ts.map +1 -1
- package/dist/uploads.js +13 -106
- package/dist/uploads.js.map +1 -1
- package/package.json +9 -2
- package/dist/handler-BPtmV7Gp.js.map +0 -1
- package/dist/handler-V5AVyt5y.js.map +0 -1
- package/dist/shared-xNnTJqaH.js +0 -104
- package/dist/shared-xNnTJqaH.js.map +0 -1
|
@@ -1,11 +1,22 @@
|
|
|
1
1
|
import { getTenantFilesystemDb } from "@rpcbase/db";
|
|
2
2
|
import { ObjectId, GridFSBucket } from "mongodb";
|
|
3
|
-
import { g as getTenantId,
|
|
3
|
+
import { g as getTenantId, d as getBucketName, i as getUserId, p as getUploadKeyHash } from "./uploads-BAxHzidK.js";
|
|
4
4
|
const resolveHeaderString$1 = (value) => {
|
|
5
5
|
if (typeof value !== "string") return null;
|
|
6
6
|
const normalized = value.trim();
|
|
7
7
|
return normalized ? normalized : null;
|
|
8
8
|
};
|
|
9
|
+
const isFileNotFoundError = (error) => {
|
|
10
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
11
|
+
return message.includes("FileNotFound");
|
|
12
|
+
};
|
|
13
|
+
const deleteGridFsFile = async (bucket, fileId) => {
|
|
14
|
+
try {
|
|
15
|
+
await bucket.delete(fileId);
|
|
16
|
+
} catch (error) {
|
|
17
|
+
if (!isFileNotFoundError(error)) throw error;
|
|
18
|
+
}
|
|
19
|
+
};
|
|
9
20
|
const deleteFile = async (_payload, ctx) => {
|
|
10
21
|
const tenantId = getTenantId(ctx);
|
|
11
22
|
if (!tenantId) {
|
|
@@ -69,16 +80,20 @@ const deleteFile = async (_payload, ctx) => {
|
|
|
69
80
|
};
|
|
70
81
|
}
|
|
71
82
|
try {
|
|
72
|
-
await bucket.
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
error: "delete_failed"
|
|
80
|
-
};
|
|
83
|
+
const variants = await bucket.find({
|
|
84
|
+
"metadata.variantOf": fileObjectId.toHexString()
|
|
85
|
+
}).toArray();
|
|
86
|
+
const variantIds = variants.map((variant) => variant?._id).filter((id) => id instanceof ObjectId);
|
|
87
|
+
await deleteGridFsFile(bucket, fileObjectId);
|
|
88
|
+
for (const variantId of variantIds) {
|
|
89
|
+
await deleteGridFsFile(bucket, variantId);
|
|
81
90
|
}
|
|
91
|
+
} catch {
|
|
92
|
+
ctx.res.status(500);
|
|
93
|
+
return {
|
|
94
|
+
ok: false,
|
|
95
|
+
error: "delete_failed"
|
|
96
|
+
};
|
|
82
97
|
}
|
|
83
98
|
ctx.res.status(204);
|
|
84
99
|
return {
|
|
@@ -100,6 +115,56 @@ const resolveHeaderBoolean = (value) => {
|
|
|
100
115
|
return null;
|
|
101
116
|
};
|
|
102
117
|
const escapeHeaderFilename = (filename) => filename.replace(/[\\"]/g, "_");
|
|
118
|
+
const parseRequestedWidth = (value) => {
|
|
119
|
+
if (typeof value !== "string") return null;
|
|
120
|
+
const parsed = Number(value.trim());
|
|
121
|
+
if (!Number.isFinite(parsed) || parsed <= 0) return null;
|
|
122
|
+
return Math.floor(parsed);
|
|
123
|
+
};
|
|
124
|
+
const resolveImageVariants = (file) => {
|
|
125
|
+
const metadata = file?.metadata;
|
|
126
|
+
if (!metadata || typeof metadata !== "object") return null;
|
|
127
|
+
const image = metadata.image;
|
|
128
|
+
if (!image || typeof image !== "object") return null;
|
|
129
|
+
const variants = image.variants;
|
|
130
|
+
if (!variants || typeof variants !== "object") return null;
|
|
131
|
+
return variants;
|
|
132
|
+
};
|
|
133
|
+
const resolveVariantFileId = (variant) => {
|
|
134
|
+
if (!variant || typeof variant !== "object") return null;
|
|
135
|
+
const fileId = variant.fileId;
|
|
136
|
+
if (typeof fileId !== "string") return null;
|
|
137
|
+
try {
|
|
138
|
+
return new ObjectId(fileId);
|
|
139
|
+
} catch {
|
|
140
|
+
return null;
|
|
141
|
+
}
|
|
142
|
+
};
|
|
143
|
+
const selectVariantFileId = (file, requestedWidth, requestedVariant) => {
|
|
144
|
+
const variants = resolveImageVariants(file);
|
|
145
|
+
if (!variants) return null;
|
|
146
|
+
let aliasWidth = null;
|
|
147
|
+
if (typeof requestedVariant === "string") {
|
|
148
|
+
const normalized = requestedVariant.trim().toLowerCase();
|
|
149
|
+
aliasWidth = normalized === "thumb" ? 320 : normalized === "small" ? 640 : normalized === "medium" ? 960 : normalized === "large" ? 1600 : null;
|
|
150
|
+
if (aliasWidth) {
|
|
151
|
+
const variantFileId = resolveVariantFileId(variants[String(aliasWidth)]);
|
|
152
|
+
if (variantFileId) return variantFileId;
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
const targetWidth = aliasWidth ?? requestedWidth;
|
|
156
|
+
if (!targetWidth) return null;
|
|
157
|
+
const widths = Object.keys(variants).map((width) => Number(width)).filter((width) => Number.isInteger(width) && width > 0).sort((a, b) => a - b);
|
|
158
|
+
const selectedWidth = widths.reduce((selected, width) => {
|
|
159
|
+
if (selected === null) return width;
|
|
160
|
+
const selectedDistance = Math.abs(selected - targetWidth);
|
|
161
|
+
const distance = Math.abs(width - targetWidth);
|
|
162
|
+
if (distance < selectedDistance) return width;
|
|
163
|
+
if (distance === selectedDistance && width > selected) return width;
|
|
164
|
+
return selected;
|
|
165
|
+
}, null);
|
|
166
|
+
return selectedWidth ? resolveVariantFileId(variants[String(selectedWidth)]) : null;
|
|
167
|
+
};
|
|
103
168
|
const getFile = async (_payload, ctx) => {
|
|
104
169
|
const tenantId = getTenantId(ctx);
|
|
105
170
|
if (!tenantId) {
|
|
@@ -144,15 +209,22 @@ const getFile = async (_payload, ctx) => {
|
|
|
144
209
|
return {};
|
|
145
210
|
}
|
|
146
211
|
}
|
|
147
|
-
const
|
|
212
|
+
const requestedWidth = parseRequestedWidth(ctx.req.query?.w);
|
|
213
|
+
const variantFileObjectId = selectVariantFileId(file, requestedWidth, ctx.req.query?.variant);
|
|
214
|
+
const [variantFile] = variantFileObjectId ? await bucket.find({
|
|
215
|
+
_id: variantFileObjectId
|
|
216
|
+
}).limit(1).toArray() : [];
|
|
217
|
+
const selectedFileObjectId = variantFile ? variantFileObjectId : fileObjectId;
|
|
218
|
+
const selectedFile = variantFile ?? file;
|
|
219
|
+
const mimeTypeFromMetadata = resolveHeaderString(selectedFile?.metadata?.mimeType);
|
|
148
220
|
const mimeType = mimeTypeFromMetadata ?? "application/octet-stream";
|
|
149
|
-
const filenameFromDb = resolveHeaderString(
|
|
221
|
+
const filenameFromDb = resolveHeaderString(selectedFile?.filename);
|
|
150
222
|
const filename = filenameFromDb ?? fileIdRaw;
|
|
151
223
|
const filenameSafe = escapeHeaderFilename(filename);
|
|
152
224
|
const cacheControl = "private, max-age=0, must-revalidate";
|
|
153
|
-
const md5 = resolveHeaderString(
|
|
154
|
-
const uploadDate =
|
|
155
|
-
const etagValue = md5 ?? `${
|
|
225
|
+
const md5 = resolveHeaderString(selectedFile?.md5);
|
|
226
|
+
const uploadDate = selectedFile?.uploadDate instanceof Date ? selectedFile.uploadDate : null;
|
|
227
|
+
const etagValue = md5 ?? `${selectedFileObjectId.toHexString()}-${String(selectedFile?.length ?? 0)}-${String(uploadDate?.getTime() ?? 0)}`;
|
|
156
228
|
const etag = md5 ? `"${etagValue}"` : `W/"${etagValue}"`;
|
|
157
229
|
const ifNoneMatch = resolveHeaderString(ctx.req.headers?.["if-none-match"]);
|
|
158
230
|
if (ifNoneMatch) {
|
|
@@ -167,7 +239,7 @@ const getFile = async (_payload, ctx) => {
|
|
|
167
239
|
}
|
|
168
240
|
ctx.res.status(200);
|
|
169
241
|
ctx.res.setHeader("Content-Type", mimeType);
|
|
170
|
-
ctx.res.setHeader("Content-Length", String(
|
|
242
|
+
ctx.res.setHeader("Content-Length", String(selectedFile?.length ?? 0));
|
|
171
243
|
ctx.res.setHeader("Content-Disposition", `inline; filename="${filenameSafe}"`);
|
|
172
244
|
ctx.res.setHeader("Cache-Control", cacheControl);
|
|
173
245
|
ctx.res.setHeader("ETag", etag);
|
|
@@ -180,7 +252,7 @@ const getFile = async (_payload, ctx) => {
|
|
|
180
252
|
ctx.res.end();
|
|
181
253
|
return {};
|
|
182
254
|
}
|
|
183
|
-
const stream = bucket.openDownloadStream(
|
|
255
|
+
const stream = bucket.openDownloadStream(selectedFileObjectId);
|
|
184
256
|
stream.on("error", () => {
|
|
185
257
|
try {
|
|
186
258
|
ctx.res.destroy();
|
|
@@ -200,4 +272,4 @@ const handler = (api) => {
|
|
|
200
272
|
export {
|
|
201
273
|
handler as default
|
|
202
274
|
};
|
|
203
|
-
//# sourceMappingURL=handler-
|
|
275
|
+
//# sourceMappingURL=handler-Cn5I8j8k.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"handler-Cn5I8j8k.js","sources":["../src/uploads/api/files/handlers/deleteFile.ts","../src/uploads/api/files/handlers/getFile.ts","../src/uploads/api/files/index.ts","../src/uploads/api/files/handler.ts"],"sourcesContent":["import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb } from \"@rpcbase/db\"\nimport { GridFSBucket, ObjectId } from \"mongodb\"\n\nimport { getBucketName, getTenantId, getUploadKeyHash, getUserId, type SessionUser } from \"../../file-uploads/shared\"\n\n\ntype DeleteResponsePayload = {\n ok: boolean\n error?: string\n}\n\nconst resolveHeaderString = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nconst isFileNotFoundError = (error: unknown): boolean => {\n const message = error instanceof Error ? error.message : String(error)\n return message.includes(\"FileNotFound\")\n}\n\nconst deleteGridFsFile = async (bucket: GridFSBucket, fileId: ObjectId): Promise<void> => {\n try {\n await bucket.delete(fileId)\n } catch (error) {\n if (!isFileNotFoundError(error)) throw error\n }\n}\n\nexport const deleteFile: ApiHandler<Record<string, never>, DeleteResponsePayload, SessionUser> = async (\n _payload,\n ctx,\n): Promise<DeleteResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const fileIdRaw = String(ctx.req.params?.fileId ?? \"\").trim()\n let fileObjectId: ObjectId\n try {\n fileObjectId = new ObjectId(fileIdRaw)\n } catch {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_file_id\" }\n }\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n ctx.res.status(500)\n return { ok: false, error: \"filesystem_db_unavailable\" }\n }\n\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const userId = getUserId(ctx)\n const uploadKeyHash = getUploadKeyHash(ctx)\n if (!userId && !uploadKeyHash) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n const [file] = await bucket.find({ _id: fileObjectId }).limit(1).toArray()\n if (!file) {\n ctx.res.status(204)\n return { ok: true }\n }\n\n const metadataUserId = resolveHeaderString((file as any)?.metadata?.userId)\n const ownerKeyHash = resolveHeaderString((file as any)?.metadata?.ownerKeyHash)\n\n const authorizedByUser = Boolean(userId && metadataUserId && userId === metadataUserId)\n const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash)\n\n if (!authorizedByUser && !authorizedByKey) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n try {\n const variants = await bucket.find({ \"metadata.variantOf\": fileObjectId.toHexString() }).toArray()\n const variantIds = variants\n .map((variant) => (variant as any)?._id)\n .filter((id): id is ObjectId => id instanceof ObjectId)\n\n await deleteGridFsFile(bucket, fileObjectId)\n for (const variantId of variantIds) {\n await deleteGridFsFile(bucket, variantId)\n }\n } catch {\n ctx.res.status(500)\n return { ok: false, error: \"delete_failed\" }\n }\n\n ctx.res.status(204)\n return { ok: true }\n}\n","import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb } from \"@rpcbase/db\"\nimport { GridFSBucket, ObjectId } from \"mongodb\"\n\nimport { getBucketName, getTenantId, getUploadKeyHash, getUserId, type SessionUser } from \"../../file-uploads/shared\"\n\n\nconst resolveHeaderString = (value: unknown): string | null => {\n if (typeof value !== \"string\") return null\n const normalized = value.trim()\n return normalized ? normalized : null\n}\n\nconst resolveHeaderBoolean = (value: unknown): boolean | null => {\n if (typeof value === \"boolean\") return value\n if (typeof value !== \"string\") return null\n const normalized = value.trim().toLowerCase()\n if (!normalized) return null\n if (normalized === \"true\") return true\n if (normalized === \"false\") return false\n return null\n}\n\nconst escapeHeaderFilename = (filename: string): string => filename.replace(/[\\\\\"]/g, \"_\")\n\nconst parseRequestedWidth = (value: unknown): number | null => {\n if (typeof value !== \"string\") return null\n const parsed = Number(value.trim())\n if (!Number.isFinite(parsed) || parsed <= 0) return null\n return Math.floor(parsed)\n}\n\nconst resolveImageVariants = (file: unknown): Record<string, unknown> | null => {\n const metadata = (file as any)?.metadata\n if (!metadata || typeof metadata !== \"object\") return null\n const image = (metadata as Record<string, unknown>).image\n if (!image || typeof image !== \"object\") return null\n const variants = (image as Record<string, unknown>).variants\n if (!variants || typeof variants !== \"object\") return null\n return variants as Record<string, unknown>\n}\n\nconst resolveVariantFileId = (variant: unknown): ObjectId | null => {\n if (!variant || typeof variant !== \"object\") return null\n const fileId = (variant as Record<string, unknown>).fileId\n if (typeof fileId !== \"string\") return null\n try {\n return new ObjectId(fileId)\n } catch {\n return null\n }\n}\n\nconst selectVariantFileId = (file: unknown, requestedWidth: number | null, requestedVariant: unknown): ObjectId | null => {\n const variants = resolveImageVariants(file)\n if (!variants) return null\n\n let aliasWidth: number | null = null\n if (typeof requestedVariant === \"string\") {\n const normalized = requestedVariant.trim().toLowerCase()\n aliasWidth = normalized === \"thumb\" ? 320 : normalized === \"small\" ? 640 : normalized === \"medium\" ? 960 : normalized === \"large\" ? 1600 : null\n if (aliasWidth) {\n const variantFileId = resolveVariantFileId(variants[String(aliasWidth)])\n if (variantFileId) return variantFileId\n }\n }\n\n const targetWidth = aliasWidth ?? requestedWidth\n if (!targetWidth) return null\n\n const widths = Object.keys(variants)\n .map((width) => Number(width))\n .filter((width) => Number.isInteger(width) && width > 0)\n .sort((a, b) => a - b)\n\n const selectedWidth = widths.reduce<number | null>((selected, width) => {\n if (selected === null) return width\n\n const selectedDistance = Math.abs(selected - targetWidth)\n const distance = Math.abs(width - targetWidth)\n if (distance < selectedDistance) return width\n if (distance === selectedDistance && width > selected) return width\n return selected\n }, null)\n return selectedWidth ? resolveVariantFileId(variants[String(selectedWidth)]) : null\n}\n\nexport const getFile: ApiHandler<Record<string, never>, Record<string, never>, SessionUser> = async (\n _payload,\n ctx,\n): Promise<Record<string, never>> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400).end()\n return {}\n }\n\n const fileIdRaw = String(ctx.req.params?.fileId ?? \"\").trim()\n let fileObjectId: ObjectId\n try {\n fileObjectId = new ObjectId(fileIdRaw)\n } catch {\n ctx.res.status(400).end()\n return {}\n }\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n ctx.res.status(500).end()\n return {}\n }\n\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const [file] = await bucket.find({ _id: fileObjectId }).limit(1).toArray()\n if (!file) {\n ctx.res.status(404).end()\n return {}\n }\n\n const isPublic = resolveHeaderBoolean((file as any)?.metadata?.isPublic) ?? false\n if (!isPublic) {\n const userId = getUserId(ctx)\n const uploadKeyHash = getUploadKeyHash(ctx)\n const metadataUserId = resolveHeaderString((file as any)?.metadata?.userId)\n const ownerKeyHash = resolveHeaderString((file as any)?.metadata?.ownerKeyHash)\n\n const authorizedByUser = Boolean(userId && metadataUserId && userId === metadataUserId)\n const authorizedByKey = Boolean(uploadKeyHash && ownerKeyHash && uploadKeyHash === ownerKeyHash)\n\n if (!authorizedByUser && !authorizedByKey) {\n ctx.res.status(401).end()\n return {}\n }\n }\n\n const requestedWidth = parseRequestedWidth((ctx.req.query as any)?.w)\n const variantFileObjectId = selectVariantFileId(file, requestedWidth, (ctx.req.query as any)?.variant)\n const [variantFile] = variantFileObjectId\n ? await bucket.find({ _id: variantFileObjectId }).limit(1).toArray()\n : []\n const selectedFileObjectId = variantFile ? variantFileObjectId as ObjectId : fileObjectId\n const selectedFile = variantFile ?? file\n\n const mimeTypeFromMetadata = resolveHeaderString((selectedFile as any)?.metadata?.mimeType)\n const mimeType = mimeTypeFromMetadata ?? \"application/octet-stream\"\n const filenameFromDb = resolveHeaderString((selectedFile as any)?.filename)\n const filename = filenameFromDb ?? fileIdRaw\n const filenameSafe = escapeHeaderFilename(filename)\n\n const cacheControl = \"private, max-age=0, must-revalidate\"\n const md5 = resolveHeaderString((selectedFile as any)?.md5)\n const uploadDate = (selectedFile as any)?.uploadDate instanceof Date ? (selectedFile as any).uploadDate : null\n const etagValue = md5 ?? `${selectedFileObjectId.toHexString()}-${String((selectedFile as any)?.length ?? 0)}-${String(uploadDate?.getTime() ?? 0)}`\n const etag = md5 ? `\"${etagValue}\"` : `W/\"${etagValue}\"`\n const ifNoneMatch = resolveHeaderString((ctx.req.headers as any)?.[\"if-none-match\"])\n if (ifNoneMatch) {\n const candidates = ifNoneMatch.split(\",\").map((value) => value.trim()).filter(Boolean)\n if (candidates.includes(\"*\") || candidates.includes(etag)) {\n ctx.res.status(304)\n ctx.res.setHeader(\"Cache-Control\", cacheControl)\n ctx.res.setHeader(\"ETag\", etag)\n ctx.res.end()\n return {}\n }\n }\n\n ctx.res.status(200)\n ctx.res.setHeader(\"Content-Type\", mimeType)\n ctx.res.setHeader(\"Content-Length\", String((selectedFile as any)?.length ?? 0))\n ctx.res.setHeader(\"Content-Disposition\", `inline; filename=\"${filenameSafe}\"`)\n ctx.res.setHeader(\"Cache-Control\", cacheControl)\n ctx.res.setHeader(\"ETag\", etag)\n ctx.res.setHeader(\"X-Content-Type-Options\", \"nosniff\")\n if (mimeType === \"image/svg+xml\") {\n ctx.res.setHeader(\n \"Content-Security-Policy\",\n \"default-src 'none'; style-src 'unsafe-inline'; sandbox; base-uri 'none'; form-action 'none'\",\n )\n }\n ctx.res.flushHeaders()\n\n if (ctx.req.method === \"HEAD\") {\n ctx.res.end()\n return {}\n }\n\n const stream = bucket.openDownloadStream(selectedFileObjectId)\n\n stream.on(\"error\", () => {\n try {\n ctx.res.destroy()\n } catch {\n //\n }\n })\n\n stream.pipe(ctx.res)\n return {}\n}\n","export const Route = \"/api/rb/files/:fileId\"\n\nexport const GetRoute = Route\nexport const DeleteRoute = Route\n","import { Api } from \"@rpcbase/api\"\n\nimport { deleteFile } from \"./handlers/deleteFile\"\nimport { getFile } from \"./handlers/getFile\"\n\nimport * as Files from \"./index\"\n\n\nexport default (api: Api) => {\n api.get(Files.GetRoute, getFile)\n api.delete(Files.DeleteRoute, deleteFile)\n}\n"],"names":["resolveHeaderString","value","normalized","trim","isFileNotFoundError","error","message","Error","String","includes","deleteGridFsFile","bucket","fileId","delete","deleteFile","_payload","ctx","tenantId","getTenantId","res","status","ok","fileIdRaw","req","params","fileObjectId","ObjectId","fsDb","getTenantFilesystemDb","nativeDb","db","bucketName","getBucketName","GridFSBucket","userId","getUserId","uploadKeyHash","getUploadKeyHash","file","find","_id","limit","toArray","metadataUserId","metadata","ownerKeyHash","authorizedByUser","Boolean","authorizedByKey","variants","toHexString","variantIds","map","variant","filter","id","variantId","resolveHeaderBoolean","toLowerCase","escapeHeaderFilename","filename","replace","parseRequestedWidth","parsed","Number","isFinite","Math","floor","resolveImageVariants","image","resolveVariantFileId","selectVariantFileId","requestedWidth","requestedVariant","aliasWidth","variantFileId","targetWidth","widths","Object","keys","width","isInteger","sort","a","b","selectedWidth","reduce","selected","selectedDistance","abs","distance","getFile","end","isPublic","query","w","variantFileObjectId","variantFile","selectedFileObjectId","selectedFile","mimeTypeFromMetadata","mimeType","filenameFromDb","filenameSafe","cacheControl","md5","uploadDate","Date","etagValue","length","getTime","etag","ifNoneMatch","headers","candidates","split","setHeader","flushHeaders","method","stream","openDownloadStream","on","destroy","pipe","Route","GetRoute","DeleteRoute","api","get","Files"],"mappings":"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"}
|
|
@@ -1,65 +1,20 @@
|
|
|
1
1
|
import { models, getTenantFilesystemDb } from "@rpcbase/db";
|
|
2
2
|
import { GridFSBucket, ObjectId } from "mongodb";
|
|
3
|
-
import { enqueueUploadPostProcessors } from "./uploads.js";
|
|
4
|
-
import { JSDOM } from "jsdom";
|
|
5
|
-
import createDOMPurify from "dompurify";
|
|
6
|
-
import { g as getTenantId, b as buildUploadsAbility, a as getUploadSessionAccessQuery, e as ensureUploadIndexes, c as getBucketName, d as getModelCtx, f as getUserId, h as getChunkSizeBytes, i as getSessionTtlMs, j as computeSha256Hex, t as toBufferPayload, n as normalizeSha256Hex, k as getMaxClientUploadBytesPerSecond, l as getRawBodyLimitBytes } from "./shared-xNnTJqaH.js";
|
|
3
|
+
import { s as sanitizeSvgProcessor, c as convertHeifToWebpProcessor, g as getTenantId, b as buildUploadsAbility, a as getUploadSessionAccessQuery, e as ensureUploadIndexes, d as getBucketName, f as enqueueUploadPostProcessors, h as getModelCtx, i as getUserId, j as getChunkSizeBytes, k as getSessionTtlMs, l as computeSha256Hex, t as toBufferPayload, n as normalizeSha256Hex, m as getMaxClientUploadBytesPerSecond, o as getRawBodyLimitBytes } from "./uploads-BAxHzidK.js";
|
|
7
4
|
import { randomBytes } from "node:crypto";
|
|
8
5
|
import { o as object, n as number, b as boolean, s as string, a as array, _ as _enum } from "./schemas-Cjdjgehl.js";
|
|
9
|
-
const
|
|
10
|
-
const window = new JSDOM("").window;
|
|
11
|
-
const DOMPurify = createDOMPurify(window);
|
|
12
|
-
const normalizeForSniff = (raw) => raw.replace(/^\uFEFF/, "").trimStart();
|
|
13
|
-
const looksLikeSvgText = (text) => {
|
|
14
|
-
const normalized = normalizeForSniff(text);
|
|
15
|
-
if (!normalized.startsWith("<")) return false;
|
|
16
|
-
return /<svg(?:\s|>)/i.test(normalized);
|
|
17
|
-
};
|
|
18
|
-
const looksLikeSvg = (sniff) => looksLikeSvgText(sniff.toString("utf8"));
|
|
19
|
-
const sanitizeSvg = (svg) => DOMPurify.sanitize(svg, {
|
|
20
|
-
USE_PROFILES: {
|
|
21
|
-
svg: true,
|
|
22
|
-
svgFilters: true
|
|
23
|
-
}
|
|
24
|
-
});
|
|
25
|
-
const sanitizeSvgProcessor = {
|
|
26
|
-
id: "sanitize-svg",
|
|
27
|
-
maxBytes: MAX_SVG_BYTES,
|
|
28
|
-
match: ({
|
|
29
|
-
sniff
|
|
30
|
-
}) => looksLikeSvg(sniff),
|
|
31
|
-
process: (data) => {
|
|
32
|
-
if (data.length > MAX_SVG_BYTES) {
|
|
33
|
-
throw new Error("svg_too_large");
|
|
34
|
-
}
|
|
35
|
-
const svgText = data.toString("utf8");
|
|
36
|
-
if (!looksLikeSvgText(svgText)) {
|
|
37
|
-
throw new Error("svg_invalid");
|
|
38
|
-
}
|
|
39
|
-
const sanitized = sanitizeSvg(svgText);
|
|
40
|
-
if (!sanitized.trim() || !looksLikeSvgText(sanitized)) {
|
|
41
|
-
throw new Error("svg_sanitize_failed");
|
|
42
|
-
}
|
|
43
|
-
const sanitizedBuffer = Buffer.from(sanitized, "utf8");
|
|
44
|
-
if (sanitizedBuffer.length > MAX_SVG_BYTES) {
|
|
45
|
-
throw new Error("svg_too_large");
|
|
46
|
-
}
|
|
47
|
-
return {
|
|
48
|
-
data: sanitizedBuffer,
|
|
49
|
-
mimeType: "image/svg+xml"
|
|
50
|
-
};
|
|
51
|
-
}
|
|
52
|
-
};
|
|
53
|
-
const uploadProcessors = Object.freeze([sanitizeSvgProcessor]);
|
|
6
|
+
const uploadProcessors = Object.freeze([sanitizeSvgProcessor, convertHeifToWebpProcessor]);
|
|
54
7
|
const getMaxUploadProcessorBytes = () => uploadProcessors.reduce((max, processor) => Math.max(max, processor.maxBytes), 0);
|
|
55
8
|
const selectUploadProcessors = (ctx) => uploadProcessors.filter((processor) => processor.match(ctx));
|
|
56
9
|
const applyUploadProcessors = async (data, ctx) => {
|
|
57
10
|
let currentData = data;
|
|
11
|
+
let currentFilename = ctx.filename;
|
|
58
12
|
let currentMimeType = ctx.clientMimeType;
|
|
59
13
|
const applied = [];
|
|
14
|
+
const metadata = {};
|
|
60
15
|
for (const processor of uploadProcessors) {
|
|
61
16
|
const processorCtx = {
|
|
62
|
-
filename:
|
|
17
|
+
filename: currentFilename,
|
|
63
18
|
clientMimeType: currentMimeType,
|
|
64
19
|
totalSize: currentData.length,
|
|
65
20
|
sniff: currentData
|
|
@@ -70,15 +25,23 @@ const applyUploadProcessors = async (data, ctx) => {
|
|
|
70
25
|
}
|
|
71
26
|
const result = await processor.process(currentData, processorCtx);
|
|
72
27
|
currentData = result.data;
|
|
28
|
+
if (typeof result.filename === "string" && result.filename.trim()) {
|
|
29
|
+
currentFilename = result.filename.trim();
|
|
30
|
+
}
|
|
73
31
|
if (typeof result.mimeType === "string" && result.mimeType.trim()) {
|
|
74
32
|
currentMimeType = result.mimeType.trim();
|
|
75
33
|
}
|
|
34
|
+
if (result.metadata && typeof result.metadata === "object") {
|
|
35
|
+
Object.assign(metadata, result.metadata);
|
|
36
|
+
}
|
|
76
37
|
applied.push(processor.id);
|
|
77
38
|
}
|
|
78
39
|
return {
|
|
79
40
|
data: currentData,
|
|
41
|
+
filename: currentFilename,
|
|
80
42
|
mimeType: currentMimeType,
|
|
81
|
-
applied
|
|
43
|
+
applied,
|
|
44
|
+
metadata
|
|
82
45
|
};
|
|
83
46
|
};
|
|
84
47
|
const waitForStreamFinished = async (stream) => new Promise((resolve, reject) => {
|
|
@@ -218,7 +181,9 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
218
181
|
const declaredMimeType = locked.mimeType.trim().toLowerCase();
|
|
219
182
|
const declaredSvg = declaredMimeType === "image/svg+xml" || locked.filename.trim().toLowerCase().endsWith(".svg");
|
|
220
183
|
let uploadStream = null;
|
|
184
|
+
let finalFilename = locked.filename;
|
|
221
185
|
let finalMimeType = locked.mimeType;
|
|
186
|
+
let finalSize = locked.totalSize;
|
|
222
187
|
let inlineProcessors = [];
|
|
223
188
|
let finalMetadata = {
|
|
224
189
|
uploadId,
|
|
@@ -277,7 +242,7 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
277
242
|
sniff
|
|
278
243
|
});
|
|
279
244
|
if (processors.length) {
|
|
280
|
-
throw new Error("svg_too_large");
|
|
245
|
+
throw new Error(processors.some((processor) => processor.id === "sanitize-svg") ? "svg_too_large" : "processor_input_too_large");
|
|
281
246
|
}
|
|
282
247
|
finalMetadata = {
|
|
283
248
|
uploadId,
|
|
@@ -320,23 +285,28 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
320
285
|
const assembled = Buffer.concat(chunks, bufferedBytes);
|
|
321
286
|
const {
|
|
322
287
|
data: processed,
|
|
288
|
+
filename: processedFilename,
|
|
323
289
|
mimeType: processedMimeType,
|
|
324
|
-
applied
|
|
290
|
+
applied,
|
|
291
|
+
metadata: processedMetadata
|
|
325
292
|
} = await applyUploadProcessors(assembled, {
|
|
326
293
|
filename: locked.filename,
|
|
327
294
|
clientMimeType: locked.mimeType
|
|
328
295
|
});
|
|
296
|
+
finalFilename = processedFilename;
|
|
329
297
|
finalMimeType = processedMimeType;
|
|
298
|
+
finalSize = processed.length;
|
|
330
299
|
inlineProcessors = applied;
|
|
331
300
|
finalMetadata = {
|
|
332
301
|
uploadId,
|
|
333
302
|
tenantId,
|
|
334
303
|
mimeType: processedMimeType,
|
|
335
|
-
totalSize:
|
|
304
|
+
totalSize: processed.length,
|
|
336
305
|
...applied.length ? {
|
|
337
306
|
processors: applied,
|
|
338
|
-
|
|
307
|
+
sourceTotalSize: locked.totalSize
|
|
339
308
|
} : {},
|
|
309
|
+
...processedMetadata,
|
|
340
310
|
...typeof locked.isPublic === "boolean" ? {
|
|
341
311
|
isPublic: locked.isPublic
|
|
342
312
|
} : {},
|
|
@@ -347,7 +317,7 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
347
317
|
userId: lockedUserId
|
|
348
318
|
} : {}
|
|
349
319
|
};
|
|
350
|
-
uploadStream = bucket.openUploadStream(
|
|
320
|
+
uploadStream = bucket.openUploadStream(finalFilename, {
|
|
351
321
|
metadata: finalMetadata
|
|
352
322
|
});
|
|
353
323
|
const finished = waitForStreamFinished(uploadStream);
|
|
@@ -363,7 +333,7 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
363
333
|
sniff
|
|
364
334
|
});
|
|
365
335
|
if (processors.length) {
|
|
366
|
-
throw new Error("svg_too_large");
|
|
336
|
+
throw new Error(processors.some((processor) => processor.id === "sanitize-svg") ? "svg_too_large" : "processor_input_too_large");
|
|
367
337
|
}
|
|
368
338
|
finalMetadata = {
|
|
369
339
|
uploadId,
|
|
@@ -403,7 +373,10 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
403
373
|
}, {
|
|
404
374
|
$set: {
|
|
405
375
|
status: "done",
|
|
406
|
-
fileId
|
|
376
|
+
fileId,
|
|
377
|
+
filename: finalFilename,
|
|
378
|
+
mimeType: finalMimeType,
|
|
379
|
+
totalSize: finalSize
|
|
407
380
|
},
|
|
408
381
|
$unset: {
|
|
409
382
|
error: ""
|
|
@@ -413,10 +386,10 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
413
386
|
tenantId,
|
|
414
387
|
uploadId,
|
|
415
388
|
fileId,
|
|
416
|
-
filename:
|
|
389
|
+
filename: finalFilename,
|
|
417
390
|
mimeType: finalMimeType,
|
|
418
391
|
clientMimeType: locked.mimeType,
|
|
419
|
-
totalSize:
|
|
392
|
+
totalSize: finalSize,
|
|
420
393
|
...typeof locked.isPublic === "boolean" ? {
|
|
421
394
|
isPublic: locked.isPublic
|
|
422
395
|
} : {},
|
|
@@ -482,6 +455,40 @@ const completeUpload = async (_payload, ctx) => {
|
|
|
482
455
|
error: message
|
|
483
456
|
};
|
|
484
457
|
}
|
|
458
|
+
if (message === "heif_too_large" || message === "processor_input_too_large") {
|
|
459
|
+
await UploadSession.updateOne({
|
|
460
|
+
$and: [{
|
|
461
|
+
_id: uploadId
|
|
462
|
+
}, getUploadSessionAccessQuery(ability, "update")]
|
|
463
|
+
}, {
|
|
464
|
+
$set: {
|
|
465
|
+
status: "error",
|
|
466
|
+
error: message
|
|
467
|
+
}
|
|
468
|
+
});
|
|
469
|
+
ctx.res.status(413);
|
|
470
|
+
return {
|
|
471
|
+
ok: false,
|
|
472
|
+
error: message
|
|
473
|
+
};
|
|
474
|
+
}
|
|
475
|
+
if (message.startsWith("heif_decode_unsupported")) {
|
|
476
|
+
await UploadSession.updateOne({
|
|
477
|
+
$and: [{
|
|
478
|
+
_id: uploadId
|
|
479
|
+
}, getUploadSessionAccessQuery(ability, "update")]
|
|
480
|
+
}, {
|
|
481
|
+
$set: {
|
|
482
|
+
status: "error",
|
|
483
|
+
error: message
|
|
484
|
+
}
|
|
485
|
+
});
|
|
486
|
+
ctx.res.status(415);
|
|
487
|
+
return {
|
|
488
|
+
ok: false,
|
|
489
|
+
error: message
|
|
490
|
+
};
|
|
491
|
+
}
|
|
485
492
|
if (message === "svg_invalid" || message === "svg_sanitize_failed") {
|
|
486
493
|
await UploadSession.updateOne({
|
|
487
494
|
$and: [{
|
|
@@ -928,4 +935,4 @@ const handler = (api) => {
|
|
|
928
935
|
export {
|
|
929
936
|
handler as default
|
|
930
937
|
};
|
|
931
|
-
//# sourceMappingURL=handler-
|
|
938
|
+
//# sourceMappingURL=handler-Da2KGCRq.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"handler-Da2KGCRq.js","sources":["../src/uploads/api/file-uploads/processors/index.ts","../src/uploads/api/file-uploads/handlers/completeUpload.ts","../src/uploads/api/file-uploads/handlers/getStatus.ts","../src/uploads/api/file-uploads/index.ts","../src/uploads/api/file-uploads/handlers/initUpload.ts","../src/uploads/api/file-uploads/handlers/uploadChunk.ts","../src/uploads/api/file-uploads/middleware/rawBodyParser.ts","../src/uploads/api/file-uploads/handler.ts"],"sourcesContent":["import { convertHeifToWebpProcessor } from \"./convertHeifToWebp\"\nimport { sanitizeSvgProcessor } from \"./sanitizeSvg\"\n\n\nexport type UploadFileProcessorContext = {\n filename: string\n clientMimeType: string\n totalSize: number\n sniff: Buffer\n}\n\nexport type UploadFileProcessorResult = {\n data: Buffer\n filename?: string\n mimeType?: string\n metadata?: Record<string, unknown>\n}\n\nexport type UploadFileProcessor = {\n id: string\n maxBytes: number\n match: (ctx: UploadFileProcessorContext) => boolean\n process: (data: Buffer, ctx: UploadFileProcessorContext) => Promise<UploadFileProcessorResult> | UploadFileProcessorResult\n}\n\nexport const uploadProcessors = Object.freeze([sanitizeSvgProcessor, convertHeifToWebpProcessor] satisfies UploadFileProcessor[])\n\nexport const getMaxUploadProcessorBytes = (): number =>\n uploadProcessors.reduce((max, processor) => Math.max(max, processor.maxBytes), 0)\n\nexport const selectUploadProcessors = (ctx: UploadFileProcessorContext): UploadFileProcessor[] =>\n uploadProcessors.filter((processor) => processor.match(ctx))\n\nexport const applyUploadProcessors = async (\n data: Buffer,\n ctx: Omit<UploadFileProcessorContext, \"sniff\" | \"totalSize\">,\n): Promise<{ data: Buffer; filename: string; mimeType: string; applied: string[]; metadata: Record<string, unknown> }> => {\n let currentData = data\n let currentFilename = ctx.filename\n let currentMimeType = ctx.clientMimeType\n const applied: string[] = []\n const metadata: Record<string, unknown> = {}\n\n for (const processor of uploadProcessors) {\n const processorCtx: UploadFileProcessorContext = {\n filename: currentFilename,\n clientMimeType: currentMimeType,\n totalSize: currentData.length,\n sniff: currentData,\n }\n\n if (!processor.match(processorCtx)) continue\n\n if (currentData.length > processor.maxBytes) {\n throw new Error(\"processor_input_too_large\")\n }\n\n const result = await processor.process(currentData, processorCtx)\n currentData = result.data\n if (typeof result.filename === \"string\" && result.filename.trim()) {\n currentFilename = result.filename.trim()\n }\n if (typeof result.mimeType === \"string\" && result.mimeType.trim()) {\n currentMimeType = result.mimeType.trim()\n }\n if (result.metadata && typeof result.metadata === \"object\") {\n Object.assign(metadata, result.metadata)\n }\n applied.push(processor.id)\n }\n\n return {\n data: currentData,\n filename: currentFilename,\n mimeType: currentMimeType,\n applied,\n metadata,\n }\n}\n","import { ApiHandler } from \"@rpcbase/api\"\nimport { getTenantFilesystemDb, models } from \"@rpcbase/db\"\nimport { GridFSBucket } from \"mongodb\"\nimport type { Model } from \"mongoose\"\n\nimport * as Uploads from \"../index\"\nimport { enqueueUploadPostProcessors } from \"../postProcessors\"\nimport { applyUploadProcessors, getMaxUploadProcessorBytes, selectUploadProcessors } from \"../processors\"\nimport {\n type SessionUser,\n type UploadChunkDoc,\n type UploadSessionDoc,\n buildUploadsAbility,\n ensureUploadIndexes,\n getBucketName,\n getModelCtx,\n getUploadSessionAccessQuery,\n getTenantId,\n} from \"../shared\"\n\n\nconst waitForStreamFinished = async (stream: NodeJS.WritableStream): Promise<void> => new Promise((resolve, reject) => {\n stream.once(\"finish\", resolve)\n stream.once(\"error\", reject)\n})\n\nconst writeToStream = async (stream: NodeJS.WritableStream, chunk: Buffer): Promise<void> => {\n const ok = stream.write(chunk)\n if (ok) return\n await new Promise<void>((resolve, reject) => {\n const onDrain = () => {\n cleanup()\n resolve()\n }\n\n const onError = (error: unknown) => {\n cleanup()\n reject(error)\n }\n\n const cleanup = () => {\n stream.off(\"drain\", onDrain)\n stream.off(\"error\", onError)\n }\n\n stream.on(\"drain\", onDrain)\n stream.on(\"error\", onError)\n })\n}\n\nconst abortUploadStream = async (stream: unknown): Promise<void> => {\n if (!stream) return\n if (typeof (stream as { abort?: unknown }).abort === \"function\") {\n try {\n await (stream as { abort: () => Promise<void> | void }).abort()\n return\n } catch {\n //\n }\n }\n try {\n ;(stream as { destroy?: () => void }).destroy?.()\n } catch {\n //\n }\n}\n\nexport const completeUpload: ApiHandler<Record<string, never>, Uploads.CompleteResponsePayload, SessionUser> = async (\n _payload,\n ctx,\n): Promise<Uploads.CompleteResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const uploadId = String(ctx.req.params?.uploadId ?? \"\").trim()\n if (!uploadId) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_upload_id\" }\n }\n\n const ability = buildUploadsAbility(ctx, tenantId)\n const modelCtx = getModelCtx(ctx, tenantId, ability)\n\n const [UploadSession, UploadChunk] = await Promise.all([\n models.get(\"RBUploadSession\", modelCtx) as Promise<Model<UploadSessionDoc>>,\n models.get(\"RBUploadChunk\", modelCtx) as Promise<Model<UploadChunkDoc>>,\n ])\n\n if (!ability.can(\"update\", \"RBUploadSession\")) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n const existing = await UploadSession.findOne({ $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"read\")] }).lean()\n if (!existing) {\n ctx.res.status(404)\n return { ok: false, error: \"not_found\" }\n }\n\n if (existing.status === \"done\" && existing.fileId) {\n return { ok: true, fileId: existing.fileId }\n }\n\n const locked = await UploadSession.findOneAndUpdate(\n { $and: [{ _id: uploadId }, { status: \"uploading\" }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"assembling\" }, $unset: { error: \"\" } },\n { returnDocument: \"after\" },\n ).lean()\n\n if (!locked) {\n ctx.res.status(409)\n return { ok: false, error: \"not_uploading\" }\n }\n\n await ensureUploadIndexes(UploadSession, UploadChunk)\n\n const fsDb = await getTenantFilesystemDb(tenantId)\n const nativeDb = fsDb.db\n if (!nativeDb) {\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"error\", error: \"filesystem_db_unavailable\" } },\n )\n ctx.res.status(500)\n return { ok: false, error: \"assembly_failed\" }\n }\n const bucketName = getBucketName()\n const bucket = new GridFSBucket(nativeDb, { bucketName })\n\n const lockedUserId = typeof locked.userId === \"string\" ? locked.userId : undefined\n const maxProcessorBytes = getMaxUploadProcessorBytes()\n const shouldBufferForProcessing = locked.totalSize <= maxProcessorBytes\n const declaredMimeType = locked.mimeType.trim().toLowerCase()\n const declaredSvg = declaredMimeType === \"image/svg+xml\" || locked.filename.trim().toLowerCase().endsWith(\".svg\")\n\n let uploadStream: NodeJS.WritableStream | null = null\n let finalFilename = locked.filename\n let finalMimeType = locked.mimeType\n let finalSize = locked.totalSize\n let inlineProcessors: string[] = []\n let finalMetadata: Record<string, unknown> = {\n uploadId,\n tenantId,\n mimeType: locked.mimeType,\n totalSize: locked.totalSize,\n ...(typeof locked.isPublic === \"boolean\" ? { isPublic: locked.isPublic } : {}),\n ...(typeof locked.ownerKeyHash === \"string\" ? { ownerKeyHash: locked.ownerKeyHash } : {}),\n ...(lockedUserId ? { userId: lockedUserId } : {}),\n }\n\n try {\n if (!shouldBufferForProcessing && declaredSvg) {\n throw new Error(\"svg_too_large\")\n }\n\n const cursor = UploadChunk.find({ uploadId }).sort({ index: 1 }).cursor() as unknown as AsyncIterable<UploadChunkDoc> & {\n close: () => Promise<void>\n }\n\n let expectedIndex = 0\n const chunks: Buffer[] = []\n let bufferedBytes = 0\n\n const pendingChunks: Buffer[] = []\n const sniffParts: Buffer[] = []\n let sniffBytes = 0\n\n try {\n for await (const chunkDoc of cursor) {\n if (chunkDoc.index !== expectedIndex) {\n throw new Error(\"missing_chunks\")\n }\n\n const chunk = chunkDoc.data\n\n if (shouldBufferForProcessing) {\n chunks.push(chunk)\n bufferedBytes += chunk.length\n } else if (!uploadStream) {\n pendingChunks.push(chunk)\n\n if (sniffBytes < maxProcessorBytes) {\n const slice = chunk.subarray(0, Math.min(chunk.length, maxProcessorBytes - sniffBytes))\n if (slice.length) {\n sniffParts.push(slice)\n sniffBytes += slice.length\n }\n }\n\n if (sniffBytes >= maxProcessorBytes) {\n const sniff = Buffer.concat(sniffParts, sniffBytes)\n const processors = selectUploadProcessors({\n filename: locked.filename,\n clientMimeType: locked.mimeType,\n totalSize: locked.totalSize,\n sniff,\n })\n\n if (processors.length) {\n throw new Error(processors.some((processor) => processor.id === \"sanitize-svg\") ? \"svg_too_large\" : \"processor_input_too_large\")\n }\n\n finalMetadata = {\n uploadId,\n tenantId,\n mimeType: locked.mimeType,\n totalSize: locked.totalSize,\n ...(typeof locked.isPublic === \"boolean\" ? { isPublic: locked.isPublic } : {}),\n ...(typeof locked.ownerKeyHash === \"string\" ? { ownerKeyHash: locked.ownerKeyHash } : {}),\n ...(lockedUserId ? { userId: lockedUserId } : {}),\n }\n uploadStream = bucket.openUploadStream(locked.filename, {\n metadata: finalMetadata,\n })\n\n for (const pending of pendingChunks) {\n await writeToStream(uploadStream, pending)\n }\n pendingChunks.length = 0\n }\n } else {\n await writeToStream(uploadStream, chunk)\n }\n\n expectedIndex += 1\n }\n } finally {\n try {\n await cursor.close()\n } catch {\n //\n }\n }\n\n if (expectedIndex !== locked.chunksTotal) {\n throw new Error(\"missing_chunks\")\n }\n\n if (shouldBufferForProcessing) {\n const assembled = Buffer.concat(chunks, bufferedBytes)\n const {\n data: processed,\n filename: processedFilename,\n mimeType: processedMimeType,\n applied,\n metadata: processedMetadata,\n } = await applyUploadProcessors(assembled, {\n filename: locked.filename,\n clientMimeType: locked.mimeType,\n })\n\n finalFilename = processedFilename\n finalMimeType = processedMimeType\n finalSize = processed.length\n inlineProcessors = applied\n finalMetadata = {\n uploadId,\n tenantId,\n mimeType: processedMimeType,\n totalSize: processed.length,\n ...(applied.length ? { processors: applied, sourceTotalSize: locked.totalSize } : {}),\n ...processedMetadata,\n ...(typeof locked.isPublic === \"boolean\" ? { isPublic: locked.isPublic } : {}),\n ...(typeof locked.ownerKeyHash === \"string\" ? { ownerKeyHash: locked.ownerKeyHash } : {}),\n ...(lockedUserId ? { userId: lockedUserId } : {}),\n }\n uploadStream = bucket.openUploadStream(finalFilename, {\n metadata: finalMetadata,\n })\n\n const finished = waitForStreamFinished(uploadStream)\n uploadStream.end(processed)\n await finished\n } else {\n if (!uploadStream) {\n const sniff = Buffer.concat(sniffParts, sniffBytes)\n const processors = selectUploadProcessors({\n filename: locked.filename,\n clientMimeType: locked.mimeType,\n totalSize: locked.totalSize,\n sniff,\n })\n\n if (processors.length) {\n throw new Error(processors.some((processor) => processor.id === \"sanitize-svg\") ? \"svg_too_large\" : \"processor_input_too_large\")\n }\n\n finalMetadata = {\n uploadId,\n tenantId,\n mimeType: locked.mimeType,\n totalSize: locked.totalSize,\n ...(typeof locked.isPublic === \"boolean\" ? { isPublic: locked.isPublic } : {}),\n ...(typeof locked.ownerKeyHash === \"string\" ? { ownerKeyHash: locked.ownerKeyHash } : {}),\n ...(lockedUserId ? { userId: lockedUserId } : {}),\n }\n uploadStream = bucket.openUploadStream(locked.filename, {\n metadata: finalMetadata,\n })\n\n for (const pending of pendingChunks) {\n await writeToStream(uploadStream, pending)\n }\n pendingChunks.length = 0\n }\n\n const finished = waitForStreamFinished(uploadStream)\n uploadStream.end()\n await finished\n }\n\n const fileId = String((uploadStream as unknown as { id?: unknown }).id ?? \"\")\n if (!fileId) {\n throw new Error(\"missing_file_id\")\n }\n\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"done\", fileId, filename: finalFilename, mimeType: finalMimeType, totalSize: finalSize }, $unset: { error: \"\" } },\n )\n\n await enqueueUploadPostProcessors({\n tenantId,\n uploadId,\n fileId,\n filename: finalFilename,\n mimeType: finalMimeType,\n clientMimeType: locked.mimeType,\n totalSize: finalSize,\n ...(typeof locked.isPublic === \"boolean\" ? { isPublic: locked.isPublic } : {}),\n ...(typeof locked.ownerKeyHash === \"string\" ? { ownerKeyHash: locked.ownerKeyHash } : {}),\n ...(lockedUserId ? { userId: lockedUserId } : {}),\n inlineProcessors,\n metadata: finalMetadata,\n }).catch((error) => {\n console.error(\"Upload post processor enqueue failed\", {\n tenantId,\n uploadId,\n fileId,\n error,\n })\n })\n\n try {\n await UploadChunk.deleteMany({ uploadId })\n } catch {\n //\n }\n\n return { ok: true, fileId }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n\n await abortUploadStream(uploadStream)\n\n if (message === \"missing_chunks\") {\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"uploading\" } },\n )\n ctx.res.status(409)\n return { ok: false, error: \"missing_chunks\" }\n }\n\n if (message === \"svg_too_large\") {\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"error\", error: message } },\n )\n ctx.res.status(413)\n return { ok: false, error: message }\n }\n\n if (message === \"heif_too_large\" || message === \"processor_input_too_large\") {\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"error\", error: message } },\n )\n ctx.res.status(413)\n return { ok: false, error: message }\n }\n\n if (message.startsWith(\"heif_decode_unsupported\")) {\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"error\", error: message } },\n )\n ctx.res.status(415)\n return { ok: false, error: message }\n }\n\n if (message === \"svg_invalid\" || message === \"svg_sanitize_failed\") {\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"error\", error: message } },\n )\n ctx.res.status(400)\n return { ok: false, error: message }\n }\n\n await UploadSession.updateOne(\n { $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] },\n { $set: { status: \"error\", error: message } },\n )\n\n ctx.res.status(500)\n return { ok: false, error: \"assembly_failed\" }\n }\n}\n","import { ApiHandler } from \"@rpcbase/api\"\nimport { models } from \"@rpcbase/db\"\nimport type { Model } from \"mongoose\"\n\nimport * as Uploads from \"../index\"\nimport {\n type SessionUser,\n type UploadChunkDoc,\n type UploadSessionDoc,\n buildUploadsAbility,\n getModelCtx,\n getUploadSessionAccessQuery,\n getTenantId,\n} from \"../shared\"\n\n\nexport const getStatus: ApiHandler<Record<string, never>, Uploads.StatusResponsePayload, SessionUser> = async (\n _payload,\n ctx,\n): Promise<Uploads.StatusResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const uploadId = String(ctx.req.params?.uploadId ?? \"\").trim()\n if (!uploadId) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_upload_id\" }\n }\n\n const ability = buildUploadsAbility(ctx, tenantId)\n const modelCtx = getModelCtx(ctx, tenantId, ability)\n\n const [UploadSession, UploadChunk] = await Promise.all([\n models.get(\"RBUploadSession\", modelCtx) as Promise<Model<UploadSessionDoc>>,\n models.get(\"RBUploadChunk\", modelCtx) as Promise<Model<UploadChunkDoc>>,\n ])\n\n if (!ability.can(\"read\", \"RBUploadSession\")) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n const session = await UploadSession.findOne({ $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"read\")] }).lean()\n if (!session) {\n ctx.res.status(404)\n return { ok: false, error: \"not_found\" }\n }\n\n const receivedDocs = await UploadChunk.find(\n { uploadId },\n { index: 1, _id: 0 },\n ).sort({ index: 1 }).lean()\n\n const received = (receivedDocs as unknown as Array<{ index?: unknown }>)\n .map((doc) => (typeof doc.index === \"number\" ? doc.index : -1))\n .filter((n) => Number.isInteger(n) && n >= 0)\n\n return {\n ok: true,\n status: session.status,\n chunkSize: session.chunkSize,\n chunksTotal: session.chunksTotal,\n received,\n ...(session.fileId ? { fileId: session.fileId } : {}),\n }\n}\n","import { z } from \"zod\"\n\n\nexport const InitRoute = \"/api/rb/file-uploads\"\nexport const ChunkRoute = \"/api/rb/file-uploads/:uploadId/chunks/:index\"\nexport const StatusRoute = \"/api/rb/file-uploads/:uploadId/status\"\nexport const CompleteRoute = \"/api/rb/file-uploads/:uploadId/complete\"\n\nexport const initRequestSchema = z.object({\n filename: z.string().min(1),\n mimeType: z.string().min(1),\n isPublic: z.boolean().optional(),\n totalSize: z.number().int().min(1),\n})\n\nexport type InitRequestPayload = z.infer<typeof initRequestSchema>\n\nexport const initResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n uploadId: z.string().optional(),\n uploadKey: z.string().optional(),\n chunkSize: z.number().int().optional(),\n chunksTotal: z.number().int().optional(),\n})\n\nexport type InitResponsePayload = z.infer<typeof initResponseSchema>\n\nexport const statusResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n status: z.enum([\"uploading\", \"assembling\", \"done\", \"error\"]).optional(),\n chunkSize: z.number().int().optional(),\n chunksTotal: z.number().int().optional(),\n received: z.array(z.number().int().min(0)).optional(),\n fileId: z.string().optional(),\n})\n\nexport type StatusResponsePayload = z.infer<typeof statusResponseSchema>\n\nexport const completeResponseSchema = z.object({\n ok: z.boolean(),\n error: z.string().optional(),\n fileId: z.string().optional(),\n})\n\nexport type CompleteResponsePayload = z.infer<typeof completeResponseSchema>\n","import { randomBytes } from \"node:crypto\"\n\nimport { ApiHandler } from \"@rpcbase/api\"\nimport { models } from \"@rpcbase/db\"\nimport { ObjectId } from \"mongodb\"\nimport type { Model } from \"mongoose\"\n\nimport * as Uploads from \"../index\"\nimport {\n type SessionUser,\n type UploadChunkDoc,\n type UploadSessionDoc,\n buildUploadsAbility,\n computeSha256Hex,\n ensureUploadIndexes,\n getChunkSizeBytes,\n getModelCtx,\n getSessionTtlMs,\n getTenantId,\n getUserId,\n} from \"../shared\"\n\n\nexport const initUpload: ApiHandler<Uploads.InitRequestPayload, Uploads.InitResponsePayload, SessionUser> = async (\n payload,\n ctx,\n): Promise<Uploads.InitResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const userId = getUserId(ctx)\n\n const parsed = Uploads.initRequestSchema.safeParse(payload ?? {})\n if (!parsed.success) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_payload\" }\n }\n\n const chunkSize = getChunkSizeBytes()\n const { filename, mimeType, totalSize, isPublic } = parsed.data\n const chunksTotal = Math.ceil(totalSize / chunkSize)\n\n const ability = buildUploadsAbility(ctx, tenantId)\n const modelCtx = getModelCtx(ctx, tenantId, ability)\n\n const [UploadSession, UploadChunk] = await Promise.all([\n models.get(\"RBUploadSession\", modelCtx) as Promise<Model<UploadSessionDoc>>,\n models.get(\"RBUploadChunk\", modelCtx) as Promise<Model<UploadChunkDoc>>,\n ])\n\n await ensureUploadIndexes(UploadSession, UploadChunk)\n\n const uploadId = new ObjectId().toString()\n const now = Date.now()\n const expiresAt = new Date(now + getSessionTtlMs())\n\n const uploadKey = userId ? null : randomBytes(32).toString(\"base64url\")\n const ownerKeyHash = uploadKey ? computeSha256Hex(Buffer.from(uploadKey)) : undefined\n\n await UploadSession.create({\n _id: uploadId,\n ...(userId ? { userId } : {}),\n ...(ownerKeyHash ? { ownerKeyHash } : {}),\n filename,\n mimeType,\n ...(typeof isPublic === \"boolean\" ? { isPublic } : {}),\n totalSize,\n chunkSize,\n chunksTotal,\n status: \"uploading\",\n createdAt: new Date(now),\n expiresAt,\n })\n\n return {\n ok: true,\n uploadId,\n chunkSize,\n chunksTotal,\n ...(uploadKey ? { uploadKey } : {}),\n }\n}\n","import { ApiHandler } from \"@rpcbase/api\"\nimport { models } from \"@rpcbase/db\"\nimport type { Model } from \"mongoose\"\n\nimport {\n type SessionUser,\n type UploadChunkDoc,\n type UploadSessionDoc,\n buildUploadsAbility,\n computeSha256Hex,\n ensureUploadIndexes,\n getModelCtx,\n getUploadSessionAccessQuery,\n getTenantId,\n normalizeSha256Hex,\n toBufferPayload,\n} from \"../shared\"\n\n\ntype ChunkResponsePayload = {\n ok: boolean\n error?: string\n}\n\nexport const uploadChunk: ApiHandler<Buffer, ChunkResponsePayload, SessionUser> = async (\n payload,\n ctx,\n): Promise<ChunkResponsePayload> => {\n const tenantId = getTenantId(ctx)\n if (!tenantId) {\n ctx.res.status(400)\n return { ok: false, error: \"tenant_missing\" }\n }\n\n const uploadId = String(ctx.req.params?.uploadId ?? \"\").trim()\n const indexRaw = String(ctx.req.params?.index ?? \"\").trim()\n const index = Number(indexRaw)\n\n if (!uploadId || !Number.isInteger(index) || index < 0) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_chunk_ref\" }\n }\n\n const ability = buildUploadsAbility(ctx, tenantId)\n const modelCtx = getModelCtx(ctx, tenantId, ability)\n\n const [UploadSession, UploadChunk] = await Promise.all([\n models.get(\"RBUploadSession\", modelCtx) as Promise<Model<UploadSessionDoc>>,\n models.get(\"RBUploadChunk\", modelCtx) as Promise<Model<UploadChunkDoc>>,\n ])\n\n if (!ability.can(\"update\", \"RBUploadSession\")) {\n ctx.res.status(401)\n return { ok: false, error: \"unauthorized\" }\n }\n\n const session = await UploadSession.findOne({ $and: [{ _id: uploadId }, getUploadSessionAccessQuery(ability, \"update\")] }).lean()\n if (!session) {\n ctx.res.status(404)\n return { ok: false, error: \"not_found\" }\n }\n\n if (session.status !== \"uploading\") {\n ctx.res.status(409)\n return { ok: false, error: \"not_uploading\" }\n }\n\n if (index >= session.chunksTotal) {\n ctx.res.status(400)\n return { ok: false, error: \"index_out_of_range\" }\n }\n\n const data = toBufferPayload(payload)\n if (!data) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_body\" }\n }\n\n const expectedSize = index === session.chunksTotal - 1\n ? session.totalSize - session.chunkSize * (session.chunksTotal - 1)\n : session.chunkSize\n\n if (data.length > expectedSize) {\n ctx.res.status(413)\n return { ok: false, error: \"chunk_too_large\" }\n }\n\n if (data.length !== expectedSize) {\n ctx.res.status(400)\n return { ok: false, error: \"invalid_chunk_size\" }\n }\n\n const checksumHeader = ctx.req.get(\"X-Chunk-SHA256\")\n const sha256 = checksumHeader ? computeSha256Hex(data) : undefined\n\n if (checksumHeader) {\n const expectedSha256 = normalizeSha256Hex(checksumHeader)\n if (sha256 !== expectedSha256) {\n ctx.res.status(400)\n return { ok: false, error: \"checksum_mismatch\" }\n }\n }\n\n await ensureUploadIndexes(UploadSession, UploadChunk)\n\n await UploadChunk.updateOne(\n { uploadId, index },\n {\n $set: {\n uploadId,\n index,\n data,\n size: data.length,\n sha256,\n expiresAt: session.expiresAt,\n },\n $setOnInsert: {\n createdAt: new Date(),\n },\n },\n { upsert: true },\n )\n\n ctx.res.status(204)\n return { ok: true }\n}\n","export const rawBodyParser = ({\n limitBytes,\n maxClientBytesPerSecond,\n}: {\n limitBytes: number\n maxClientBytesPerSecond?: number | null\n}) => {\n return (req: any, res: any, next: any) => {\n const contentType = typeof req?.headers?.[\"content-type\"] === \"string\"\n ? String(req.headers[\"content-type\"])\n : \"\"\n\n if (!contentType.includes(\"application/octet-stream\")) {\n next()\n return\n }\n\n let total = 0\n const chunks: Buffer[] = []\n let done = false\n let paused = false\n let throttleTimeout: ReturnType<typeof setTimeout> | null = null\n\n const rateBytesPerSecond = typeof maxClientBytesPerSecond === \"number\" && maxClientBytesPerSecond > 0\n ? maxClientBytesPerSecond\n : null\n\n const cleanup = () => {\n req.off(\"data\", onData)\n req.off(\"end\", onEnd)\n req.off(\"error\", onError)\n req.off(\"aborted\", onAborted)\n if (throttleTimeout) {\n clearTimeout(throttleTimeout)\n throttleTimeout = null\n }\n }\n\n const finish = (error?: unknown) => {\n if (done) return\n done = true\n\n cleanup()\n\n if (error) {\n next(error)\n return\n }\n\n req.body = Buffer.concat(chunks, total)\n next()\n }\n\n const onData = (chunk: any) => {\n if (done) return\n const buffer = Buffer.isBuffer(chunk) ? chunk : Buffer.from(chunk)\n total += buffer.length\n\n if (total > limitBytes) {\n done = true\n cleanup()\n req.destroy()\n res.status(413).json({ ok: false, error: \"chunk_too_large\" })\n return\n }\n\n chunks.push(buffer)\n\n if (!rateBytesPerSecond) return\n\n const now = Date.now()\n const clientKey = getClientKey(req)\n const state = getClientRateState(clientKey, rateBytesPerSecond, now)\n const waitMs = consumeRateBudget(state, buffer.length, rateBytesPerSecond, now)\n\n if (waitMs > 0 && !paused) {\n paused = true\n req.pause()\n throttleTimeout = setTimeout(() => {\n throttleTimeout = null\n paused = false\n if (done) return\n try {\n req.resume()\n } catch {\n //\n }\n }, waitMs)\n }\n }\n\n const onEnd = () => finish()\n const onError = (err: unknown) => finish(err)\n const onAborted = () => finish(new Error(\"request_aborted\"))\n\n req.on(\"data\", onData)\n req.on(\"end\", onEnd)\n req.on(\"error\", onError)\n req.on(\"aborted\", onAborted)\n }\n}\n\ntype ClientRateState = {\n tokens: number\n lastRefillMs: number\n lastSeenMs: number\n}\n\nconst MAX_BURST_SECONDS = 1\nconst STALE_CLIENT_MS = 15 * 60 * 1000\n\nconst clientRateStates = new Map<string, ClientRateState>()\nlet lastCleanupMs = 0\n\nconst getClientKey = (req: any): string => {\n const rawClientIp = typeof req?.clientIp === \"string\" ? req.clientIp : \"\"\n if (rawClientIp.trim()) return rawClientIp.trim()\n const rawIp = typeof req?.ip === \"string\" ? req.ip : \"\"\n return rawIp.trim() || \"unknown\"\n}\n\nconst maybeCleanupStates = (now: number) => {\n if (now - lastCleanupMs < 60_000) return\n lastCleanupMs = now\n\n if (clientRateStates.size < 2000) return\n\n for (const [key, state] of clientRateStates) {\n if (now - state.lastSeenMs > STALE_CLIENT_MS) {\n clientRateStates.delete(key)\n }\n }\n}\n\nconst getClientRateState = (key: string, rateBytesPerSecond: number, now: number): ClientRateState => {\n maybeCleanupStates(now)\n\n const capacity = rateBytesPerSecond * MAX_BURST_SECONDS\n const existing = clientRateStates.get(key)\n if (existing) {\n existing.lastSeenMs = now\n existing.tokens = Math.min(capacity, existing.tokens)\n return existing\n }\n\n const next: ClientRateState = {\n tokens: capacity,\n lastRefillMs: now,\n lastSeenMs: now,\n }\n clientRateStates.set(key, next)\n return next\n}\n\nconst consumeRateBudget = (\n state: ClientRateState,\n bytes: number,\n rateBytesPerSecond: number,\n now: number,\n): number => {\n const capacity = rateBytesPerSecond * MAX_BURST_SECONDS\n const elapsedMs = Math.max(0, now - state.lastRefillMs)\n\n if (elapsedMs > 0) {\n state.tokens = Math.min(capacity, state.tokens + (elapsedMs * rateBytesPerSecond) / 1000)\n state.lastRefillMs = now\n }\n\n state.tokens -= bytes\n\n if (state.tokens >= 0) return 0\n return Math.ceil((-state.tokens / rateBytesPerSecond) * 1000)\n}\n","import { Api } from \"@rpcbase/api\"\n\nimport { completeUpload } from \"./handlers/completeUpload\"\nimport { getStatus } from \"./handlers/getStatus\"\nimport { initUpload } from \"./handlers/initUpload\"\nimport { uploadChunk } from \"./handlers/uploadChunk\"\nimport { rawBodyParser } from \"./middleware/rawBodyParser\"\nimport { getChunkSizeBytes, getMaxClientUploadBytesPerSecond, getRawBodyLimitBytes, type SessionUser } from \"./shared\"\n\nimport * as Uploads from \"./index\"\n\n\nexport default (api: Api<SessionUser>) => {\n const chunkSizeBytes = getChunkSizeBytes()\n api.use(\n Uploads.InitRoute,\n rawBodyParser({\n limitBytes: getRawBodyLimitBytes(chunkSizeBytes),\n maxClientBytesPerSecond: getMaxClientUploadBytesPerSecond(),\n }),\n )\n\n api.post(Uploads.InitRoute, initUpload)\n api.put(Uploads.ChunkRoute, uploadChunk)\n api.get(Uploads.StatusRoute, getStatus)\n api.post(Uploads.CompleteRoute, completeUpload)\n}\n"],"names":["uploadProcessors","Object","freeze","sanitizeSvgProcessor","convertHeifToWebpProcessor","getMaxUploadProcessorBytes","reduce","max","processor","Math","maxBytes","selectUploadProcessors","ctx","filter","match","applyUploadProcessors","data","currentData","currentFilename","filename","currentMimeType","clientMimeType","applied","metadata","processorCtx","totalSize","length","sniff","Error","result","process","trim","mimeType","assign","push","id","waitForStreamFinished","stream","Promise","resolve","reject","once","writeToStream","chunk","ok","write","onDrain","cleanup","onError","error","off","on","abortUploadStream","abort","destroy","completeUpload","_payload","tenantId","getTenantId","res","status","uploadId","String","req","params","ability","buildUploadsAbility","modelCtx","getModelCtx","UploadSession","UploadChunk","all","models","get","can","existing","findOne","$and","_id","getUploadSessionAccessQuery","lean","fileId","locked","findOneAndUpdate","$set","$unset","returnDocument","ensureUploadIndexes","fsDb","getTenantFilesystemDb","nativeDb","db","updateOne","bucketName","getBucketName","bucket","GridFSBucket","lockedUserId","userId","undefined","maxProcessorBytes","shouldBufferForProcessing","declaredMimeType","toLowerCase","declaredSvg","endsWith","uploadStream","finalFilename","finalMimeType","finalSize","inlineProcessors","finalMetadata","isPublic","ownerKeyHash","cursor","find","sort","index","expectedIndex","chunks","bufferedBytes","pendingChunks","sniffParts","sniffBytes","chunkDoc","slice","subarray","min","Buffer","concat","processors","some","openUploadStream","pending","close","chunksTotal","assembled","processed","processedFilename","processedMimeType","processedMetadata","sourceTotalSize","finished","end","enqueueUploadPostProcessors","catch","console","deleteMany","message","startsWith","getStatus","session","receivedDocs","received","map","doc","n","Number","isInteger","chunkSize","InitRoute","ChunkRoute","StatusRoute","CompleteRoute","initRequestSchema","z","string","boolean","optional","number","int","uploadKey","initUpload","payload","getUserId","parsed","Uploads","safeParse","success","getChunkSizeBytes","ceil","ObjectId","toString","now","Date","expiresAt","getSessionTtlMs","randomBytes","computeSha256Hex","from","create","createdAt","uploadChunk","indexRaw","toBufferPayload","expectedSize","checksumHeader","sha256","expectedSha256","normalizeSha256Hex","size","$setOnInsert","upsert","rawBodyParser","limitBytes","maxClientBytesPerSecond","next","contentType","headers","includes","total","done","paused","throttleTimeout","rateBytesPerSecond","onData","onEnd","onAborted","clearTimeout","finish","body","buffer","isBuffer","json","clientKey","getClientKey","state","getClientRateState","waitMs","consumeRateBudget","pause","setTimeout","resume","err","MAX_BURST_SECONDS","STALE_CLIENT_MS","clientRateStates","Map","lastCleanupMs","rawClientIp","clientIp","rawIp","ip","maybeCleanupStates","key","lastSeenMs","delete","capacity","tokens","lastRefillMs","set","bytes","elapsedMs","api","chunkSizeBytes","use","getRawBodyLimitBytes","getMaxClientUploadBytesPerSecond","post","put"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"completeUpload.d.ts","sourceRoot":"","sources":["../../../../../src/uploads/api/file-uploads/handlers/completeUpload.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAA;AAKzC,OAAO,KAAK,OAAO,MAAM,UAAU,CAAA;AAGnC,OAAO,EACL,KAAK,WAAW,EASjB,MAAM,WAAW,CAAA;AAiDlB,eAAO,MAAM,cAAc,EAAE,UAAU,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,uBAAuB,EAAE,WAAW,
|
|
1
|
+
{"version":3,"file":"completeUpload.d.ts","sourceRoot":"","sources":["../../../../../src/uploads/api/file-uploads/handlers/completeUpload.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAA;AAKzC,OAAO,KAAK,OAAO,MAAM,UAAU,CAAA;AAGnC,OAAO,EACL,KAAK,WAAW,EASjB,MAAM,WAAW,CAAA;AAiDlB,eAAO,MAAM,cAAc,EAAE,UAAU,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,uBAAuB,EAAE,WAAW,CAwV1G,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"imageVariantsPostProcessor.d.ts","sourceRoot":"","sources":["../../../../src/uploads/api/file-uploads/imageVariantsPostProcessor.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { UploadFileProcessor, UploadFileProcessorContext } from './index';
|
|
2
|
+
export declare const convertHeifToWebpTaskName = "rb-upload-convert-heif-to-webp";
|
|
3
|
+
export type ConvertHeifToWebpTaskPayload = {
|
|
4
|
+
filename: string;
|
|
5
|
+
mimeType: string;
|
|
6
|
+
inputBase64: string;
|
|
7
|
+
};
|
|
8
|
+
export type ConvertHeifToWebpTaskResult = {
|
|
9
|
+
dataBase64: string;
|
|
10
|
+
filename: string;
|
|
11
|
+
mimeType: "image/webp";
|
|
12
|
+
metadata: {
|
|
13
|
+
width?: number;
|
|
14
|
+
height?: number;
|
|
15
|
+
sourceFilename: string;
|
|
16
|
+
sourceMimeType: string;
|
|
17
|
+
};
|
|
18
|
+
};
|
|
19
|
+
declare module "@rpcbase/worker" {
|
|
20
|
+
interface WorkerTasksMap {
|
|
21
|
+
"rb-upload-convert-heif-to-webp": ConvertHeifToWebpTaskPayload;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
export declare const hasHeifDeclaration: ({ filename, clientMimeType }: Pick<UploadFileProcessorContext, "filename" | "clientMimeType">) => boolean;
|
|
25
|
+
export declare const looksLikeHevcHeif: (sniff: Buffer) => boolean;
|
|
26
|
+
export declare const isHeifUpload: ({ filename, clientMimeType, sniff }: UploadFileProcessorContext) => boolean;
|
|
27
|
+
export declare const toWebpFilename: (filename: string) => string;
|
|
28
|
+
export declare const getSharpHeifSupportDiagnostic: () => string;
|
|
29
|
+
export declare const convertHeifToWebp: (input: Buffer, payload: Pick<ConvertHeifToWebpTaskPayload, "filename" | "mimeType">) => Promise<ConvertHeifToWebpTaskResult>;
|
|
30
|
+
export declare const convertHeifToWebpProcessor: UploadFileProcessor;
|
|
31
|
+
//# sourceMappingURL=convertHeifToWebp.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"convertHeifToWebp.d.ts","sourceRoot":"","sources":["../../../../../src/uploads/api/file-uploads/processors/convertHeifToWebp.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,mBAAmB,EAAE,0BAA0B,EAAE,MAAM,SAAS,CAAA;AAoB9E,eAAO,MAAM,yBAAyB,mCAAmC,CAAA;AAEzE,MAAM,MAAM,4BAA4B,GAAG;IACzC,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,MAAM,CAAA;IAChB,WAAW,EAAE,MAAM,CAAA;CACpB,CAAA;AAED,MAAM,MAAM,2BAA2B,GAAG;IACxC,UAAU,EAAE,MAAM,CAAA;IAClB,QAAQ,EAAE,MAAM,CAAA;IAChB,QAAQ,EAAE,YAAY,CAAA;IACtB,QAAQ,EAAE;QACR,KAAK,CAAC,EAAE,MAAM,CAAA;QACd,MAAM,CAAC,EAAE,MAAM,CAAA;QACf,cAAc,EAAE,MAAM,CAAA;QACtB,cAAc,EAAE,MAAM,CAAA;KACvB,CAAA;CACF,CAAA;AAED,OAAO,QAAQ,iBAAiB,CAAC;IAC/B,UAAU,cAAc;QACtB,gCAAgC,EAAE,4BAA4B,CAAA;KAC/D;CACF;AAyBD,eAAO,MAAM,kBAAkB,GAAI,8BAA8B,IAAI,CAAC,0BAA0B,EAAE,UAAU,GAAG,gBAAgB,CAAC,KAAG,OAC5C,CAAA;AAEvF,eAAO,MAAM,iBAAiB,GAAI,OAAO,MAAM,KAAG,OAAsE,CAAA;AAExH,eAAO,MAAM,YAAY,GAAI,qCAAqC,0BAA0B,KAAG,OACjB,CAAA;AAE9E,eAAO,MAAM,cAAc,GAAI,UAAU,MAAM,KAAG,MAKjD,CAAA;AAcD,eAAO,MAAM,6BAA6B,QAAO,MAQhD,CAAA;AAkBD,eAAO,MAAM,iBAAiB,GAC5B,OAAO,MAAM,EACb,SAAS,IAAI,CAAC,4BAA4B,EAAE,UAAU,GAAG,UAAU,CAAC,KACnE,OAAO,CAAC,2BAA2B,CAqBrC,CAAA;AAoDD,eAAO,MAAM,0BAA0B,EAAE,mBAqBxC,CAAA"}
|