-
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
-
90
-
91
-
92
-
93
-
94
-
95
-
96
-
97
-
98
-
99
-
100
-
101
-
102
-
103
-
104
-
105
-
106
-
107
-
108
-
109
-
110
-
111
-
112
-
113
-
114
-
115
-
116
-
117
-
118
-
119
-
120
-
121
-
122
-
123
-
124
-
125
-
126
-
127
-
128
-
129
-
130
-
131
-
132
-
133
-
134
-
135
-
136
-
137
-
138
-
139
-
140
-
141
-
142
-
143
-
144
-
145
-
146
-
147
-
148
-
149
-
150
-
151
-
152
-
153
-
154
-
155
-
156
-
157
-
158
-
159
-
160
-
161
-
162
-
163
-
164
-
165
-
166
-
167
-
168
-
169
-
170
-
171
-
172
-
173
-
174
-
175
-
176
//! Rollup plugin for Gleam language.
//
//! SPDX-FileCopyrightText: 2024 Shota FUJI <pockawoooh@gmail.com>
//! SPDX-License-Identifier: Apache-2.0
import { execFile } from "node:child_process";
import { readFile } from "node:fs/promises";
import * as path from "node:path";
import { promisify } from "node:util";
import { fileURLToPath } from "node:url";
import * as toml from "smol-toml";
import readdirp from "readdirp";
import type { Plugin } from "rollup";
interface GleamTOML {
name: string;
version: string;
}
function isGleamTOML(x: unknown): x is GleamTOML {
if (typeof x !== "object" || !x) {
return false;
}
if (!("name" in x && typeof x.name === "string" && x.name)) {
return false;
}
if (!("version" in x && typeof x.version === "string" && x.version)) {
return false;
}
return true;
}
export interface GleamPluginOptions {
/**
* Path to `gleam.toml` file.
* This plugin looks for source directory and build directory based on this file location.
*
* Current working directory will be used to resolve relative paths.
*
* @default "./gleam.toml"
*/
gleamToml?: URL | string;
/**
* Path to or name of the Gleam binary file to invoke.
* @default "gleam"
*/
bin?: string;
buildOptions?: {
/**
* Emit compile time warnings as errors.
* Enables `--warnings-as-errors` CLI flag.
*/
warningsAsErrors?: boolean;
};
}
export function gleam({
bin = "gleam",
buildOptions = {},
gleamToml: gleamTomlPathOrUrl = "./gleam.toml",
}: GleamPluginOptions = {}): Plugin {
/**
* Parsed contents of `gleam.toml`.
*/
let gleamToml: GleamTOML | null = null;
const projectRoot =
typeof gleamTomlPathOrUrl === "string"
? path.resolve(gleamTomlPathOrUrl, "..")
: fileURLToPath(new URL("./", gleamTomlPathOrUrl));
// Gleam expects a project to have `src/` directory at project root.
const srcDir = path.resolve(projectRoot, "src");
// Gleam compiler outputs artifacts under `build/` directory at project root.
// Directory structure inside is not documentated, but this is the only way
// to access built JS files. There is no way to specify output directory also.
const jsOutDir = path.resolve(projectRoot, "build/dev/javascript");
const buildCommandArgs = ["build", "--target", "javascript"];
if (buildOptions.warningsAsErrors) {
buildCommandArgs.push("--warnings-as-errors");
}
// Build command won't change during the plugin's lifetime.
// It's fine to bind everything upfront.
const buildProject = promisify(execFile).bind(null, bin, buildCommandArgs, {
cwd: projectRoot,
});
return {
name: "gleam",
async buildStart() {
// Changes to `gleam.toml` should trigger rerun of this hook.
// Otherwise, if `name` field got changed for example, Rollup tries to access nonexistent
// files based on an old name (build/dev/javascript/old_name/foo.mjs).
this.addWatchFile(
typeof gleamTomlPathOrUrl === "string"
? gleamTomlPathOrUrl
: fileURLToPath(gleamTomlPathOrUrl),
);
const parsed = toml.parse(await readFile(gleamTomlPathOrUrl, { encoding: "utf8" }));
if (!isGleamTOML(parsed)) {
// TypeScript can't narrow types using `never`. Putting `return` after this line
// triggers `Unreachable code detected.` so we have to *return never*.
// <https://github.com/microsoft/TypeScript/issues/12825>
// Following code contains the same workaround for this reason.
return this.error(`gleam.toml does not comform to official schema.`);
}
gleamToml = parsed;
},
async transform(_code, id) {
// .gleam files imported by non-Gleam modules (e.g. .js, .ts) run through this branch.
// This branch triggers a build then returns proxy code that re-exports everything from
// the generated .mjs file.
if (id.endsWith(".gleam")) {
if (!gleamToml) {
return this.error(
"Unable to resolve transpiled Gleam file without `gleam.toml`.",
);
}
const absPath = path.resolve(srcDir, id);
if (!absPath.startsWith(srcDir)) {
this.error("Gleam files must be inside the src/ directory.");
}
const modulePath = absPath
// `+1` ... removing path separator
.slice(srcDir.length + 1)
.replace(/\.gleam$/, ".mjs");
const transpiledMjsPath = path.resolve(jsOutDir, gleamToml.name, modulePath);
// Scan every .gleam files and watch them. This might be slow when the number
// of files got large. However, manually watching comes with performance cost
// too and it also brings management cost (properly closing watcher, reducing
// the number of watch targets). Since this logic doesn't need file contents,
// I believe the performance cost it brings is tolerable.
for await (const entry of readdirp(srcDir)) {
const resolved = path.resolve(srcDir, entry.path);
if (!resolved.endsWith(".gleam")) {
continue;
}
// The .gleam file pointed by `id` is already (or will be) in watched files.
// Adding this results in duplicated watched file.
if (resolved === id) {
continue;
}
this.addWatchFile(resolved);
}
// Build after starting watching other .gleam files.
// Otherwise fixing an error in another file does not trigger rebuild, which
// leaves bundler stucked in an error state.
await buildProject();
return {
code: `export * from ${JSON.stringify(transpiledMjsPath)}`,
};
}
},
};
}
export default gleam;