@aklinker1/aframe 0.2.5 → 0.4.0
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.
- package/README.md +20 -7
- package/bin/aframe.ts +1 -21
- package/package.json +7 -17
- package/src/client/server.ts +53 -39
- package/src/config.ts +17 -30
- package/src/dev-server.ts +45 -0
- package/src/index.ts +3 -35
- package/src/prerenderer.ts +63 -0
package/README.md
CHANGED
|
@@ -13,6 +13,7 @@ Simple wrapper around Vite for creating pre-rendered, client-side web apps with
|
|
|
13
13
|
📁 public/
|
|
14
14
|
📄 favicon.ico
|
|
15
15
|
📁 server/
|
|
16
|
+
📄 env.d.ts
|
|
16
17
|
📄 main.ts
|
|
17
18
|
📄 .env
|
|
18
19
|
📄 aframe.config.ts
|
|
@@ -34,6 +35,11 @@ export default defineConfig({
|
|
|
34
35
|
});
|
|
35
36
|
```
|
|
36
37
|
|
|
38
|
+
```ts
|
|
39
|
+
// server/env.d.ts
|
|
40
|
+
/// <reference types="@aklinker1/aframe/env" />
|
|
41
|
+
```
|
|
42
|
+
|
|
37
43
|
```ts
|
|
38
44
|
// server/main.ts
|
|
39
45
|
import { Elysia } from "elysia";
|
|
@@ -62,6 +68,11 @@ export default app;
|
|
|
62
68
|
</html>
|
|
63
69
|
```
|
|
64
70
|
|
|
71
|
+
```sh
|
|
72
|
+
bun add elysia @aklinker1/aframe
|
|
73
|
+
bun add -D puppeteer vite
|
|
74
|
+
```
|
|
75
|
+
|
|
65
76
|
```jsonc
|
|
66
77
|
// package.json
|
|
67
78
|
{
|
|
@@ -69,16 +80,18 @@ export default app;
|
|
|
69
80
|
"version": "1.0.0",
|
|
70
81
|
"packageManager": "bun@1.2.2",
|
|
71
82
|
"scripts": {
|
|
72
|
-
"
|
|
73
|
-
"
|
|
83
|
+
"aframe": "bun node_modules/@aklinker1/aframe/bin/aframe.ts",
|
|
84
|
+
"dev": "bun --silent aframe",
|
|
85
|
+
"build": "bun --silent aframe build",
|
|
74
86
|
"preview": "bun --cwd .output --env-file ../.env server-entry.js"
|
|
75
87
|
},
|
|
76
88
|
"dependencies": {
|
|
77
|
-
"@aklinker1/aframe": "
|
|
78
|
-
"
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
"vite": "
|
|
89
|
+
"@aklinker1/aframe": "latest",
|
|
90
|
+
"elysia": "latest"
|
|
91
|
+
},
|
|
92
|
+
"devDependencies": {
|
|
93
|
+
"vite": "latest",
|
|
94
|
+
"puppeteer": "latest",
|
|
82
95
|
}
|
|
83
96
|
}
|
|
84
97
|
```
|
package/bin/aframe.ts
CHANGED
|
@@ -10,27 +10,7 @@ async function dev(root?: string) {
|
|
|
10
10
|
console.log(`${BOLD}${CYAN}ℹ${RESET} Spinning up dev servers...${RESET}`);
|
|
11
11
|
const config = await resolveConfig(root, "serve", "development");
|
|
12
12
|
const devServer = await createServer(config);
|
|
13
|
-
|
|
14
|
-
await devServer.listen(config.appPort).then(() => {
|
|
15
|
-
const js = [
|
|
16
|
-
`import server from '${config.serverModule}';`,
|
|
17
|
-
`server.listen(${config.serverPort});`,
|
|
18
|
-
].join("\n");
|
|
19
|
-
Bun.spawn({
|
|
20
|
-
cmd: [
|
|
21
|
-
"bun",
|
|
22
|
-
"--watch",
|
|
23
|
-
"--define",
|
|
24
|
-
`import.meta.publicDir:"${config.publicDir}"`,
|
|
25
|
-
"--define",
|
|
26
|
-
`import.meta.command:"serve"`,
|
|
27
|
-
"--eval",
|
|
28
|
-
js,
|
|
29
|
-
],
|
|
30
|
-
stdio: ["inherit", "inherit", "inherit"],
|
|
31
|
-
cwd: config.rootDir,
|
|
32
|
-
});
|
|
33
|
-
});
|
|
13
|
+
await devServer.listen();
|
|
34
14
|
|
|
35
15
|
console.log(`${GREEN}✔${RESET} Dev servers started in ${devServerTimer()}`);
|
|
36
16
|
console.log(
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@aklinker1/aframe",
|
|
3
|
-
"version": "0.
|
|
4
|
-
"packageManager": "bun@1.2.
|
|
3
|
+
"version": "0.4.0",
|
|
4
|
+
"packageManager": "bun@1.2.5",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"exports": {
|
|
@@ -19,34 +19,24 @@
|
|
|
19
19
|
],
|
|
20
20
|
"scripts": {
|
|
21
21
|
"check": "check",
|
|
22
|
-
"
|
|
23
|
-
"
|
|
22
|
+
"aframe": "bun --silent bin/aframe.ts",
|
|
23
|
+
"dev": "bun aframe demo",
|
|
24
|
+
"build": "bun aframe build demo",
|
|
24
25
|
"preview": "bun --cwd demo/.output --env-file ../.env server-entry.js"
|
|
25
26
|
},
|
|
26
27
|
"dependencies": {},
|
|
27
28
|
"devDependencies": {
|
|
28
29
|
"@aklinker1/check": "^1.4.5",
|
|
29
|
-
"@prerenderer/renderer-puppeteer": "1.2.4",
|
|
30
30
|
"@types/bun": "latest",
|
|
31
31
|
"oxlint": "^0.15.11",
|
|
32
32
|
"prettier": "^3.5.2",
|
|
33
33
|
"publint": "^0.3.6",
|
|
34
|
-
"puppeteer": "^24.
|
|
34
|
+
"puppeteer": "^24.4.0",
|
|
35
35
|
"typescript": "^5.0.0",
|
|
36
36
|
"vite": "^6.1.1"
|
|
37
37
|
},
|
|
38
38
|
"peerDependencies": {
|
|
39
39
|
"vite": "*",
|
|
40
|
-
"
|
|
41
|
-
"@prerenderer/renderer-jsdom": "*",
|
|
42
|
-
"@prerenderer/renderer-puppeteer": "*"
|
|
43
|
-
},
|
|
44
|
-
"peerDependenciesMeta": {
|
|
45
|
-
"@prerenderer/prerenderer-jsdom": {
|
|
46
|
-
"optional": true
|
|
47
|
-
},
|
|
48
|
-
"@prerenderer/renderer-puppeteer": {
|
|
49
|
-
"optional": true
|
|
50
|
-
}
|
|
40
|
+
"puppeteer": "*"
|
|
51
41
|
}
|
|
52
42
|
}
|
package/src/client/server.ts
CHANGED
|
@@ -11,53 +11,67 @@ export interface AframeServer {
|
|
|
11
11
|
/**
|
|
12
12
|
* Fetches a file from the `public` directory.
|
|
13
13
|
*/
|
|
14
|
-
export
|
|
15
|
-
|
|
14
|
+
export function fetchStatic(options?: {
|
|
15
|
+
/** Override the fetch behavior of a file. */
|
|
16
|
+
onFetch?: (
|
|
17
|
+
path: string,
|
|
18
|
+
file: BunFile,
|
|
19
|
+
) => Promise<Response | undefined> | Response | undefined;
|
|
20
|
+
}): (request: Request) => Promise<Response> {
|
|
21
|
+
return async (request) => {
|
|
22
|
+
const path = new URL(request.url).pathname.replace(/\/+$/, "");
|
|
16
23
|
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
24
|
+
const paths = [
|
|
25
|
+
`${import.meta.publicDir}${path}`,
|
|
26
|
+
`${import.meta.publicDir}${path}/index.html`,
|
|
27
|
+
];
|
|
21
28
|
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
29
|
+
// Only fallback on the root HTML file when building application
|
|
30
|
+
if (import.meta.command === "build") {
|
|
31
|
+
paths.push(`${import.meta.publicDir}/index.html`);
|
|
32
|
+
}
|
|
26
33
|
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
34
|
+
for (const path of paths) {
|
|
35
|
+
const isHtml = path.includes(".html");
|
|
36
|
+
const gzFile = Bun.file(path + ".gz");
|
|
37
|
+
const file = Bun.file(path);
|
|
30
38
|
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
39
|
+
if (await isFile(gzFile)) {
|
|
40
|
+
const customResponse = await options?.onFetch?.(path, file);
|
|
41
|
+
if (customResponse) return customResponse;
|
|
42
|
+
return new Response(gzFile.stream(), {
|
|
43
|
+
headers: {
|
|
44
|
+
...(isHtml ? {} : headers),
|
|
45
|
+
"content-type": file.type,
|
|
46
|
+
"content-encoding": "gzip",
|
|
47
|
+
},
|
|
48
|
+
});
|
|
49
|
+
}
|
|
40
50
|
|
|
41
|
-
|
|
42
|
-
|
|
51
|
+
if (await isFile(file)) {
|
|
52
|
+
const customResponse = await options?.onFetch?.(path, file);
|
|
53
|
+
if (customResponse) return customResponse;
|
|
54
|
+
|
|
55
|
+
return new Response(file.stream(), { headers });
|
|
56
|
+
}
|
|
43
57
|
}
|
|
44
|
-
}
|
|
45
58
|
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
59
|
+
return new Response(
|
|
60
|
+
`<html>
|
|
61
|
+
<body>
|
|
62
|
+
This is a placeholder for your root <code>index.html</code> file during development.
|
|
63
|
+
<br/>
|
|
64
|
+
In production (or via the app's dev server), this path will fallback on the root <code>index.html</code>.
|
|
65
|
+
</body>
|
|
66
|
+
</html>`,
|
|
67
|
+
{
|
|
68
|
+
headers: {
|
|
69
|
+
"Content-Type": "text/html",
|
|
70
|
+
...headers,
|
|
71
|
+
},
|
|
58
72
|
},
|
|
59
|
-
|
|
60
|
-
|
|
73
|
+
);
|
|
74
|
+
};
|
|
61
75
|
}
|
|
62
76
|
|
|
63
77
|
async function isFile(file: BunFile): Promise<boolean> {
|
package/src/config.ts
CHANGED
|
@@ -1,12 +1,25 @@
|
|
|
1
1
|
import * as vite from "vite";
|
|
2
|
-
import type { PrerendererOptions } from "@prerenderer/prerenderer";
|
|
3
2
|
import { resolve, join, relative } from "node:path/posix";
|
|
4
3
|
import { mkdir } from "node:fs/promises";
|
|
5
4
|
|
|
6
5
|
export type UserConfig = {
|
|
7
6
|
vite?: vite.UserConfigExport;
|
|
8
7
|
prerenderedRoutes?: string[];
|
|
9
|
-
prerenderer?:
|
|
8
|
+
prerenderer?: PrerendererConfig;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
export type PrerendererConfig = {
|
|
12
|
+
/** Wait for an selector`document.querySelector` to be in the DOM before grabbing the HTML. */
|
|
13
|
+
waitForSelector?: string;
|
|
14
|
+
/** When `waitForSelector` is set, also wait for the element to be visible before grabbing the HTML. */
|
|
15
|
+
waitForSelectorVisible?: boolean;
|
|
16
|
+
/** Wait a set timeout in milliseconds before grabbing the HTML. */
|
|
17
|
+
waitForTimeout?: number;
|
|
18
|
+
/**
|
|
19
|
+
* Timeout before prerendering throws an error.
|
|
20
|
+
* @default 30e3
|
|
21
|
+
*/
|
|
22
|
+
timeout?: number;
|
|
10
23
|
};
|
|
11
24
|
|
|
12
25
|
export type ResolvedConfig = {
|
|
@@ -24,7 +37,7 @@ export type ResolvedConfig = {
|
|
|
24
37
|
serverPort: number;
|
|
25
38
|
vite: vite.InlineConfig;
|
|
26
39
|
prerenderedRoutes: string[];
|
|
27
|
-
prerenderer:
|
|
40
|
+
prerenderer: PrerendererConfig;
|
|
28
41
|
};
|
|
29
42
|
|
|
30
43
|
export function defineConfig(config: UserConfig): UserConfig {
|
|
@@ -103,24 +116,6 @@ export async function resolveConfig(
|
|
|
103
116
|
},
|
|
104
117
|
);
|
|
105
118
|
|
|
106
|
-
const prerenderer = async (): Promise<PrerendererOptions> => {
|
|
107
|
-
const rendererModule =
|
|
108
|
-
tryResolve("@prerenderer/renderer-puppeteer") ??
|
|
109
|
-
tryResolve("@prerenderer/renderer-jsdom");
|
|
110
|
-
if (!rendererModule)
|
|
111
|
-
throw Error(
|
|
112
|
-
`No renderer installed. Did you forget to install @prerenderer/renderer-puppeteer or @prerenderer/renderer-jsdom?`,
|
|
113
|
-
);
|
|
114
|
-
|
|
115
|
-
const { default: Renderer } = await import(rendererModule);
|
|
116
|
-
const renderer = new Renderer();
|
|
117
|
-
return {
|
|
118
|
-
...userConfig.prerenderer,
|
|
119
|
-
renderer,
|
|
120
|
-
staticDir: appOutDir,
|
|
121
|
-
};
|
|
122
|
-
};
|
|
123
|
-
|
|
124
119
|
return {
|
|
125
120
|
rootDir,
|
|
126
121
|
appDir,
|
|
@@ -136,15 +131,7 @@ export async function resolveConfig(
|
|
|
136
131
|
serverPort,
|
|
137
132
|
|
|
138
133
|
prerenderedRoutes: userConfig.prerenderedRoutes ?? ["/"],
|
|
139
|
-
prerenderer,
|
|
140
134
|
vite: viteConfig,
|
|
135
|
+
prerenderer: userConfig.prerenderer ?? {},
|
|
141
136
|
};
|
|
142
137
|
}
|
|
143
|
-
|
|
144
|
-
function tryResolve(specifier: string): string | undefined {
|
|
145
|
-
try {
|
|
146
|
-
return import.meta.resolve(specifier);
|
|
147
|
-
} catch {
|
|
148
|
-
return undefined;
|
|
149
|
-
}
|
|
150
|
-
}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { createServer as createViteServer, type ViteDevServer } from "vite";
|
|
2
|
+
import type { ResolvedConfig } from "./config";
|
|
3
|
+
import type { Subprocess } from "bun";
|
|
4
|
+
|
|
5
|
+
export async function createServer(
|
|
6
|
+
config: ResolvedConfig,
|
|
7
|
+
): Promise<ViteDevServer> {
|
|
8
|
+
const devServer = await createViteServer(config.vite);
|
|
9
|
+
const ogListen = devServer.listen.bind(devServer);
|
|
10
|
+
const ogClose = devServer.close.bind(devServer);
|
|
11
|
+
|
|
12
|
+
let serverProcess: Subprocess | undefined;
|
|
13
|
+
const startServer = () => {
|
|
14
|
+
const js = [
|
|
15
|
+
`import server from '${config.serverModule}';`,
|
|
16
|
+
`server.listen(${config.serverPort});`,
|
|
17
|
+
].join("\n");
|
|
18
|
+
return Bun.spawn({
|
|
19
|
+
cmd: [
|
|
20
|
+
"bun",
|
|
21
|
+
"--watch",
|
|
22
|
+
"--define",
|
|
23
|
+
`import.meta.publicDir:"${config.publicDir}"`,
|
|
24
|
+
"--define",
|
|
25
|
+
`import.meta.command:"serve"`,
|
|
26
|
+
"--eval",
|
|
27
|
+
js,
|
|
28
|
+
],
|
|
29
|
+
stdio: ["inherit", "inherit", "inherit"],
|
|
30
|
+
cwd: config.rootDir,
|
|
31
|
+
});
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
devServer.listen = async (port, isRestart) => {
|
|
35
|
+
const res = await ogListen(port, isRestart);
|
|
36
|
+
serverProcess = startServer();
|
|
37
|
+
return res;
|
|
38
|
+
};
|
|
39
|
+
devServer.close = async () => {
|
|
40
|
+
void ogClose();
|
|
41
|
+
serverProcess?.kill("SIGINT");
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
return devServer;
|
|
45
|
+
}
|
package/src/index.ts
CHANGED
|
@@ -1,4 +1,3 @@
|
|
|
1
|
-
import Prerenderer from "@prerenderer/prerenderer";
|
|
2
1
|
import type { BunPlugin } from "bun";
|
|
3
2
|
import { lstatSync } from "node:fs";
|
|
4
3
|
import { mkdir, rm, writeFile } from "node:fs/promises";
|
|
@@ -7,14 +6,10 @@ import * as vite from "vite";
|
|
|
7
6
|
import { BLUE, BOLD, CYAN, DIM, GREEN, MAGENTA, RESET } from "./color";
|
|
8
7
|
import type { ResolvedConfig } from "./config";
|
|
9
8
|
import { createTimer } from "./timer";
|
|
9
|
+
import { prerenderPages } from "./prerenderer";
|
|
10
10
|
|
|
11
11
|
export * from "./config";
|
|
12
|
-
|
|
13
|
-
export async function createServer(
|
|
14
|
-
config: ResolvedConfig,
|
|
15
|
-
): Promise<vite.ViteDevServer> {
|
|
16
|
-
return await vite.createServer(config.vite);
|
|
17
|
-
}
|
|
12
|
+
export * from "./dev-server";
|
|
18
13
|
|
|
19
14
|
export async function build(config: ResolvedConfig) {
|
|
20
15
|
const buildTimer = createTimer();
|
|
@@ -62,34 +57,7 @@ export async function build(config: ResolvedConfig) {
|
|
|
62
57
|
.map((route) => ` ${DIM}-${RESET} ${CYAN}${route}${RESET}`)
|
|
63
58
|
.join("\n"),
|
|
64
59
|
);
|
|
65
|
-
const
|
|
66
|
-
const prerendered = await prerenderer
|
|
67
|
-
.initialize()
|
|
68
|
-
.then(() =>
|
|
69
|
-
prerenderer.renderRoutes(
|
|
70
|
-
config.prerenderedRoutes.map((route) => `${route}?prerender`),
|
|
71
|
-
),
|
|
72
|
-
)
|
|
73
|
-
.then((renderedRoutes) =>
|
|
74
|
-
Promise.all(
|
|
75
|
-
renderedRoutes.map(async (route) => {
|
|
76
|
-
const dir = join(config.prerenderToDir, route.route);
|
|
77
|
-
const file = join(dir, "index.html");
|
|
78
|
-
await mkdir(dir, { recursive: true });
|
|
79
|
-
await writeFile(file, route.html.trim());
|
|
80
|
-
return {
|
|
81
|
-
...route,
|
|
82
|
-
file,
|
|
83
|
-
};
|
|
84
|
-
}),
|
|
85
|
-
),
|
|
86
|
-
)
|
|
87
|
-
.catch((err) => {
|
|
88
|
-
throw err;
|
|
89
|
-
})
|
|
90
|
-
.finally(() => {
|
|
91
|
-
prerenderer.destroy();
|
|
92
|
-
});
|
|
60
|
+
const prerendered = await prerenderPages(config);
|
|
93
61
|
console.log(`${GREEN}✔${RESET} Prerendered in ${prerenderTimer()}`);
|
|
94
62
|
|
|
95
63
|
console.log();
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import {} from "node:url";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import { mkdir, writeFile } from "node:fs/promises";
|
|
4
|
+
import type { Browser } from "puppeteer";
|
|
5
|
+
import { createServer } from "./dev-server";
|
|
6
|
+
import type { ResolvedConfig } from "./config";
|
|
7
|
+
|
|
8
|
+
export type PrerenderedRoute = {
|
|
9
|
+
route: string;
|
|
10
|
+
file: string;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
export async function prerenderPages(
|
|
14
|
+
config: ResolvedConfig,
|
|
15
|
+
): Promise<PrerenderedRoute[]> {
|
|
16
|
+
const puppeteer = await import("puppeteer");
|
|
17
|
+
const timeout = config.prerenderer.timeout ?? 30e3;
|
|
18
|
+
|
|
19
|
+
const server = await createServer(config);
|
|
20
|
+
await server.listen();
|
|
21
|
+
|
|
22
|
+
const results: PrerenderedRoute[] = [];
|
|
23
|
+
let browser: Browser | undefined;
|
|
24
|
+
|
|
25
|
+
try {
|
|
26
|
+
browser = await puppeteer.launch({
|
|
27
|
+
headless: false,
|
|
28
|
+
timeout,
|
|
29
|
+
});
|
|
30
|
+
for (const route of config.prerenderedRoutes) {
|
|
31
|
+
const url = new URL(route, `http://localhost:${config.appPort}`);
|
|
32
|
+
const page = await browser.newPage();
|
|
33
|
+
page.setDefaultTimeout(timeout);
|
|
34
|
+
page.setDefaultNavigationTimeout(timeout);
|
|
35
|
+
await page.goto(url.href);
|
|
36
|
+
if (config.prerenderer.waitForSelector) {
|
|
37
|
+
await page.waitForSelector(config.prerenderer.waitForSelector, {
|
|
38
|
+
visible: config.prerenderer.waitForSelectorVisible,
|
|
39
|
+
});
|
|
40
|
+
} else if (config.prerenderer.waitForTimeout != null) {
|
|
41
|
+
await new Promise((res) =>
|
|
42
|
+
setTimeout(res, config.prerenderer.waitForTimeout),
|
|
43
|
+
);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
const html = await page.content();
|
|
47
|
+
await page.close();
|
|
48
|
+
|
|
49
|
+
const dir = join(config.appOutDir, route.substring(1));
|
|
50
|
+
const file = join(dir, "index.html");
|
|
51
|
+
await mkdir(dir, { recursive: true });
|
|
52
|
+
await writeFile(file, html);
|
|
53
|
+
results.push({
|
|
54
|
+
file,
|
|
55
|
+
route,
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
} finally {
|
|
59
|
+
await Promise.allSettled([server.close(), browser?.close()]);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
return results;
|
|
63
|
+
}
|