mirror of
https://github.com/shuaiplus/nodewarden.git
synced 2026-06-20 13:00:39 +00:00
bb3fe41330
- Added `processSendFileUpload` function to handle file uploads for sends. - Integrated JWT token creation and verification for secure file uploads. - Updated `handleCreateFileSendV2` and `handleGetSendFileUpload` to use new upload URL generation. - Refactored upload handling in `handleUploadSendFile` and `handlePublicUploadSendFile` to utilize the new upload process. - Introduced `uploadDirectEncryptedPayload` for handling direct uploads with progress tracking. - Enhanced API routes to support both POST and PUT methods for attachment uploads. - Added localization strings for upload progress messages. - Created utility functions for direct upload URL building and payload parsing.
105 lines
3.3 KiB
TypeScript
105 lines
3.3 KiB
TypeScript
import { LIMITS } from '../config/limits';
|
|
import { DEFAULT_DEV_SECRET, Env } from '../types';
|
|
import { errorResponse } from './response';
|
|
|
|
export interface DirectUploadPayload {
|
|
body: ReadableStream;
|
|
contentType: string;
|
|
size: number;
|
|
}
|
|
|
|
interface ParseDirectUploadOptions {
|
|
expectedSize?: number | null;
|
|
expectedFileName?: string | null;
|
|
maxFileSize: number;
|
|
tooLargeMessage: string;
|
|
missingBodyMessage?: string;
|
|
contentLengthRequiredMessage?: string;
|
|
sizeMismatchMessage?: string;
|
|
fileNameMismatchMessage?: string;
|
|
}
|
|
|
|
export function buildDirectUploadUrl(request: Request, path: string, token: string): string {
|
|
const version = '2023-11-03';
|
|
const expiresAt = '2099-12-31T23:59:59Z';
|
|
const origin = new URL(request.url).origin;
|
|
return `${origin}${path}?sv=${encodeURIComponent(version)}&se=${encodeURIComponent(expiresAt)}&token=${encodeURIComponent(token)}`;
|
|
}
|
|
|
|
export function getSafeJwtSecret(env: Env): string | null {
|
|
const secret = (env.JWT_SECRET || '').trim();
|
|
if (!secret || secret.length < LIMITS.auth.jwtSecretMinLength || secret === DEFAULT_DEV_SECRET) {
|
|
return null;
|
|
}
|
|
return secret;
|
|
}
|
|
|
|
function parseContentLength(request: Request): number | null {
|
|
const raw = request.headers.get('content-length');
|
|
if (!raw) return null;
|
|
const value = Number(raw);
|
|
if (!Number.isFinite(value) || value < 0) return null;
|
|
return Math.floor(value);
|
|
}
|
|
|
|
export async function parseDirectUploadPayload(
|
|
request: Request,
|
|
options: ParseDirectUploadOptions
|
|
): Promise<DirectUploadPayload | Response> {
|
|
const {
|
|
expectedSize = null,
|
|
expectedFileName = null,
|
|
maxFileSize,
|
|
tooLargeMessage,
|
|
missingBodyMessage = 'No file uploaded',
|
|
contentLengthRequiredMessage = 'Content-Length is required for direct uploads',
|
|
sizeMismatchMessage,
|
|
fileNameMismatchMessage,
|
|
} = options;
|
|
const contentType = request.headers.get('content-type') || '';
|
|
|
|
if (contentType.includes('multipart/form-data')) {
|
|
const formData = await request.formData();
|
|
const file = formData.get('data') as File | null;
|
|
if (!file) {
|
|
return errorResponse(missingBodyMessage, 400);
|
|
}
|
|
if (file.size > maxFileSize) {
|
|
return errorResponse(tooLargeMessage, 413);
|
|
}
|
|
if (expectedFileName && file.name !== expectedFileName) {
|
|
return errorResponse(fileNameMismatchMessage || 'File name does not match.', 400);
|
|
}
|
|
if (expectedSize !== null && expectedSize !== undefined && file.size !== expectedSize) {
|
|
return errorResponse(sizeMismatchMessage || 'File size does not match.', 400);
|
|
}
|
|
return {
|
|
body: file.stream(),
|
|
contentType: file.type || 'application/octet-stream',
|
|
size: file.size,
|
|
};
|
|
}
|
|
|
|
if (!request.body) {
|
|
return errorResponse(missingBodyMessage, 400);
|
|
}
|
|
|
|
const declaredSize = parseContentLength(request);
|
|
const uploadSize = declaredSize ?? (expectedSize && expectedSize > 0 ? expectedSize : null);
|
|
if (uploadSize === null) {
|
|
return errorResponse(contentLengthRequiredMessage, 400);
|
|
}
|
|
if (uploadSize > maxFileSize) {
|
|
return errorResponse(tooLargeMessage, 413);
|
|
}
|
|
if (expectedSize !== null && expectedSize !== undefined && uploadSize !== expectedSize) {
|
|
return errorResponse(sizeMismatchMessage || 'File size does not match.', 400);
|
|
}
|
|
|
|
return {
|
|
body: request.body,
|
|
contentType: contentType || 'application/octet-stream',
|
|
size: uploadSize,
|
|
};
|
|
}
|