This takes part of the work being done in #96 that was reverted but still useful. Note Tailwind and Fresh weren't upgraded because there's no security vulnerability in either, and I have found the new versions to be worse in performance. Thos will likely stay at those fixed versions going forward.
247 lines
7.1 KiB
TypeScript
247 lines
7.1 KiB
TypeScript
import { Handler, RouteConfig } from 'fresh/server.ts';
|
|
import { join } from '@std/path';
|
|
import { parse, stringify } from '@libs/xml';
|
|
|
|
import { FreshContextState } from '/lib/types.ts';
|
|
import { AppConfig } from '/lib/config.ts';
|
|
import Locker from '/lib/interfaces/locker.ts';
|
|
import {
|
|
addDavPrefixToKeys,
|
|
buildPropFindResponse,
|
|
getProperDestinationPath,
|
|
getPropertyNames,
|
|
} from '/lib/utils/webdav.ts';
|
|
import { ensureUserPathIsValidAndSecurelyAccessible, FileModel } from '/lib/models/files.ts';
|
|
|
|
interface Data {}
|
|
|
|
export const config: RouteConfig = {
|
|
routeOverride: '/dav/:filePath*',
|
|
};
|
|
|
|
export const handler: Handler<Data, FreshContextState> = async (request, context) => {
|
|
if (!context.state.user) {
|
|
return new Response('Unauthorized', {
|
|
status: 401,
|
|
headers: { 'www-authenticate': 'Basic realm="bewCloud", charset="UTF-8"' },
|
|
});
|
|
}
|
|
|
|
let { filePath } = context.params;
|
|
|
|
if (!filePath) {
|
|
filePath = '/';
|
|
}
|
|
|
|
filePath = decodeURIComponent(filePath);
|
|
|
|
const userId = context.state.user.id;
|
|
|
|
const rootPath = join(await AppConfig.getFilesRootPath(), userId);
|
|
|
|
if (request.method === 'OPTIONS') {
|
|
const headers = new Headers({
|
|
DAV: '1, 2',
|
|
'Ms-Author-Via': 'DAV',
|
|
Allow: 'OPTIONS, DELETE, PROPFIND',
|
|
'Content-Length': '0',
|
|
Date: new Date().toUTCString(),
|
|
});
|
|
|
|
return new Response(null, { status: 200, headers });
|
|
}
|
|
|
|
if (request.method === 'GET') {
|
|
try {
|
|
const fileResult = await FileModel.get(userId, filePath);
|
|
|
|
if (!fileResult.success) {
|
|
return new Response('Not Found', { status: 404 });
|
|
}
|
|
|
|
return new Response(fileResult.contents! as BodyInit, {
|
|
status: 200,
|
|
headers: {
|
|
'cache-control': 'no-cache, no-store, must-revalidate',
|
|
'content-type': fileResult.contentType!,
|
|
'content-length': fileResult.byteSize!.toString(),
|
|
},
|
|
});
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
|
|
return new Response('Not Found', { status: 404 });
|
|
}
|
|
|
|
if (request.method === 'DELETE') {
|
|
try {
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, filePath);
|
|
|
|
await Deno.remove(join(rootPath, filePath));
|
|
|
|
return new Response(null, { status: 204 });
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
|
|
return new Response('Not Found', { status: 404 });
|
|
}
|
|
|
|
if (request.method === 'PUT') {
|
|
const contentLengthString = request.headers.get('content-length');
|
|
const contentLength = contentLengthString ? parseInt(contentLengthString, 10) : null;
|
|
const body = contentLength === 0 ? new Blob([new Uint8Array([0])]).stream() : request.clone().body;
|
|
|
|
try {
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, filePath);
|
|
|
|
const newFile = await Deno.open(join(rootPath, filePath), {
|
|
create: true,
|
|
write: true,
|
|
truncate: true,
|
|
});
|
|
|
|
await body?.pipeTo(newFile.writable);
|
|
|
|
return new Response('Created', { status: 201 });
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
|
|
return new Response('Not Found', { status: 404 });
|
|
}
|
|
|
|
if (request.method === 'COPY') {
|
|
const newFilePath = request.headers.get('destination');
|
|
if (newFilePath) {
|
|
try {
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, filePath);
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, getProperDestinationPath(newFilePath));
|
|
|
|
await Deno.copyFile(join(rootPath, filePath), join(rootPath, getProperDestinationPath(newFilePath)));
|
|
return new Response('Created', { status: 201 });
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
}
|
|
|
|
return new Response('Bad Request', { status: 400 });
|
|
}
|
|
|
|
if (request.method === 'MOVE') {
|
|
const newFilePath = request.headers.get('destination');
|
|
if (newFilePath) {
|
|
try {
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, filePath);
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, getProperDestinationPath(newFilePath));
|
|
|
|
await Deno.rename(join(rootPath, filePath), join(rootPath, getProperDestinationPath(newFilePath)));
|
|
return new Response('Created', { status: 201 });
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (request.method === 'MKCOL') {
|
|
try {
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, filePath);
|
|
await Deno.mkdir(join(rootPath, filePath), { recursive: true });
|
|
return new Response('Created', { status: 201 });
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
|
|
return new Response('Not Found', { status: 404 });
|
|
}
|
|
|
|
if (request.method === 'LOCK') {
|
|
const depthString = request.headers.get('depth');
|
|
const depth = depthString ? parseInt(depthString, 10) : null;
|
|
const xml = await request.clone().text();
|
|
const parsedXml = parse(xml) as Record<string, any>;
|
|
|
|
const lockToken = crypto.randomUUID();
|
|
const lock = new Locker(`dav:${lockToken}`);
|
|
|
|
await lock.acquire();
|
|
|
|
const responseXml: Record<string, any> = {
|
|
xml: {
|
|
'@version': '1.0',
|
|
'@encoding': 'UTF-8',
|
|
},
|
|
prop: {
|
|
'@xmlns:D': 'DAV:',
|
|
lockdiscovery: {
|
|
activelock: {
|
|
locktype: { write: null },
|
|
lockscope: { exclusive: null },
|
|
depth,
|
|
owner: {
|
|
href: parsedXml['D:lockinfo']?.['D:owner']?.['D:href'],
|
|
},
|
|
timeout: 'Second-600',
|
|
locktoken: { href: lockToken },
|
|
lockroot: { href: filePath },
|
|
},
|
|
},
|
|
},
|
|
};
|
|
|
|
const responseString = stringify(addDavPrefixToKeys(responseXml));
|
|
|
|
return new Response(responseString, {
|
|
status: 200,
|
|
headers: {
|
|
'Content-Type': 'application/xml; charset=utf-8',
|
|
'Lock-Token': `<${lockToken}>`,
|
|
'Content-Length': responseString.length.toString(),
|
|
Date: new Date().toUTCString(),
|
|
},
|
|
});
|
|
}
|
|
|
|
if (request.method === 'UNLOCK') {
|
|
const lockToken = request.headers.get('Lock-Token');
|
|
const lock = new Locker(`dav:${lockToken}`);
|
|
lock.release();
|
|
|
|
return new Response(null, {
|
|
status: 204,
|
|
headers: { Date: new Date().toUTCString() },
|
|
});
|
|
}
|
|
|
|
if (request.method === 'PROPFIND') {
|
|
const depthString = request.headers.get('depth');
|
|
const depth = depthString ? parseInt(depthString, 10) : null;
|
|
const xml = await request.clone().text();
|
|
|
|
const parsedXml = parse(xml);
|
|
|
|
const properties = getPropertyNames(parsedXml);
|
|
|
|
await ensureUserPathIsValidAndSecurelyAccessible(userId, filePath);
|
|
|
|
const responseXml = await buildPropFindResponse(properties, rootPath, filePath, depth);
|
|
|
|
return responseXml['D:multistatus']['D:response'].length === 0
|
|
? new Response('Not Found', {
|
|
status: 404,
|
|
headers: new Headers({ 'Content-Type': 'text/plain; charset=utf-8' }),
|
|
})
|
|
: new Response(stringify(responseXml), {
|
|
status: 207,
|
|
headers: new Headers({
|
|
'Content-Type': 'text/xml; charset=utf-8',
|
|
'Access-Control-Allow-Headers': '*',
|
|
'Access-Control-Allow-Methods': '*',
|
|
}),
|
|
});
|
|
}
|
|
|
|
return new Response(null, { status: 405 });
|
|
};
|