silverbullet/common/spaces/disk_space_primitives.ts

172 lines
5.0 KiB
TypeScript
Raw Normal View History

2023-08-20 23:51:00 +08:00
import * as path from "https://deno.land/std@0.189.0/path/mod.ts";
import { readAll } from "https://deno.land/std@0.165.0/streams/conversion.ts";
import { SpacePrimitives } from "./space_primitives.ts";
import { mime } from "https://deno.land/x/mimetypes@v1.0.0/mod.ts";
import { walk } from "https://deno.land/std@0.198.0/fs/walk.ts";
2023-08-20 23:51:00 +08:00
import { FileMeta } from "$sb/types.ts";
2022-09-12 20:50:37 +08:00
function lookupContentType(path: string): string {
return mime.getType(path) || "application/octet-stream";
2022-09-12 20:50:37 +08:00
}
2023-01-14 19:04:51 +08:00
function normalizeForwardSlashPath(path: string) {
return path.replaceAll("\\", "/");
}
2023-01-13 22:41:29 +08:00
const excludedFiles = ["data.db", "data.db-journal", "sync.json"];
export class DiskSpacePrimitives implements SpacePrimitives {
rootPath: string;
2023-05-29 15:53:49 +08:00
constructor(rootPath: string) {
this.rootPath = Deno.realPathSync(rootPath);
}
2022-04-30 00:54:27 +08:00
safePath(p: string): string {
const realPath = path.resolve(p);
2022-04-30 00:54:27 +08:00
if (!realPath.startsWith(this.rootPath)) {
throw Error(`Path ${p} is not in the space`);
}
return realPath;
}
2022-09-12 20:50:37 +08:00
filenameToPath(pageName: string) {
return this.safePath(path.join(this.rootPath, pageName));
}
2022-09-12 20:50:37 +08:00
pathToFilename(fullPath: string): string {
return fullPath.substring(this.rootPath.length + 1);
}
2022-09-12 20:50:37 +08:00
async readFile(
name: string,
): Promise<{ data: Uint8Array; meta: FileMeta }> {
2022-09-12 20:50:37 +08:00
const localPath = this.filenameToPath(name);
try {
const s = await Deno.stat(localPath);
const contentType = lookupContentType(name);
const f = await Deno.open(localPath, { read: true });
const data = await readAll(f);
2024-01-28 21:13:37 +08:00
f.close();
return {
2022-09-12 20:50:37 +08:00
data,
meta: {
2022-09-12 20:50:37 +08:00
name: name,
2023-11-13 17:32:40 +08:00
created: s.birthtime?.getTime() || s.mtime?.getTime() || 0,
lastModified: s.mtime?.getTime() || 0,
2022-05-17 17:53:17 +08:00
perm: "rw",
2022-09-12 20:50:37 +08:00
size: s.size,
contentType: contentType,
},
};
2022-10-16 01:02:56 +08:00
} catch {
// console.error("Error while reading file", name, e);
throw Error("Not found");
}
}
2022-09-12 20:50:37 +08:00
async writeFile(
name: string,
data: Uint8Array,
_selfUpdate?: boolean,
meta?: FileMeta,
2022-09-12 20:50:37 +08:00
): Promise<FileMeta> {
2022-10-11 00:19:08 +08:00
const localPath = this.filenameToPath(name);
try {
2022-03-23 22:41:12 +08:00
// Ensure parent folder exists
await Deno.mkdir(path.dirname(localPath), { recursive: true });
2022-03-23 22:41:12 +08:00
const file = await Deno.open(localPath, {
write: true,
create: true,
truncate: true,
});
2022-03-23 22:41:12 +08:00
// Actually write the file
2024-01-28 21:37:54 +08:00
await file.write(data);
if (meta?.lastModified) {
// console.log("Seting mtime to", new Date(meta.lastModified));
2024-01-28 21:37:54 +08:00
await file.utime(new Date(), new Date(meta.lastModified));
}
file.close();
2022-09-12 20:50:37 +08:00
2022-03-23 22:41:12 +08:00
// Fetch new metadata
return this.getFileMeta(name);
} catch (e) {
2022-09-12 20:50:37 +08:00
console.error("Error while writing file", name, e);
throw Error(`Could not write ${name}`);
}
}
2022-09-12 20:50:37 +08:00
async getFileMeta(name: string): Promise<FileMeta> {
const localPath = this.filenameToPath(name);
try {
const s = await Deno.stat(localPath);
return {
2022-09-12 20:50:37 +08:00
name: name,
size: s.size,
contentType: lookupContentType(name),
2023-11-13 17:32:40 +08:00
created: s.birthtime?.getTime() || s.mtime?.getTime() || 0,
lastModified: s.mtime?.getTime() || 0,
2022-05-17 17:53:17 +08:00
perm: "rw",
};
2022-10-16 01:02:56 +08:00
} catch {
2022-06-28 20:14:15 +08:00
// console.error("Error while getting page meta", pageName, e);
2022-09-12 20:50:37 +08:00
throw Error(`Could not get meta for ${name}`);
}
}
2022-09-12 20:50:37 +08:00
async deleteFile(name: string): Promise<void> {
const localPath = this.filenameToPath(name);
await Deno.remove(localPath);
}
2022-09-12 20:50:37 +08:00
async fetchFileList(): Promise<FileMeta[]> {
const allFiles: FileMeta[] = [];
for await (
const file of walk(this.rootPath, {
includeDirs: false,
// Exclude hidden directories
2022-11-29 15:50:09 +08:00
skip: [
// Dynamically builds a regexp that matches hidden directories INSIDE the rootPath
// (but if the rootPath is hidden, it stil lists files inside of it, fixing #130)
new RegExp(`^${escapeRegExp(this.rootPath)}.*\\/\\..+$`),
],
})
) {
const fullPath = file.path;
try {
const s = await Deno.stat(fullPath);
// console.log(fullPath, s.isSymlink);
2023-01-13 22:41:29 +08:00
const name = fullPath.substring(this.rootPath.length + 1);
if (excludedFiles.includes(name)) {
continue;
}
allFiles.push({
2023-01-14 19:04:51 +08:00
name: normalizeForwardSlashPath(name),
2023-11-13 17:32:40 +08:00
created: s.birthtime?.getTime() || s.mtime?.getTime() || 0,
lastModified: s.mtime?.getTime() || 0,
contentType: mime.getType(fullPath) || "application/octet-stream",
size: s.size,
perm: "rw",
});
} catch (e: any) {
if (e instanceof Deno.errors.NotFound) {
// Ignore, temporariy file already deleted by the time we got here
} else {
console.error("Failed to stat", fullPath, e);
}
}
}
return allFiles;
}
}
2022-11-29 15:50:09 +08:00
function escapeRegExp(string: string) {
return string.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string
}