1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
|
import Fs from "node:fs/promises";
import Path from "node:path";
import { execSync } from "node:child_process";
async function generateBootstrapFile(output, content) {
let previousContent = undefined;
try {
previousContent = await Fs.readFile(output, "utf8");
} catch (e) {
if (e.code !== "ENOENT") {
throw e;
}
}
const contentHasChanged = previousContent !== content;
if (contentHasChanged) {
await Fs.writeFile(output, content, "utf8");
}
}
export function plugin(config) {
return {
name: "extract-client-components",
setup(build) {
if (
config.bootstrapOutput &&
typeof config.bootstrapOutput !== "string"
) {
console.error("bootstrapOutput must be a string");
return;
}
const bootstrapOutput = config.bootstrapOutput || "./bootstrap.js";
if (!config.target) {
console.error("target is required");
return;
}
if (typeof config.target !== "string") {
console.error("target must be a string");
return;
}
build.onStart(async () => {
try {
/* TODO: Make sure `server_reason_react.extract_client_components` is available in $PATH */
const bootstrapContent = execSync(
`server_reason_react.extract_client_components ${config.target}`,
{ encoding: "utf8" },
);
await generateBootstrapFile(bootstrapOutput, bootstrapContent);
} catch (e) {
console.log("Extraction of client components failed:");
console.error(e);
return;
}
});
build.onResolve({ filter: /.*/ }, (args) => {
const isEntryPoint = args.kind === "entry-point";
if (isEntryPoint) {
return {
path: args.path,
namespace: "entrypoint",
};
}
return null;
});
build.onLoad({ filter: /.*/, namespace: "entrypoint" }, async (args) => {
const filePath = args.path.replace(/^entrypoint:/, "");
const entryPointContents = await Fs.readFile(filePath, "utf8");
const relativeBootstrapOutput = Path.relative(
Path.dirname(filePath),
bootstrapOutput,
);
const contents = `
require("./${relativeBootstrapOutput}");
${entryPointContents}`;
return {
loader: "jsx",
contents,
resolveDir: Path.dirname(Path.resolve(process.cwd(), filePath)),
};
});
},
};
}
|