silverbullet/plugs/federation/federation.ts

232 lines
6.0 KiB
TypeScript
Raw Normal View History

import "$sb/lib/native_fetch.ts";
2023-07-30 17:30:01 +08:00
import { federatedPathToUrl } from "$sb/lib/resolve.ts";
import { readFederationConfigs } from "./config.ts";
import { datastore } from "$sb/syscalls.ts";
2024-02-29 22:23:05 +08:00
import type { FileMeta } from "../../plug-api/types.ts";
import { wildcardPathToRegex } from "./util.ts";
async function responseToFileMeta(
r: Response,
name: string,
): Promise<FileMeta> {
2023-07-30 17:30:01 +08:00
const federationConfigs = await readFederationConfigs();
// Default permission is "ro" unless explicitly set otherwise
let perm: "ro" | "rw" = "ro";
const federationConfig = federationConfigs.find((config) =>
name.startsWith(config.uri)
);
if (federationConfig?.perm) {
perm = federationConfig.perm;
}
return {
name: name,
size: r.headers.get("Content-length")
? +r.headers.get("Content-length")!
: 0,
contentType: r.headers.get("Content-type")!,
2023-07-30 17:30:01 +08:00
perm,
created: +(r.headers.get("X-Created") || "0"),
lastModified: +(r.headers.get("X-Last-Modified") || "0"),
};
}
const fileListingPrefixCacheKey = `federationListCache`;
const listingCacheTimeout = 1000 * 5;
2023-08-01 03:06:15 +08:00
const listingFetchTimeout = 2000;
2023-07-30 17:30:01 +08:00
type FileListingCacheEntry = {
items: FileMeta[];
lastUpdated: number;
};
export async function listFiles(): Promise<FileMeta[]> {
let fileMetas: FileMeta[] = [];
// Fetch them all in parallel
2023-07-30 17:30:01 +08:00
try {
await Promise.all((await readFederationConfigs()).map(async (config) => {
const items = await listFilesCached(config.uri);
fileMetas = fileMetas.concat(items);
2023-07-30 17:30:01 +08:00
}));
// console.log("All of em: ", fileMetas);
return fileMetas;
} catch (e: any) {
console.error("Error listing federation files", e);
return [];
}
}
export async function listFilesCached(
uri: string,
supportWildcards = false,
): Promise<FileMeta[]> {
const uriParts = uri.split("/");
const rootUri = uriParts[0];
const prefix = uriParts.slice(1).join("/");
console.log(
"Fetching listing from federated",
rootUri,
"with prefix",
prefix,
);
const cachedListing = await datastore.get(
[fileListingPrefixCacheKey, rootUri],
) as FileListingCacheEntry;
let items: FileMeta[] = [];
if (
cachedListing &&
cachedListing.lastUpdated > Date.now() - listingCacheTimeout
) {
console.info("Using cached listing", cachedListing.items.length);
items = cachedListing.items;
} else {
const indexUrl = `${federatedPathToUrl(rootUri)}/index.json`;
try {
const fetchController = new AbortController();
const timeout = setTimeout(
() => fetchController.abort(),
listingFetchTimeout,
);
const r = await nativeFetch(indexUrl, {
method: "GET",
headers: {
"X-Sync-Mode": "true",
"Cache-Control": "no-cache",
},
signal: fetchController.signal,
});
clearTimeout(timeout);
if (r.status !== 200) {
throw new Error(`Got status ${r.status}`);
}
const jsonResult = await r.json();
// Transform them a little bit
items = jsonResult.map((meta: FileMeta) => ({
...meta,
perm: "ro",
name: `${rootUri}/${meta.name}`,
}));
// Cache the entire listing
await datastore.set([fileListingPrefixCacheKey, rootUri], {
items,
lastUpdated: Date.now(),
} as FileListingCacheEntry);
} catch (e: any) {
console.error("Failed to process", indexUrl, e);
if (cachedListing) {
console.info("Using cached listing");
return cachedListing.items;
}
}
}
// And then filter based on prefix before returning
if (!supportWildcards) {
return items.filter((meta: FileMeta) => meta.name.startsWith(uri));
} else {
const prefixRegex = wildcardPathToRegex(uri);
return items.filter((meta) => prefixRegex.test(meta.name));
}
}
export async function readFile(
name: string,
): Promise<{ data: Uint8Array; meta: FileMeta }> {
2023-07-30 17:30:01 +08:00
const url = federatedPathToUrl(name);
2024-01-02 18:32:57 +08:00
console.log("Fetching federated file", url);
const r = await nativeFetch(url, {
method: "GET",
headers: {
Accept: "application/octet-stream",
},
});
2023-07-30 17:30:01 +08:00
if (r.status === 503) {
throw new Error("Offline");
}
const fileMeta = await responseToFileMeta(r, name);
if (r.status === 404) {
throw Error("Not found");
}
const data = await r.arrayBuffer();
if (!r.ok) {
return errorResult(name, `**Error**: Could not load`);
}
return {
data: new Uint8Array(data),
meta: fileMeta,
};
}
function errorResult(
name: string,
error: string,
): { data: Uint8Array; meta: FileMeta } {
return {
data: new TextEncoder().encode(error),
meta: {
name,
contentType: "text/markdown",
created: 0,
lastModified: 0,
size: 0,
perm: "ro",
},
};
}
export async function writeFile(
name: string,
data: Uint8Array,
): Promise<FileMeta> {
2023-07-30 05:41:37 +08:00
throw new Error("Writing federation file, not yet supported");
// const url = resolveFederated(name);
// console.log("Writing federation file", url);
2023-07-30 05:41:37 +08:00
// const r = await nativeFetch(url, {
// method: "PUT",
// body: data,
// });
// const fileMeta = await responseToFileMeta(r, name);
// if (!r.ok) {
// throw new Error("Could not write file");
// }
2023-07-30 05:41:37 +08:00
// return fileMeta;
}
export async function deleteFile(
name: string,
): Promise<void> {
2023-07-30 05:41:37 +08:00
throw new Error("Writing federation file, not yet supported");
// console.log("Deleting federation file", name);
// const url = resolveFederated(name);
// const r = await nativeFetch(url, { method: "DELETE" });
// if (!r.ok) {
// throw Error("Failed to delete file");
// }
}
export async function getFileMeta(name: string): Promise<FileMeta> {
2023-07-30 17:30:01 +08:00
const url = federatedPathToUrl(name);
2023-08-16 02:15:27 +08:00
console.info("Fetching federation file meta", url);
const r = await nativeFetch(url, {
method: "GET",
headers: {
2023-12-22 01:37:50 +08:00
"X-Sync-Mode": "true",
2023-08-16 02:15:27 +08:00
"X-Get-Meta": "true",
},
});
2023-07-30 17:30:01 +08:00
if (r.status === 503) {
throw new Error("Offline");
}
const fileMeta = await responseToFileMeta(r, name);
if (!r.ok) {
throw new Error("Not found");
}
return fileMeta;
}