Add url to server response

This commit is contained in:
Benjamin Toby 2026-03-19 06:33:47 +01:00
parent f722d4ae47
commit 7804a34951
6 changed files with 126 additions and 31 deletions

View File

@ -6,6 +6,7 @@ import handleRoutes from "./handle-routes";
import isDevelopment from "../../utils/is-development"; import isDevelopment from "../../utils/is-development";
import grabConstants from "../../utils/grab-constants"; import grabConstants from "../../utils/grab-constants";
import { AppData } from "../../data/app-data"; import { AppData } from "../../data/app-data";
import { existsSync } from "fs";
export default async function (params) { export default async function (params) {
const port = grabAppPort(); const port = grabAppPort();
const { PUBLIC_DIR } = grabDirNames(); const { PUBLIC_DIR } = grabDirNames();
@ -61,18 +62,43 @@ export default async function (params) {
return await handleRoutes({ req, server }); return await handleRoutes({ req, server });
} }
if (url.pathname.startsWith("/public/")) { if (url.pathname.startsWith("/public/")) {
const file = Bun.file(path.join(PUBLIC_DIR, url.pathname.replace(/^\/public/, ""))); try {
let res_opts = {}; const file_path = path.join(PUBLIC_DIR, url.pathname.replace(/^\/public/, ""));
if (!is_dev && url.pathname.match(/__bunext/)) { if (!existsSync(file_path)) {
res_opts.headers = { return new Response(`Public File Doesn't Exist`, {
"Cache-Control": `public, max-age=${AppData["BunextStaticFilesCacheExpiry"]}, must-revalidate`, status: 404,
}; });
}
const file = Bun.file(file_path);
let res_opts = {};
if (!is_dev && url.pathname.match(/__bunext/)) {
res_opts.headers = {
"Cache-Control": `public, max-age=${AppData["BunextStaticFilesCacheExpiry"]}, must-revalidate`,
};
}
return new Response(file, res_opts);
}
catch (error) {
return new Response(`Public File Not Found`, {
status: 404,
});
} }
return new Response(file, res_opts);
} }
if (url.pathname.startsWith("/favicon.")) { // if (url.pathname.startsWith("/favicon.") ) {
const file = Bun.file(path.join(PUBLIC_DIR, url.pathname)); if (url.pathname.match(/\..*$/)) {
return new Response(file); try {
const file_path = path.join(PUBLIC_DIR, url.pathname);
if (!existsSync(file_path)) {
return new Response(`File Doesn't Exist`, {
status: 404,
});
}
const file = Bun.file(file_path);
return new Response(file);
}
catch (error) {
return new Response(`File Not Found`, { status: 404 });
}
} }
return await handleWebPages({ req }); return await handleWebPages({ req });
} }

View File

@ -5,6 +5,7 @@ import AppNames from "../../../utils/grab-app-names";
import { existsSync } from "fs"; import { existsSync } from "fs";
import grabPageErrorComponent from "./grab-page-error-component"; import grabPageErrorComponent from "./grab-page-error-component";
import grabPageBundledReactComponent from "./grab-page-bundled-react-component"; import grabPageBundledReactComponent from "./grab-page-bundled-react-component";
import _ from "lodash";
class NotFoundError extends Error { class NotFoundError extends Error {
} }
export default async function grabPageComponent({ req, file_path: passed_file_path, }) { export default async function grabPageComponent({ req, file_path: passed_file_path, }) {
@ -50,21 +51,40 @@ export default async function grabPageComponent({ req, file_path: passed_file_pa
const now = Date.now(); const now = Date.now();
const module = await import(file_path); const module = await import(file_path);
const serverRes = await (async () => { const serverRes = await (async () => {
const default_props = {
url: {
..._.pick(url, [
"host",
"hostname",
"pathname",
"origin",
"port",
"search",
"searchParams",
"hash",
"href",
"password",
"protocol",
"username",
]),
},
query: match?.query,
};
try { try {
if (routeParams) { if (routeParams) {
const serverData = await module["server"]?.(routeParams); const serverData = await module["server"]?.(routeParams);
return { return {
...serverData, ...serverData,
query: match?.query, ...default_props,
}; };
} }
return { return {
query: match?.query, ...default_props,
}; };
} }
catch (error) { catch (error) {
return { return {
query: match?.query, ...default_props,
}; };
} }
})(); })();

View File

@ -2,7 +2,7 @@
"name": "@moduletrace/bunext", "name": "@moduletrace/bunext",
"module": "index.ts", "module": "index.ts",
"type": "module", "type": "module",
"version": "1.0.4", "version": "1.0.5",
"bin": { "bin": {
"bunext": "dist/index.js" "bunext": "dist/index.js"
}, },

View File

@ -7,6 +7,7 @@ import handleRoutes from "./handle-routes";
import isDevelopment from "../../utils/is-development"; import isDevelopment from "../../utils/is-development";
import grabConstants from "../../utils/grab-constants"; import grabConstants from "../../utils/grab-constants";
import { AppData } from "../../data/app-data"; import { AppData } from "../../data/app-data";
import { existsSync } from "fs";
type Params = { type Params = {
dev?: boolean; dev?: boolean;
@ -91,28 +92,52 @@ export default async function (params?: Params): Promise<ServeOptions> {
} }
if (url.pathname.startsWith("/public/")) { if (url.pathname.startsWith("/public/")) {
const file = Bun.file( try {
path.join( const file_path = path.join(
PUBLIC_DIR, PUBLIC_DIR,
url.pathname.replace(/^\/public/, ""), url.pathname.replace(/^\/public/, ""),
), );
);
let res_opts: ResponseInit = {}; if (!existsSync(file_path)) {
return new Response(`Public File Doesn't Exist`, {
status: 404,
});
}
if (!is_dev && url.pathname.match(/__bunext/)) { const file = Bun.file(file_path);
res_opts.headers = {
"Cache-Control": `public, max-age=${AppData["BunextStaticFilesCacheExpiry"]}, must-revalidate`, let res_opts: ResponseInit = {};
};
if (!is_dev && url.pathname.match(/__bunext/)) {
res_opts.headers = {
"Cache-Control": `public, max-age=${AppData["BunextStaticFilesCacheExpiry"]}, must-revalidate`,
};
}
return new Response(file, res_opts);
} catch (error) {
return new Response(`Public File Not Found`, {
status: 404,
});
} }
return new Response(file, res_opts);
} }
if (url.pathname.startsWith("/favicon.")) { // if (url.pathname.startsWith("/favicon.") ) {
const file = Bun.file(path.join(PUBLIC_DIR, url.pathname)); if (url.pathname.match(/\..*$/)) {
try {
const file_path = path.join(PUBLIC_DIR, url.pathname);
return new Response(file); if (!existsSync(file_path)) {
return new Response(`File Doesn't Exist`, {
status: 404,
});
}
const file = Bun.file(file_path);
return new Response(file);
} catch (error) {
return new Response(`File Not Found`, { status: 404 });
}
} }
return await handleWebPages({ req }); return await handleWebPages({ req });

View File

@ -12,6 +12,7 @@ import AppNames from "../../../utils/grab-app-names";
import { existsSync } from "fs"; import { existsSync } from "fs";
import grabPageErrorComponent from "./grab-page-error-component"; import grabPageErrorComponent from "./grab-page-error-component";
import grabPageBundledReactComponent from "./grab-page-bundled-react-component"; import grabPageBundledReactComponent from "./grab-page-bundled-react-component";
import _ from "lodash";
class NotFoundError extends Error {} class NotFoundError extends Error {}
@ -84,20 +85,40 @@ export default async function grabPageComponent({
const module: BunextPageModule = await import(file_path); const module: BunextPageModule = await import(file_path);
const serverRes: BunextPageModuleServerReturn = await (async () => { const serverRes: BunextPageModuleServerReturn = await (async () => {
const default_props: BunextPageModuleServerReturn = {
url: {
...(_.pick<URL, keyof URL>(url!, [
"host",
"hostname",
"pathname",
"origin",
"port",
"search",
"searchParams",
"hash",
"href",
"password",
"protocol",
"username",
]) as any),
},
query: match?.query,
};
try { try {
if (routeParams) { if (routeParams) {
const serverData = await module["server"]?.(routeParams); const serverData = await module["server"]?.(routeParams);
return { return {
...serverData, ...serverData,
query: match?.query, ...default_props,
}; };
} }
return { return {
query: match?.query, ...default_props,
}; };
} catch (error) { } catch (error) {
return { return {
query: match?.query, ...default_props,
}; };
} }
})(); })();

View File

@ -220,8 +220,11 @@ export type BunextPageModuleServerReturn<
* Expiry time of the cache in seconds * Expiry time of the cache in seconds
*/ */
cacheExpiry?: number; cacheExpiry?: number;
url: BunextPageModuleServerReturnURLObject;
}; };
export type BunextPageModuleServerReturnURLObject = URL & {};
export type BunextPageModuleServerRedirect = { export type BunextPageModuleServerRedirect = {
destination: string; destination: string;
permanent?: boolean; permanent?: boolean;