137 lines
4.0 KiB
TypeScript
137 lines
4.0 KiB
TypeScript
import { Hook, Manifest } from "../../lib/plugos/types.ts";
|
|
import { System } from "../../lib/plugos/system.ts";
|
|
import type { Context, Next } from "hono/mod.ts";
|
|
import { EndpointHookT } from "$lib/manifest.ts";
|
|
|
|
export type EndpointRequest = {
|
|
method: string;
|
|
path: string;
|
|
query: { [key: string]: string };
|
|
headers: { [key: string]: string };
|
|
body: any;
|
|
};
|
|
|
|
export type EndpointResponse = {
|
|
status: number;
|
|
headers?: { [key: string]: string };
|
|
body: any;
|
|
};
|
|
|
|
export class EndpointHook implements Hook<EndpointHookT> {
|
|
readonly prefix: string;
|
|
|
|
constructor(prefix: string) {
|
|
this.prefix = prefix;
|
|
}
|
|
|
|
public async handleRequest(
|
|
system: System<EndpointHookT>,
|
|
ctx: Context,
|
|
next: Next,
|
|
) {
|
|
const req = ctx.req;
|
|
const url = new URL(req.url);
|
|
const requestPath = url.pathname;
|
|
if (!requestPath.startsWith(this.prefix)) {
|
|
return next();
|
|
}
|
|
console.log("Endpoint request", requestPath);
|
|
// Iterate over all loaded plugins
|
|
for (const [plugName, plug] of system.loadedPlugs.entries()) {
|
|
const manifest = plug.manifest;
|
|
if (!manifest) {
|
|
continue;
|
|
}
|
|
const functions = manifest.functions;
|
|
// console.log("Checking plug", plugName);
|
|
const prefix = `${this.prefix}${plugName}`;
|
|
if (!requestPath.startsWith(prefix)) {
|
|
continue;
|
|
}
|
|
for (const [name, functionDef] of Object.entries(functions)) {
|
|
if (!functionDef.http) {
|
|
continue;
|
|
}
|
|
// console.log("Got config", functionDef);
|
|
const endpoints = Array.isArray(functionDef.http)
|
|
? functionDef.http
|
|
: [functionDef.http];
|
|
// console.log(endpoints);
|
|
for (const { path, method } of endpoints) {
|
|
const prefixedPath = `${prefix}${path}`;
|
|
if (
|
|
prefixedPath === requestPath &&
|
|
((method || "GET") === req.method || method === "ANY")
|
|
) {
|
|
try {
|
|
const response: EndpointResponse = await plug.invoke(name, [
|
|
{
|
|
path: url.pathname,
|
|
method: req.method,
|
|
body: await req.text(),
|
|
query: Object.fromEntries(
|
|
url.searchParams.entries(),
|
|
),
|
|
headers: req.header(),
|
|
} as EndpointRequest,
|
|
]);
|
|
if (response.headers) {
|
|
for (
|
|
const [key, value] of Object.entries(
|
|
response.headers,
|
|
)
|
|
) {
|
|
ctx.header(key, value);
|
|
}
|
|
}
|
|
ctx.status(response.status);
|
|
console.log("Going to return", response.body);
|
|
if (typeof response.body === "string") {
|
|
return ctx.text(response.body);
|
|
} else if (response.body instanceof Uint8Array) {
|
|
return ctx.body(response.body);
|
|
} else {
|
|
return ctx.json(response.body);
|
|
}
|
|
} catch (e: any) {
|
|
console.error("Error executing function", e);
|
|
return ctx.body(e.message, 500);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// console.log("Shouldn't get here");
|
|
await next();
|
|
}
|
|
|
|
apply(): void {
|
|
}
|
|
|
|
validateManifest(manifest: Manifest<EndpointHookT>): string[] {
|
|
const errors = [];
|
|
for (const functionDef of Object.values(manifest.functions)) {
|
|
if (!functionDef.http) {
|
|
continue;
|
|
}
|
|
const endpoints = Array.isArray(functionDef.http)
|
|
? functionDef.http
|
|
: [functionDef.http];
|
|
for (const { path, method } of endpoints) {
|
|
if (!path) {
|
|
errors.push("Path not defined for endpoint");
|
|
}
|
|
if (
|
|
method &&
|
|
["GET", "POST", "PUT", "DELETE", "ANY"].indexOf(method) === -1
|
|
) {
|
|
errors.push(
|
|
`Invalid method ${method} for end point with with ${path}`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
}
|