xplorajs 0.3.3 → 0.3.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -40,6 +40,9 @@ export async function build() {
40
40
  await mkdir(join(process.cwd(), "dist"), {
41
41
  recursive: true
42
42
  });
43
+ await mkdir(join(process.cwd(), ".xplora"), {
44
+ recursive: true
45
+ });
43
46
  const pages = await glob("src/app/**/*.tsx", {
44
47
  ignore: [
45
48
  "**/node_modules/**"
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/commands/build.ts"],"sourcesContent":["import { mkdir, writeFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport { glob } from \"fast-glob\";\nimport React from \"react\";\nimport { generateStaticPage } from \"xplorajs-react\";\n\ninterface Route {\n path: string;\n file: string;\n isDynamic: boolean;\n params: string[];\n}\n\nfunction convertToRoute(filePath: string): Route {\n const relativePath = filePath.replace(\"src/app/\", \"\");\n const path = relativePath\n .replace(/\\.tsx$/, \"\")\n .replace(/\\/page$/, \"\")\n .replace(/\\[([^\\]]+)\\]/g, \":$1\");\n\n const params = (relativePath.match(/\\[([^\\]]+)\\]/g) || []).map((param) =>\n param.slice(1, -1),\n );\n\n return {\n path: path === \"page\" ? \"/\" : `/${path}`,\n file: filePath,\n isDynamic: params.length > 0,\n params,\n };\n}\n\nasync function processPage(page: string, route: Route) {\n try {\n const module = await import(join(process.cwd(), page));\n const PageComponent = module.default;\n\n const getStaticProps = module.getStaticProps;\n let props = {};\n\n if (getStaticProps) {\n const result = await getStaticProps();\n props = result.props;\n }\n\n const outputPath = join(process.cwd(), \"dist\", route.path, \"index.html\");\n await generateStaticPage({\n component: React.createElement(PageComponent, props),\n outputPath,\n props,\n });\n\n console.log(`Generated ${outputPath}`);\n } catch (error) {\n console.error(`Error processing ${page}:`, error);\n }\n}\n\nexport async function build() {\n console.log(\"Building application...\");\n\n await mkdir(join(process.cwd(), \"dist\"), { recursive: true });\n\n const pages = await glob(\"src/app/**/*.tsx\", {\n ignore: [\"**/node_modules/**\"],\n });\n\n const routes: Route[] = [];\n\n for (const page of pages) {\n const route = convertToRoute(page);\n routes.push(route);\n await processPage(page, route);\n }\n\n const routesConfig = {\n routes,\n generatedAt: new Date().toISOString(),\n };\n\n await writeFile(\n join(process.cwd(), \".xplora\", \"routes.json\"),\n JSON.stringify(routesConfig, null, 2),\n );\n\n console.log(\"Build completed!\");\n console.log(\"Routes:\", routes.map((r) => r.path).join(\"\\n\"));\n}\n"],"names":["mkdir","writeFile","join","glob","React","generateStaticPage","convertToRoute","filePath","relativePath","replace","path","params","match","map","param","slice","file","isDynamic","length","processPage","page","route","module","process","cwd","PageComponent","default","getStaticProps","props","result","outputPath","component","createElement","console","log","error","build","recursive","pages","ignore","routes","push","routesConfig","generatedAt","Date","toISOString","JSON","stringify","r"],"mappings":"AAAA,SAASA,KAAK,EAAEC,SAAS,QAAQ,mBAAmB;AACpD,SAASC,IAAI,QAAQ,YAAY;AACjC,SAASC,IAAI,QAAQ,YAAY;AACjC,OAAOC,WAAW,QAAQ;AAC1B,SAASC,kBAAkB,QAAQ,iBAAiB;AASpD,SAASC,eAAeC,QAAgB;IACtC,MAAMC,eAAeD,SAASE,OAAO,CAAC,YAAY;IAClD,MAAMC,OAAOF,aACVC,OAAO,CAAC,UAAU,IAClBA,OAAO,CAAC,WAAW,IACnBA,OAAO,CAAC,iBAAiB;IAE5B,MAAME,SAAS,AAACH,CAAAA,aAAaI,KAAK,CAAC,oBAAoB,EAAE,AAAD,EAAGC,GAAG,CAAC,CAACC,QAC9DA,MAAMC,KAAK,CAAC,GAAG,CAAC;IAGlB,OAAO;QACLL,MAAMA,SAAS,SAAS,MAAM,CAAC,CAAC,EAAEA,MAAM;QACxCM,MAAMT;QACNU,WAAWN,OAAOO,MAAM,GAAG;QAC3BP;IACF;AACF;AAEA,eAAeQ,YAAYC,IAAY,EAAEC,KAAY;IACnD,IAAI;QACF,MAAMC,SAAS,MAAM,MAAM,CAACpB,KAAKqB,QAAQC,GAAG,IAAIJ;QAChD,MAAMK,gBAAgBH,OAAOI,OAAO;QAEpC,MAAMC,iBAAiBL,OAAOK,cAAc;QAC5C,IAAIC,QAAQ,CAAC;QAEb,IAAID,gBAAgB;YAClB,MAAME,SAAS,MAAMF;YACrBC,QAAQC,OAAOD,KAAK;QACtB;QAEA,MAAME,aAAa5B,KAAKqB,QAAQC,GAAG,IAAI,QAAQH,MAAMX,IAAI,EAAE;QAC3D,MAAML,mBAAmB;YACvB0B,WAAW3B,MAAM4B,aAAa,CAACP,eAAeG;YAC9CE;YACAF;QACF;QAEAK,QAAQC,GAAG,CAAC,CAAC,UAAU,EAAEJ,YAAY;IACvC,EAAE,OAAOK,OAAO;QACdF,QAAQE,KAAK,CAAC,CAAC,iBAAiB,EAAEf,KAAK,CAAC,CAAC,EAAEe;IAC7C;AACF;AAEA,OAAO,eAAeC;IACpBH,QAAQC,GAAG,CAAC;IAEZ,MAAMlC,MAAME,KAAKqB,QAAQC,GAAG,IAAI,SAAS;QAAEa,WAAW;IAAK;IAE3D,MAAMC,QAAQ,MAAMnC,KAAK,oBAAoB;QAC3CoC,QAAQ;YAAC;SAAqB;IAChC;IAEA,MAAMC,SAAkB,EAAE;IAE1B,KAAK,MAAMpB,QAAQkB,MAAO;QACxB,MAAMjB,QAAQf,eAAec;QAC7BoB,OAAOC,IAAI,CAACpB;QACZ,MAAMF,YAAYC,MAAMC;IAC1B;IAEA,MAAMqB,eAAe;QACnBF;QACAG,aAAa,IAAIC,OAAOC,WAAW;IACrC;IAEA,MAAM5C,UACJC,KAAKqB,QAAQC,GAAG,IAAI,WAAW,gBAC/BsB,KAAKC,SAAS,CAACL,cAAc,MAAM;IAGrCT,QAAQC,GAAG,CAAC;IACZD,QAAQC,GAAG,CAAC,WAAWM,OAAO3B,GAAG,CAAC,CAACmC,IAAMA,EAAEtC,IAAI,EAAER,IAAI,CAAC;AACxD"}
1
+ {"version":3,"sources":["../../src/commands/build.ts"],"sourcesContent":["import { mkdir, writeFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport { glob } from \"fast-glob\";\nimport React from \"react\";\nimport { generateStaticPage } from \"xplorajs-react\";\n\ninterface Route {\n\tpath: string;\n\tfile: string;\n\tisDynamic: boolean;\n\tparams: string[];\n}\n\nfunction convertToRoute(filePath: string): Route {\n\tconst relativePath = filePath.replace(\"src/app/\", \"\");\n\tconst path = relativePath\n\t\t.replace(/\\.tsx$/, \"\")\n\t\t.replace(/\\/page$/, \"\")\n\t\t.replace(/\\[([^\\]]+)\\]/g, \":$1\");\n\n\tconst params = (relativePath.match(/\\[([^\\]]+)\\]/g) || []).map((param) =>\n\t\tparam.slice(1, -1),\n\t);\n\n\treturn {\n\t\tpath: path === \"page\" ? \"/\" : `/${path}`,\n\t\tfile: filePath,\n\t\tisDynamic: params.length > 0,\n\t\tparams,\n\t};\n}\n\nasync function processPage(page: string, route: Route) {\n\ttry {\n\t\tconst module = await import(join(process.cwd(), page));\n\t\tconst PageComponent = module.default;\n\n\t\tconst getStaticProps = module.getStaticProps;\n\t\tlet props = {};\n\n\t\tif (getStaticProps) {\n\t\t\tconst result = await getStaticProps();\n\t\t\tprops = result.props;\n\t\t}\n\n\t\tconst outputPath = join(process.cwd(), \"dist\", route.path, \"index.html\");\n\t\tawait generateStaticPage({\n\t\t\tcomponent: React.createElement(PageComponent, props),\n\t\t\toutputPath,\n\t\t\tprops,\n\t\t});\n\n\t\tconsole.log(`Generated ${outputPath}`);\n\t} catch (error) {\n\t\tconsole.error(`Error processing ${page}:`, error);\n\t}\n}\n\nexport async function build() {\n\tconsole.log(\"Building application...\");\n\n\tawait mkdir(join(process.cwd(), \"dist\"), { recursive: true });\n\tawait mkdir(join(process.cwd(), \".xplora\"), { recursive: true });\n\n\tconst pages = await glob(\"src/app/**/*.tsx\", {\n\t\tignore: [\"**/node_modules/**\"],\n\t});\n\n\tconst routes: Route[] = [];\n\n\tfor (const page of pages) {\n\t\tconst route = convertToRoute(page);\n\t\troutes.push(route);\n\t\tawait processPage(page, route);\n\t}\n\n\tconst routesConfig = {\n\t\troutes,\n\t\tgeneratedAt: new Date().toISOString(),\n\t};\n\n\tawait writeFile(\n\t\tjoin(process.cwd(), \".xplora\", \"routes.json\"),\n\t\tJSON.stringify(routesConfig, null, 2),\n\t);\n\n\tconsole.log(\"Build completed!\");\n\tconsole.log(\"Routes:\", routes.map((r) => r.path).join(\"\\n\"));\n}\n"],"names":["mkdir","writeFile","join","glob","React","generateStaticPage","convertToRoute","filePath","relativePath","replace","path","params","match","map","param","slice","file","isDynamic","length","processPage","page","route","module","process","cwd","PageComponent","default","getStaticProps","props","result","outputPath","component","createElement","console","log","error","build","recursive","pages","ignore","routes","push","routesConfig","generatedAt","Date","toISOString","JSON","stringify","r"],"mappings":"AAAA,SAASA,KAAK,EAAEC,SAAS,QAAQ,mBAAmB;AACpD,SAASC,IAAI,QAAQ,YAAY;AACjC,SAASC,IAAI,QAAQ,YAAY;AACjC,OAAOC,WAAW,QAAQ;AAC1B,SAASC,kBAAkB,QAAQ,iBAAiB;AASpD,SAASC,eAAeC,QAAgB;IACvC,MAAMC,eAAeD,SAASE,OAAO,CAAC,YAAY;IAClD,MAAMC,OAAOF,aACXC,OAAO,CAAC,UAAU,IAClBA,OAAO,CAAC,WAAW,IACnBA,OAAO,CAAC,iBAAiB;IAE3B,MAAME,SAAS,AAACH,CAAAA,aAAaI,KAAK,CAAC,oBAAoB,EAAE,AAAD,EAAGC,GAAG,CAAC,CAACC,QAC/DA,MAAMC,KAAK,CAAC,GAAG,CAAC;IAGjB,OAAO;QACNL,MAAMA,SAAS,SAAS,MAAM,CAAC,CAAC,EAAEA,MAAM;QACxCM,MAAMT;QACNU,WAAWN,OAAOO,MAAM,GAAG;QAC3BP;IACD;AACD;AAEA,eAAeQ,YAAYC,IAAY,EAAEC,KAAY;IACpD,IAAI;QACH,MAAMC,SAAS,MAAM,MAAM,CAACpB,KAAKqB,QAAQC,GAAG,IAAIJ;QAChD,MAAMK,gBAAgBH,OAAOI,OAAO;QAEpC,MAAMC,iBAAiBL,OAAOK,cAAc;QAC5C,IAAIC,QAAQ,CAAC;QAEb,IAAID,gBAAgB;YACnB,MAAME,SAAS,MAAMF;YACrBC,QAAQC,OAAOD,KAAK;QACrB;QAEA,MAAME,aAAa5B,KAAKqB,QAAQC,GAAG,IAAI,QAAQH,MAAMX,IAAI,EAAE;QAC3D,MAAML,mBAAmB;YACxB0B,WAAW3B,MAAM4B,aAAa,CAACP,eAAeG;YAC9CE;YACAF;QACD;QAEAK,QAAQC,GAAG,CAAC,CAAC,UAAU,EAAEJ,YAAY;IACtC,EAAE,OAAOK,OAAO;QACfF,QAAQE,KAAK,CAAC,CAAC,iBAAiB,EAAEf,KAAK,CAAC,CAAC,EAAEe;IAC5C;AACD;AAEA,OAAO,eAAeC;IACrBH,QAAQC,GAAG,CAAC;IAEZ,MAAMlC,MAAME,KAAKqB,QAAQC,GAAG,IAAI,SAAS;QAAEa,WAAW;IAAK;IAC3D,MAAMrC,MAAME,KAAKqB,QAAQC,GAAG,IAAI,YAAY;QAAEa,WAAW;IAAK;IAE9D,MAAMC,QAAQ,MAAMnC,KAAK,oBAAoB;QAC5CoC,QAAQ;YAAC;SAAqB;IAC/B;IAEA,MAAMC,SAAkB,EAAE;IAE1B,KAAK,MAAMpB,QAAQkB,MAAO;QACzB,MAAMjB,QAAQf,eAAec;QAC7BoB,OAAOC,IAAI,CAACpB;QACZ,MAAMF,YAAYC,MAAMC;IACzB;IAEA,MAAMqB,eAAe;QACpBF;QACAG,aAAa,IAAIC,OAAOC,WAAW;IACpC;IAEA,MAAM5C,UACLC,KAAKqB,QAAQC,GAAG,IAAI,WAAW,gBAC/BsB,KAAKC,SAAS,CAACL,cAAc,MAAM;IAGpCT,QAAQC,GAAG,CAAC;IACZD,QAAQC,GAAG,CAAC,WAAWM,OAAO3B,GAAG,CAAC,CAACmC,IAAMA,EAAEtC,IAAI,EAAER,IAAI,CAAC;AACvD"}
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/commands/dev.ts"],"sourcesContent":["import { readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { serve } from \"bun\";\nimport { watch } from \"chokidar\";\nimport React from \"react\";\nimport type { ComponentType } from \"react\";\nimport { WebSocketServer } from \"ws\";\nimport { renderToStream } from \"xplorajs-react\";\nimport { build } from \"./build\";\n\n// biome-ignore lint/suspicious/noExplicitAny: <intended>\nconst pages = new Map<string, ComponentType<Record<string, any>>>();\n\nasync function loadPages() {\n pages.clear();\n const routes = await loadRoutes();\n\n for (const route of routes) {\n const abs = join(process.cwd(), route.file);\n delete import.meta.require?.cache?.[abs];\n pages.set(\n route.path,\n // biome-ignore lint/suspicious/noExplicitAny: <intended>\n (await import(abs)).default as ComponentType<Record<string, any>>,\n );\n }\n}\n\nasync function loadRoutes() {\n try {\n const routesPath = join(process.cwd(), \".xplora\", \"routes.json\");\n const routesContent = readFileSync(routesPath, \"utf-8\");\n return JSON.parse(routesContent).routes;\n } catch {\n return [];\n }\n}\n\nexport async function dev() {\n console.log(\"Starting development server...\");\n\n const wss = new WebSocketServer({ port: 3001 });\n\n const watcher = watch([\"src/**/*\"], {\n ignored: /(^|[\\/\\\\])\\../,\n persistent: true,\n });\n\n watcher.on(\"change\", async (path: string) => {\n console.log(`File ${path} has been changed`);\n await build();\n await loadPages();\n\n for (const client of wss.clients) {\n if (client.readyState === WebSocket.OPEN) {\n client.send(JSON.stringify({ type: \"reload\" }));\n }\n }\n });\n\n await build();\n await loadPages();\n\n serve({\n port: 3000,\n development: true,\n async fetch(req: Request) {\n const url = new URL(req.url);\n const path = url.pathname;\n\n if (path === \"/\") {\n return new Response(\"Hello from XploraJS!\");\n }\n\n if (path.startsWith(\"/assets/\")) {\n const f = Bun.file(join(process.cwd(), \"dist\", path));\n if (await f.exists()) return new Response(f);\n }\n\n const Page = pages.get(path);\n if (!Page) return new Response(\"Not Found\", { status: 404 });\n\n const stream = await renderToStream(React.createElement(Page));\n\n const head = `<!DOCTYPE html><html><head>\n\t\t\t\t<meta charset=\"utf-8\"/>\n\t\t\t\t<link rel=\"stylesheet\" href=\"/assets/style.css\"/>\n\t\t\t\t<script>window.process={env:{NODE_ENV:\"development\"}};</script>\n\t\t\t\t</head><body><div id=\"root\">`;\n\n const foot = `</div>\n\t\t\t\t<script type=\"module\">\n\t\t\t\t\timport * as RefreshRuntime from \"https://esm.sh/react-refresh@0.17.0/runtime\";\n\t\t\t\t\tif (window.process?.env?.NODE_ENV === \"development\") {\n\t\t\t\t\t\tconsole.log(\"Fast refresh runtime loaded in development mode\");\n\t\t\t\t\t}\n\t\t\t\t\tRefreshRuntime.injectIntoGlobalHook(window);\n\t\t\t\t\twindow.$RefreshReg$=()=>{};\n\t\t\t\t\twindow.$RefreshSig$=()=>t=>t;\n\t\t\t\t</script>\n\t\t\t\t<script>\n\t\t\t\t\tconst ws=new WebSocket(\"ws://localhost:3001\");\n\t\t\t\t\tws.onmessage=async e=>{\n\t\t\t\t\t\tif(e.data===\"css\"){\n\t\t\t\t\t\t\tdocument.querySelectorAll('link[rel=\"stylesheet\"]').forEach(l=>l.href=\"/assets/style.css?v=\"+Date.now());\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif(e.data===\"reload\"){\n\t\t\t\t\t\t\tconsole.log(\"Reloading...\");\n\t\t\t\t\t\t\tlocation.reload();\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\t\t\t\t</script>\n\t\t\t</body></html>`;\n\n return new Response(\n new ReadableStream({\n start(ctrl) {\n ctrl.enqueue(new TextEncoder().encode(head));\n if (typeof stream === \"string\") {\n ctrl.enqueue(new TextEncoder().encode(stream));\n } else {\n stream.pipeTo(\n new WritableStream({\n write(c) {\n ctrl.enqueue(c);\n },\n close() {\n ctrl.enqueue(new TextEncoder().encode(foot));\n ctrl.close();\n },\n }),\n );\n }\n },\n }),\n { headers: { \"Content-Type\": \"text/html; charset=utf-8\" } },\n );\n },\n });\n\n console.log(\"Development server running at http://localhost:3000\");\n}\n"],"names":["readFileSync","join","serve","watch","React","WebSocketServer","renderToStream","build","pages","Map","loadPages","clear","routes","loadRoutes","route","abs","process","cwd","file","require","cache","set","path","default","routesPath","routesContent","JSON","parse","dev","console","log","wss","port","watcher","ignored","persistent","on","client","clients","readyState","WebSocket","OPEN","send","stringify","type","development","fetch","req","url","URL","pathname","Response","startsWith","f","Bun","exists","Page","get","status","stream","createElement","head","foot","ReadableStream","start","ctrl","enqueue","TextEncoder","encode","pipeTo","WritableStream","write","c","close","headers"],"mappings":"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"}
1
+ {"version":3,"sources":["../../src/commands/dev.ts"],"sourcesContent":["import { readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { serve } from \"bun\";\nimport { watch } from \"chokidar\";\nimport React from \"react\";\nimport type { ComponentType } from \"react\";\nimport { WebSocketServer } from \"ws\";\nimport { renderToStream } from \"xplorajs-react\";\nimport { build } from \"./build\";\n\n// biome-ignore lint/suspicious/noExplicitAny: <intended>\nconst pages = new Map<string, ComponentType<Record<string, any>>>();\n\nasync function loadPages() {\n\tpages.clear();\n\tconst routes = await loadRoutes();\n\n\tfor (const route of routes) {\n\t\tconst abs = join(process.cwd(), route.file);\n\t\tdelete import.meta.require?.cache?.[abs];\n\t\tpages.set(\n\t\t\troute.path,\n\t\t\t// biome-ignore lint/suspicious/noExplicitAny: <intended>\n\t\t\t(await import(abs)).default as ComponentType<Record<string, any>>,\n\t\t);\n\t}\n}\n\nasync function loadRoutes() {\n\ttry {\n\t\tconst routesPath = join(process.cwd(), \".xplora\", \"routes.json\");\n\t\tconst routesContent = readFileSync(routesPath, \"utf-8\");\n\t\treturn JSON.parse(routesContent).routes;\n\t} catch {\n\t\treturn [];\n\t}\n}\n\nexport async function dev() {\n\tconsole.log(\"Starting development server...\");\n\n\tconst wss = new WebSocketServer({ port: 3001 });\n\n\tconst watcher = watch([\"src/**/*\"], {\n\t\tignored: /(^|[\\/\\\\])\\../,\n\t\tpersistent: true,\n\t});\n\n\twatcher.on(\"change\", async (path: string) => {\n\t\tconsole.log(`File ${path} has been changed`);\n\t\tawait build();\n\t\tawait loadPages();\n\n\t\tfor (const client of wss.clients) {\n\t\t\tif (client.readyState === WebSocket.OPEN) {\n\t\t\t\tclient.send(JSON.stringify({ type: \"reload\" }));\n\t\t\t}\n\t\t}\n\t});\n\n\tawait build();\n\tawait loadPages();\n\n\tserve({\n\t\tport: 3000,\n\t\tdevelopment: true,\n\t\tasync fetch(req: Request) {\n\t\t\tconst url = new URL(req.url);\n\t\t\tconst path = url.pathname;\n\n\t\t\tif (path === \"/\") {\n\t\t\t\treturn new Response(\"Hello from XploraJS!\");\n\t\t\t}\n\n\t\t\tif (path.startsWith(\"/assets/\")) {\n\t\t\t\tconst f = Bun.file(join(process.cwd(), \"dist\", path));\n\t\t\t\tif (await f.exists()) return new Response(f);\n\t\t\t}\n\n\t\t\tconst Page = pages.get(path);\n\t\t\tif (!Page) return new Response(\"Not Found\", { status: 404 });\n\n\t\t\tconst stream = await renderToStream(React.createElement(Page));\n\n\t\t\tconst head = `<!DOCTYPE html><html><head>\n\t\t\t\t<meta charset=\"utf-8\"/>\n\t\t\t\t<link rel=\"stylesheet\" href=\"/assets/style.css\"/>\n\t\t\t\t<script>window.process={env:{NODE_ENV:\"development\"}};</script>\n\t\t\t\t</head><body><div id=\"root\">`;\n\n\t\t\tconst foot = `</div>\n\t\t\t\t<script type=\"module\">\n\t\t\t\t\timport * as RefreshRuntime from \"https://esm.sh/react-refresh@0.17.0/runtime\";\n\t\t\t\t\tif (window.process?.env?.NODE_ENV === \"development\") {\n\t\t\t\t\t\tconsole.log(\"Fast refresh runtime loaded in development mode\");\n\t\t\t\t\t}\n\t\t\t\t\tRefreshRuntime.injectIntoGlobalHook(window);\n\t\t\t\t\twindow.$RefreshReg$=()=>{};\n\t\t\t\t\twindow.$RefreshSig$=()=>t=>t;\n\t\t\t\t</script>\n\t\t\t\t<script>\n\t\t\t\t\tconst ws=new WebSocket(\"ws://localhost:3001\");\n\t\t\t\t\tws.onmessage=async e=>{\n\t\t\t\t\t\tif(e.data===\"css\"){\n\t\t\t\t\t\t\tdocument.querySelectorAll('link[rel=\"stylesheet\"]').forEach(l=>l.href=\"/assets/style.css?v=\"+Date.now());\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif(e.data===\"reload\"){\n\t\t\t\t\t\t\tconsole.log(\"Reloading...\");\n\t\t\t\t\t\t\tlocation.reload();\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\t\t\t\t</script>\n\t\t\t</body></html>`;\n\n\t\t\treturn new Response(\n\t\t\t\tnew ReadableStream({\n\t\t\t\t\tstart(ctrl) {\n\t\t\t\t\t\tctrl.enqueue(new TextEncoder().encode(head));\n\t\t\t\t\t\tif (typeof stream === \"string\") {\n\t\t\t\t\t\t\tctrl.enqueue(new TextEncoder().encode(stream));\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tstream.pipeTo(\n\t\t\t\t\t\t\t\tnew WritableStream({\n\t\t\t\t\t\t\t\t\twrite(c) {\n\t\t\t\t\t\t\t\t\t\tctrl.enqueue(c);\n\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\tclose() {\n\t\t\t\t\t\t\t\t\t\tctrl.enqueue(new TextEncoder().encode(foot));\n\t\t\t\t\t\t\t\t\t\tctrl.close();\n\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t\t);\n\t\t\t\t\t\t}\n\t\t\t\t\t},\n\t\t\t\t}),\n\t\t\t\t{ headers: { \"Content-Type\": \"text/html; charset=utf-8\" } },\n\t\t\t);\n\t\t},\n\t});\n\n\tconsole.log(\"Development server running at http://localhost:3000\");\n}\n"],"names":["readFileSync","join","serve","watch","React","WebSocketServer","renderToStream","build","pages","Map","loadPages","clear","routes","loadRoutes","route","abs","process","cwd","file","require","cache","set","path","default","routesPath","routesContent","JSON","parse","dev","console","log","wss","port","watcher","ignored","persistent","on","client","clients","readyState","WebSocket","OPEN","send","stringify","type","development","fetch","req","url","URL","pathname","Response","startsWith","f","Bun","exists","Page","get","status","stream","createElement","head","foot","ReadableStream","start","ctrl","enqueue","TextEncoder","encode","pipeTo","WritableStream","write","c","close","headers"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "xplorajs",
3
- "version": "0.3.3",
3
+ "version": "0.3.4",
4
4
  "type": "module",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",