silverbullet/common/spaces/http_space_primitives.ts

140 lines
3.6 KiB
TypeScript
Raw Normal View History

2022-10-16 01:02:56 +08:00
import { FileMeta } from "../types.ts";
import { SpacePrimitives } from "./space_primitives.ts";
import { flushCachesAndUnregisterServiceWorker } from "../sw_util.ts";
2022-04-07 21:21:30 +08:00
export class HttpSpacePrimitives implements SpacePrimitives {
2023-01-13 22:41:29 +08:00
constructor(
readonly url: string,
readonly expectedSpacePath?: string,
readonly syncMode = false,
2023-01-13 22:41:29 +08:00
) {
2022-04-30 00:54:27 +08:00
}
public async authenticatedFetch(
2022-04-30 00:54:27 +08:00
url: string,
options: RequestInit,
2022-04-30 00:54:27 +08:00
): Promise<Response> {
if (!options.headers) {
options.headers = {};
}
if (this.syncMode) {
options.headers = { ...options.headers, ...{ "X-Sync-Mode": "true" } };
2023-01-13 22:41:29 +08:00
}
const result = await fetch(url, { ...options });
if (
this.getRealStatus(result) === 401
) {
// Invalid credentials, reloading the browser should trigger authentication
console.log("Going to redirect after", url);
location.href = "/.auth?refer=" + location.pathname;
throw new Error("Invalid credentials");
2022-04-30 00:54:27 +08:00
}
return result;
}
getRealStatus(r: Response) {
if (r.headers.get("X-Status")) {
return +r.headers.get("X-Status")!;
}
return r.status;
}
2023-01-13 22:41:29 +08:00
async fetchFileList(): Promise<FileMeta[]> {
const resp = await this.authenticatedFetch(this.url, {
method: "GET",
});
if (
this.getRealStatus(resp) === 200 &&
this.expectedSpacePath &&
resp.headers.get("X-Space-Path") !== this.expectedSpacePath
) {
await flushCachesAndUnregisterServiceWorker();
alert("Space folder path different on server, reloading the page");
location.reload();
}
return resp.json();
}
2022-09-12 20:50:37 +08:00
async readFile(
name: string,
): Promise<{ data: Uint8Array; meta: FileMeta }> {
2023-01-13 22:41:29 +08:00
const res = await this.authenticatedFetch(
`${this.url}/${encodeURI(name)}`,
2023-01-13 22:41:29 +08:00
{
method: "GET",
},
);
if (this.getRealStatus(res) === 404) {
throw new Error(`Not found`);
2022-09-12 20:50:37 +08:00
}
return {
data: new Uint8Array(await res.arrayBuffer()),
2022-09-12 20:50:37 +08:00
meta: this.responseToMeta(name, res),
};
}
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> {
2023-01-13 22:41:29 +08:00
const headers: Record<string, string> = {
"Content-Type": "application/octet-stream",
};
if (meta) {
headers["X-Last-Modified"] = "" + meta.lastModified;
headers["X-Perm"] = "" + meta.perm;
2023-01-13 22:41:29 +08:00
}
const res = await this.authenticatedFetch(
`${this.url}/${encodeURI(name)}`,
2023-01-13 22:41:29 +08:00
{
method: "PUT",
headers,
body: data,
2022-09-12 20:50:37 +08:00
},
2023-01-13 22:41:29 +08:00
);
2022-09-12 20:50:37 +08:00
const newMeta = this.responseToMeta(name, res);
return newMeta;
}
2022-09-12 20:50:37 +08:00
async deleteFile(name: string): Promise<void> {
2023-01-13 22:41:29 +08:00
const req = await this.authenticatedFetch(
`${this.url}/${encodeURI(name)}`,
2023-01-13 22:41:29 +08:00
{
method: "DELETE",
},
);
if (this.getRealStatus(req) !== 200) {
2022-09-12 20:50:37 +08:00
throw Error(`Failed to delete file: ${req.statusText}`);
}
}
2022-09-12 20:50:37 +08:00
async getFileMeta(name: string): Promise<FileMeta> {
2023-01-13 22:41:29 +08:00
const res = await this.authenticatedFetch(
`${this.url}/${encodeURI(name)}`,
2023-01-13 22:41:29 +08:00
{
method: "OPTIONS",
},
);
if (this.getRealStatus(res) === 404) {
throw new Error(`Not found`);
2022-09-12 20:50:37 +08:00
}
return this.responseToMeta(name, res);
}
private responseToMeta(name: string, res: Response): FileMeta {
return {
name,
size: +res.headers.get("X-Content-Length")!,
2022-09-12 20:50:37 +08:00
contentType: res.headers.get("Content-type")!,
lastModified: +(res.headers.get("X-Last-Modified") || "0"),
2022-09-12 20:50:37 +08:00
perm: (res.headers.get("X-Permission") as "rw" | "ro") || "rw",
};
}
}