mirror of
https://github.com/bewcloud/bewcloud.git
synced 2026-03-11 08:54:49 +00:00
Implement max file upload config options and simplify request method handling
Fixes #154 Related to #155
This commit is contained in:
parent
69a916d709
commit
f50423028b
10 changed files with 36 additions and 151 deletions
|
|
@ -20,9 +20,11 @@ const config: PartialDeep<Config> = {
|
|||
// rootPath: 'data-files',
|
||||
// allowPublicSharing: false, // If true, public file sharing will be allowed (still requires a user to enable sharing for a given file or directory)
|
||||
// allowDirectoryDownloads: false, // If true, directories can be downloaded as zip files
|
||||
// maxUploadSizeInMegabytes: 100, // The maximum upload size in megabytes. Overrides the core.maxRequestSizeInMegabytes setting on /dav and /api/files/upload endpoints.
|
||||
// },
|
||||
// core: {
|
||||
// enabledApps: ['dashboard', 'files', 'news', 'notes', 'photos', 'expenses', 'contacts', 'calendar'], // The apps to show, in order of appearance in the header. The first app will be the default one shown after logging in. At least one is required.
|
||||
// maxRequestSizeInMegabytes: 12, // The maximum request size in megabytes.
|
||||
// },
|
||||
// visuals: {
|
||||
// title: 'My own cloud',
|
||||
|
|
|
|||
|
|
@ -1,6 +1,6 @@
|
|||
services:
|
||||
website:
|
||||
image: ghcr.io/bewcloud/bewcloud:v4.1.0
|
||||
image: ghcr.io/bewcloud/bewcloud:v4.1.1
|
||||
# NOTE: uncomment below (and comment above) only if you pulled the repo and want to build the image locally
|
||||
# build:
|
||||
# context: .
|
||||
|
|
|
|||
|
|
@ -25,9 +25,11 @@ export class AppConfig {
|
|||
rootPath: 'data-files',
|
||||
allowPublicSharing: false,
|
||||
allowDirectoryDownloads: false,
|
||||
maxUploadSizeInMegabytes: 100,
|
||||
},
|
||||
core: {
|
||||
enabledApps: ['dashboard', 'files', 'news', 'notes', 'photos', 'expenses', 'contacts', 'calendar'],
|
||||
maxRequestSizeInMegabytes: 12,
|
||||
},
|
||||
visuals: {
|
||||
title: '',
|
||||
|
|
|
|||
30
lib/page.ts
30
lib/page.ts
|
|
@ -17,15 +17,7 @@ export interface Page {
|
|||
patch?: RequestHandler;
|
||||
delete?: RequestHandler;
|
||||
options?: RequestHandler;
|
||||
copy?: RequestHandler;
|
||||
move?: RequestHandler;
|
||||
mkcol?: RequestHandler;
|
||||
mkcalendar?: RequestHandler;
|
||||
lock?: RequestHandler;
|
||||
unlock?: RequestHandler;
|
||||
propfind?: RequestHandler;
|
||||
proppatch?: RequestHandler;
|
||||
report?: RequestHandler;
|
||||
catchAll?: RequestHandler;
|
||||
}
|
||||
|
||||
type AccessMode = 'public' | 'user';
|
||||
|
|
@ -61,15 +53,7 @@ export default function page(
|
|||
patch,
|
||||
delete: deleteAction,
|
||||
options,
|
||||
copy,
|
||||
move,
|
||||
mkcol,
|
||||
mkcalendar,
|
||||
lock,
|
||||
unlock,
|
||||
propfind,
|
||||
proppatch,
|
||||
report,
|
||||
catchAll,
|
||||
accessMode,
|
||||
}: Params,
|
||||
): Page {
|
||||
|
|
@ -80,14 +64,6 @@ export default function page(
|
|||
patch: patch ? permissioned(patch, accessMode) : undefined,
|
||||
delete: deleteAction ? permissioned(deleteAction, accessMode) : undefined,
|
||||
options: options ? permissioned(options, accessMode) : undefined,
|
||||
copy: copy ? permissioned(copy, accessMode) : undefined,
|
||||
move: move ? permissioned(move, accessMode) : undefined,
|
||||
mkcol: mkcol ? permissioned(mkcol, accessMode) : undefined,
|
||||
mkcalendar: mkcalendar ? permissioned(mkcalendar, accessMode) : undefined,
|
||||
lock: lock ? permissioned(lock, accessMode) : undefined,
|
||||
unlock: unlock ? permissioned(unlock, accessMode) : undefined,
|
||||
propfind: propfind ? permissioned(propfind, accessMode) : undefined,
|
||||
proppatch: proppatch ? permissioned(proppatch, accessMode) : undefined,
|
||||
report: report ? permissioned(report, accessMode) : undefined,
|
||||
catchAll: catchAll ? permissioned(catchAll, accessMode) : undefined,
|
||||
};
|
||||
}
|
||||
|
|
|
|||
|
|
@ -175,10 +175,14 @@ export interface Config {
|
|||
allowPublicSharing: boolean;
|
||||
/** If true, directories can be downloaded as zip files */
|
||||
allowDirectoryDownloads: boolean;
|
||||
/** The maximum upload size in megabytes. Overrides the core.maxRequestSizeInMegabytes setting on /dav and /api/files/upload endpoints. */
|
||||
maxUploadSizeInMegabytes: number;
|
||||
};
|
||||
core: {
|
||||
/** The apps to show, in order of appearance in the header. The first app will be the default one shown after logging in. At least one is required. */
|
||||
enabledApps: OptionalApp[];
|
||||
/** The maximum request size in megabytes. */
|
||||
maxRequestSizeInMegabytes: number;
|
||||
};
|
||||
visuals: {
|
||||
/** An override title of the application. Empty shows the default title. */
|
||||
|
|
|
|||
13
main.ts
13
main.ts
|
|
@ -1,9 +1,13 @@
|
|||
import routes, { Route } from './routes.ts';
|
||||
import { startCrons } from './crons/index.ts';
|
||||
import { Page } from './lib/page.ts';
|
||||
import { AppConfig } from './lib/config.ts';
|
||||
|
||||
const MAX_REQUEST_SIZE_IN_MEGABYTES = 12;
|
||||
const config = await AppConfig.getConfig();
|
||||
const MAX_REQUEST_SIZE_IN_MEGABYTES = config.core.maxRequestSizeInMegabytes;
|
||||
const MAX_UPLOAD_SIZE_IN_MEGABYTES = config.files.maxUploadSizeInMegabytes;
|
||||
const MAX_REQUEST_SIZE_IN_BYTES = MAX_REQUEST_SIZE_IN_MEGABYTES * 1024 * 1024;
|
||||
const MAX_UPLOAD_SIZE_IN_BYTES = MAX_UPLOAD_SIZE_IN_MEGABYTES * 1024 * 1024;
|
||||
|
||||
function applyCorsHeadersToResponse(origin: string, response: Response) {
|
||||
const headers = response.headers;
|
||||
|
|
@ -53,12 +57,15 @@ function handleLogging(request: Request, response: Response) {
|
|||
|
||||
async function handler(request: Request) {
|
||||
const contentLength = request.headers.get('content-length');
|
||||
const path = new URL(request.url).pathname;
|
||||
|
||||
if (contentLength && parseInt(contentLength, 10) > MAX_REQUEST_SIZE_IN_BYTES) {
|
||||
const isUploadRequest = path.startsWith('/api/files/upload') || path.startsWith('/dav');
|
||||
const maxSizeInBytes = isUploadRequest ? MAX_UPLOAD_SIZE_IN_BYTES : MAX_REQUEST_SIZE_IN_BYTES;
|
||||
|
||||
if (contentLength && parseInt(contentLength, 10) > maxSizeInBytes) {
|
||||
return new Response('Payload too large', { status: 413 });
|
||||
}
|
||||
|
||||
const path = new URL(request.url).pathname;
|
||||
const origin = request.headers.get('Origin') || '*';
|
||||
|
||||
// CORS headers for non-DAV routes
|
||||
|
|
|
|||
|
|
@ -75,14 +75,6 @@ export default page({
|
|||
put: get,
|
||||
delete: get,
|
||||
options: get,
|
||||
copy: get,
|
||||
move: get,
|
||||
mkcol: get,
|
||||
mkcalendar: get,
|
||||
lock: get,
|
||||
unlock: get,
|
||||
propfind: get,
|
||||
proppatch: get,
|
||||
report: get,
|
||||
catchAll: get,
|
||||
accessMode: 'public',
|
||||
});
|
||||
|
|
|
|||
|
|
@ -75,13 +75,6 @@ export default page({
|
|||
put: get,
|
||||
delete: get,
|
||||
options: get,
|
||||
copy: get,
|
||||
move: get,
|
||||
mkcol: get,
|
||||
lock: get,
|
||||
unlock: get,
|
||||
propfind: get,
|
||||
proppatch: get,
|
||||
report: get,
|
||||
catchAll: get,
|
||||
accessMode: 'public',
|
||||
});
|
||||
|
|
|
|||
|
|
@ -257,12 +257,6 @@ export default page({
|
|||
delete: handler,
|
||||
put: handler,
|
||||
options: handler,
|
||||
copy: handler,
|
||||
move: handler,
|
||||
mkcol: handler,
|
||||
lock: handler,
|
||||
unlock: handler,
|
||||
propfind: handler,
|
||||
report: handler,
|
||||
catchAll: handler,
|
||||
accessMode: 'public',
|
||||
});
|
||||
|
|
|
|||
107
routes.ts
107
routes.ts
|
|
@ -33,14 +33,7 @@ function createPageRouteHandler(id: string, pathname: string) {
|
|||
patch,
|
||||
delete: deleteAction,
|
||||
options,
|
||||
copy,
|
||||
move,
|
||||
mkcol,
|
||||
lock,
|
||||
unlock,
|
||||
propfind,
|
||||
proppatch,
|
||||
report,
|
||||
catchAll,
|
||||
} = page;
|
||||
|
||||
const { user, session, tokenData } = (await getDataFromRequest(request)) || {};
|
||||
|
|
@ -148,95 +141,17 @@ function createPageRouteHandler(id: string, pathname: string) {
|
|||
});
|
||||
}
|
||||
break;
|
||||
case 'COPY':
|
||||
if (copy) {
|
||||
return await copy({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'MOVE':
|
||||
if (move) {
|
||||
return await move({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'MKCOL':
|
||||
if (mkcol) {
|
||||
return await mkcol({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'LOCK':
|
||||
if (lock) {
|
||||
return await lock({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'UNLOCK':
|
||||
if (unlock) {
|
||||
return await unlock({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'PROPFIND':
|
||||
if (propfind) {
|
||||
return await propfind({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'PROPPATCH':
|
||||
if (proppatch) {
|
||||
return await proppatch({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case 'REPORT':
|
||||
if (report) {
|
||||
return await report({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
break;
|
||||
default:
|
||||
if (catchAll) {
|
||||
return await catchAll({
|
||||
request,
|
||||
match,
|
||||
user,
|
||||
session: { userSession: session, tokenData },
|
||||
isRunningLocally,
|
||||
});
|
||||
}
|
||||
|
||||
return new Response('Not Implemented', { status: 501 });
|
||||
}
|
||||
|
||||
|
|
|
|||
Loading…
Reference in a new issue