import { serve, Server, ServerRequest, decode } from "../deps.ts";
export class Fastro { listen = async ( options?: ListenOptions, callback?: (error: Error | undefined, address: string | undefined) => void, ): Promise<void> => { try { let opt = options ? options : { port: 8080 }; this.#server = serve(opt); this.loadPlugin(this); if (!callback) console.info(opt); else callback(undefined, opt as any); for await (const req of this.#server) { await this.requestHandler(req); } } catch (error) { const errStr = "SERVER_LISTEN_ERROR"; if (callback) callback(FastroError(errStr, error), undefined); else throw FastroError(errStr, error); } };
private loadPlugin(fastro: Fastro) { let afterLoadRouter: Router[]; let afterLoadPlugin: Instance[]; const loop = (plugins: Instance[]) => { plugins.forEach((p) => { afterLoadRouter = [...this.#router]; afterLoadPlugin = [...this.#plugins]; p.plugin(fastro, () => { const list = this.#plugins.filter((plugin) => !afterLoadPlugin.includes(plugin) ); if (list.length > 0) loop(list); if (p.prefix) { this.#router .filter((r) => !afterLoadRouter.includes(r)) .forEach((x) => { const url = x.url === "/" ? "" : x.url; x.url = `/${p.prefix}${url}`; }); } }); }); }; loop(this.#plugins); this.#router = [...new Set(this.#router)]; return this; }
register(plugin: Plugin): Fastro; register(prefix: string, plugin: Plugin): Fastro; register(prefixOrPlugin: string | Plugin, plugin?: Plugin) { if (typeof prefixOrPlugin !== "string") { this.#plugins.push({ plugin: prefixOrPlugin }); } if (typeof prefixOrPlugin === "string" && plugin) { this.#plugins.push({ prefix: prefixOrPlugin, plugin }); } return this; }
route(router: Router) { this.#router.push(router); return this; }
all(url: string, handler: Handler) { this.route({ method: "GET", url, handler }); this.route({ method: "POST", url, handler }); this.route({ method: "HEAD", url, handler }); this.route({ method: "PATCH", url, handler }); this.route({ method: "OPTIONS", url, handler }); this.route({ method: "PUT", url, handler }); this.route({ method: "DELETE", url, handler }); }
get(url: string, handler: Handler) { return this.route({ method: "GET", url, handler }); }
post(url: string, handler: Handler) { return this.route({ method: "POST", url, handler }); }
head(url: string, handler: Handler) { return this.route({ method: "HEAD", url, handler }); }
patch(url: string, handler: Handler) { return this.route({ method: "PATCH", url, handler }); }
put(url: string, handler: Handler) { return this.route({ method: "PUT", url, handler }); }
options(url: string, handler: Handler) { return this.route({ method: "OPTIONS", url, handler }); }
delete(url: string, handler: Handler) { return this.route({ method: "DELETE", url, handler }); }
async close(): Promise<void> { if (this.#server) { this.#server.close(); } }
decorate(instance: { (instance: Fastro): void }) { instance(this); return this; }
decorateRequest(request: { (req: Request): void }) { request(this.#request); return this; }
use(handler: Handler): Fastro; use(url: string, handler: Handler): Fastro; use(handlerOrUrl: string | Handler, handler?: Handler) { if (typeof handlerOrUrl !== "string") { this.#middlewares.push({ url: "/", handler: handlerOrUrl }); } if (handler && (typeof handlerOrUrl === "string")) { this.#middlewares.push({ url: handlerOrUrl, handler }); } return this; }
private getFunctionParameter(incoming: string) { let incomingSplit = incoming.substr(1, incoming.length).split("/"); const params: string[] = []; incomingSplit.splice(0, 2); incomingSplit .map((path, idx) => { return { path, idx }; }) .forEach((value) => { params.push(incomingSplit[value.idx]); });
return params; }
function(url: string, handler: Handler) { const fn = { url, handler }; this.#functions.push(fn); return this; }
private functionHandler(req: Request) { const [func] = this.#functions.filter(( fn, ) => (fn.url && this.checkFunctionUrl(req.url, fn.url))); if (!func) return this.forward(req); if (func.url) { req.parameter = this.getParameter(req.url, func.url); req.functionParameter = this.getFunctionParameter( req.url, ); } func.handler(req, () => { this.forward(req); }); }
forward(req: Request) { if (this.#middlewares.length > 0) return this.middlewareHandler(req); this.routeHandler(req); }
private middlewareHandler(req: Request) { try { const [middleware] = this.#middlewares.filter((mid) => { return (mid.url && this.checkUrl(req.url, mid.url)); }); if (!middleware) return this.routeHandler(req); if (middleware.url) { req.parameter = this.getParameter(req.url, middleware.url); } middleware.handler(req, () => this.routeHandler(req, false)); } catch (error) { throw FastroError("SERVER_MIDDLEWARE_HANDLER_ERROR", error); } }
private requestHandler = async (req: ServerRequest) => { try { const request = req as Request; request.payload = req.body ? decode(await Deno.readAll(req.body)) : undefined; request.send = (payload, status, headers): boolean => { return this.send(payload, status, headers, req); }; const mutatedRequest = Object.assign(this.#request, request); if (this.#functions.length > 0) { return this.functionHandler(mutatedRequest); } this.forward(mutatedRequest); } catch (error) { throw FastroError("SERVER_REQUEST_HANDLER_ERROR", error); } };
private routeHandler = async ( req: Request, mutate?: boolean, ) => { try { if (mutate) return; const [route] = this.#router.filter((value) => { return this.checkUrl(req.url, value.url) && (req.method == value.method); }); if (!route) { return req.respond({ body: `${req.url} not found`, status: 404 }); } req.parameter = this.getParameter(req.url, route.url); return route.handler(req, () => {}); } catch (error) { throw FastroError("SERVER_ROUTE_HANDLER_ERROR", error); } };
private send<T>( payload: string | T, status: number | undefined = 200, headers: Headers | undefined = new Headers(), req: ServerRequest, ) { try { let body: any; headers.set("X-Powered-By", "fastro"); if (typeof payload === "string") body = payload; else body = JSON.stringify(payload); req.respond({ status, headers, body }); return true; } catch (error) { throw FastroError("SERVER_SEND_ERROR", error); } }
private checkFunctionUrl(incoming: string, registered: string) { try { if (!registered.includes(":")) return incoming.includes(registered); const incomingSplit = incoming.substr(1, incoming.length).split("/"); const regSplit = registered.substr(1, registered.length).split("/"); const [firstIncome, secondIncome] = incomingSplit; const [firstReg, secondReg] = regSplit; if (firstReg.includes(":") && secondReg.includes(":")) return true; if (registered.includes(":")) { if (firstIncome.includes(firstReg)) return true; if (secondIncome && secondIncome.includes(secondReg)) return true; } } catch (error) { throw FastroError("CHECK_FN_URL_ERROR", error); } }
private checkUrl(incoming: string, registered: string): boolean { try { if (!registered.includes(":")) return incoming === registered; const incomingSplit = incoming.substr(1, incoming.length).split("/"); const regsSplit = registered.substr(1, registered.length).split("/"); const paths = regsSplit .filter((r) => !r.includes(":")) .filter((p) => { const str = `/${p}`; return incoming.includes(str); }); return paths.length > 0 && (incomingSplit.length === regsSplit.length); } catch (error) { throw FastroError("CHECK_URL_ERROR", error); } }
private getParameter(incoming: string, registered: string) { try { const incomingSplit = incoming.substr(1, incoming.length).split("/"); const registeredSplit = registered.substr(1, registered.length).split( "/", ); const param: Parameter = {}; registeredSplit .map((path, idx) => { return { path, idx }; }) .filter((value) => value.path.startsWith(":")) .map((value) => { const name = value.path.substr(1, value.path.length); param[name] = incomingSplit[value.idx]; }); return param; } catch (error) { throw FastroError("GET_URL_PARAMETER_ERROR", error); } }
[key: string]: any #server!: Server; #router: Router[] = []; #middlewares: Middleware[] = []; #functions: Middleware[] = []; #plugins: Instance[] = []; #request = new Request();}
export class Request extends ServerRequest { parameter!: Parameter;
functionParameter!: string[];
payload!: string | undefined; send!: { <T>(payload: string | T, status?: number, headers?: Headers): boolean; }; [key: string]: any}interface Router { method: string; url: string; handler(req: Request, callback: Function): any; [key: string]: any;}interface ListenOptions { port: number; hostname?: string;}interface Middleware { url?: string; handler(req: Request, callback: Function): any;}interface Plugin { (fastro: Fastro, callback: Function): any;}interface Instance { plugin: Plugin; prefix?: string;}interface Handler { (req: Request, callback: Function): any;}interface Parameter { [key: string]: string;}function FastroError(title: string, error: Error) { error.name = title; return error;}