208 lines
5.7 KiB
JavaScript
208 lines
5.7 KiB
JavaScript
|
/* eslint-disable import/no-extraneous-dependencies */
|
||
|
import fs from "fs";
|
||
|
import { join } from "path";
|
||
|
import { URL, fileURLToPath } from "url";
|
||
|
import { minify } from "terser";
|
||
|
import { babel, presetTypescript } from "$repo-utils/babel-top-level";
|
||
|
import { IS_BABEL_8 } from "$repo-utils";
|
||
|
import { gzipSync } from "zlib";
|
||
|
|
||
|
import {
|
||
|
getHelperMetadata,
|
||
|
stringifyMetadata,
|
||
|
} from "./build-helper-metadata.js";
|
||
|
|
||
|
const HELPERS_FOLDER = new URL("../src/helpers", import.meta.url);
|
||
|
const IGNORED_FILES = new Set(["package.json", "tsconfig.json"]);
|
||
|
|
||
|
export default async function generateHelpers() {
|
||
|
let output = `/*
|
||
|
* This file is auto-generated! Do not modify it directly.
|
||
|
* To re-generate run 'yarn gulp generate-runtime-helpers'
|
||
|
*/
|
||
|
|
||
|
import template from "@babel/template";
|
||
|
import type * as t from "@babel/types";
|
||
|
|
||
|
interface Helper {
|
||
|
minVersion: string;
|
||
|
ast: () => t.Program;
|
||
|
metadata: HelperMetadata;
|
||
|
}
|
||
|
|
||
|
export interface HelperMetadata {
|
||
|
globals: string[];
|
||
|
locals: { [name: string]: string[] };
|
||
|
dependencies: { [name: string]: string[] };
|
||
|
exportBindingAssignments: string[];
|
||
|
exportName: string;
|
||
|
}
|
||
|
|
||
|
function helper(minVersion: string, source: string, metadata: HelperMetadata): Helper {
|
||
|
return Object.freeze({
|
||
|
minVersion,
|
||
|
ast: () => template.program.ast(source, { preserveComments: true }),
|
||
|
metadata,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
export { helpers as default };
|
||
|
const helpers: Record<string, Helper> = {
|
||
|
__proto__: null,
|
||
|
`;
|
||
|
|
||
|
let babel7extraOutput = "";
|
||
|
|
||
|
for (const file of (await fs.promises.readdir(HELPERS_FOLDER)).sort()) {
|
||
|
if (IGNORED_FILES.has(file)) continue;
|
||
|
if (file.startsWith(".")) continue; // ignore e.g. vim swap files
|
||
|
|
||
|
const [helperName] = file.split(".");
|
||
|
|
||
|
const isTs = file.endsWith(".ts");
|
||
|
|
||
|
const filePath = join(fileURLToPath(HELPERS_FOLDER), file);
|
||
|
if (!file.endsWith(".js") && !isTs) {
|
||
|
console.error("ignoring", filePath);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
let code = await fs.promises.readFile(filePath, "utf8");
|
||
|
const minVersionMatch = code.match(
|
||
|
/^\s*\/\*\s*@minVersion\s+(?<minVersion>\S+)\s*\*\/\s*$/m
|
||
|
);
|
||
|
if (!minVersionMatch) {
|
||
|
throw new Error(`@minVersion number missing in ${filePath}`);
|
||
|
}
|
||
|
const { minVersion } = minVersionMatch.groups;
|
||
|
|
||
|
const onlyBabel7 = code.includes("@onlyBabel7");
|
||
|
const mangleFns = code.includes("@mangleFns");
|
||
|
const noMangleFns = [];
|
||
|
|
||
|
code = babel.transformSync(code, {
|
||
|
configFile: false,
|
||
|
babelrc: false,
|
||
|
filename: filePath,
|
||
|
presets: [
|
||
|
[
|
||
|
presetTypescript,
|
||
|
{
|
||
|
onlyRemoveTypeImports: true,
|
||
|
optimizeConstEnums: true,
|
||
|
},
|
||
|
],
|
||
|
],
|
||
|
plugins: [
|
||
|
/**
|
||
|
* @type {import("@babel/core").PluginObj}
|
||
|
*/
|
||
|
({ types: t }) => ({
|
||
|
// These pre/post hooks are needed because the TS transform is,
|
||
|
// when building in the old Babel e2e test, removing the
|
||
|
// `export { OverloadYield as default }` in the OverloadYield helper.
|
||
|
// TODO: Remove in Babel 8.
|
||
|
pre(file) {
|
||
|
if (!process.env.IS_BABEL_OLD_E2E) return;
|
||
|
file.metadata.exportName = null;
|
||
|
file.path.traverse({
|
||
|
ExportSpecifier(path) {
|
||
|
if (path.node.exported.name === "default") {
|
||
|
file.metadata.exportName = path.node.local.name;
|
||
|
}
|
||
|
},
|
||
|
});
|
||
|
},
|
||
|
post(file) {
|
||
|
if (!process.env.IS_BABEL_OLD_E2E) return;
|
||
|
if (!file.metadata.exportName) return;
|
||
|
file.path.traverse({
|
||
|
ExportNamedDeclaration(path) {
|
||
|
if (
|
||
|
!path.node.declaration &&
|
||
|
path.node.specifiers.length === 0
|
||
|
) {
|
||
|
path.node.specifiers.push(
|
||
|
t.exportSpecifier(
|
||
|
t.identifier(file.metadata.exportName),
|
||
|
t.identifier("default")
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
},
|
||
|
});
|
||
|
},
|
||
|
visitor: {
|
||
|
ImportDeclaration(path) {
|
||
|
const source = path.node.source;
|
||
|
source.value = source.value
|
||
|
.replace(/\.ts$/, "")
|
||
|
.replace(/^\.\//, "");
|
||
|
},
|
||
|
FunctionDeclaration(path) {
|
||
|
if (
|
||
|
mangleFns &&
|
||
|
path.node.leadingComments?.find(c =>
|
||
|
c.value.includes("@no-mangle")
|
||
|
)
|
||
|
) {
|
||
|
const name = path.node.id.name;
|
||
|
if (name) noMangleFns.push(name);
|
||
|
}
|
||
|
},
|
||
|
},
|
||
|
}),
|
||
|
],
|
||
|
}).code;
|
||
|
code = (
|
||
|
await minify(code, {
|
||
|
ecma: 5,
|
||
|
mangle: {
|
||
|
keep_fnames: mangleFns ? new RegExp(noMangleFns.join("|")) : true,
|
||
|
},
|
||
|
// The _typeof helper has a custom directive that we must keep
|
||
|
compress: {
|
||
|
directives: false,
|
||
|
passes: 10,
|
||
|
unsafe: true,
|
||
|
unsafe_proto: true,
|
||
|
},
|
||
|
})
|
||
|
).code;
|
||
|
|
||
|
let metadata;
|
||
|
// eslint-disable-next-line prefer-const
|
||
|
[code, metadata] = getHelperMetadata(babel, code, helperName);
|
||
|
|
||
|
const helperStr = `\
|
||
|
// size: ${code.length}, gzip size: ${gzipSync(code).length}
|
||
|
${JSON.stringify(helperName)}: helper(
|
||
|
${JSON.stringify(minVersion)},
|
||
|
${JSON.stringify(code)},
|
||
|
${stringifyMetadata(metadata)}
|
||
|
),
|
||
|
`;
|
||
|
|
||
|
if (onlyBabel7) {
|
||
|
if (!IS_BABEL_8()) babel7extraOutput += helperStr;
|
||
|
} else {
|
||
|
output += helperStr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
output += "};";
|
||
|
|
||
|
if (babel7extraOutput) {
|
||
|
output += `
|
||
|
|
||
|
if (!process.env.BABEL_8_BREAKING) {
|
||
|
Object.assign(helpers, {
|
||
|
${babel7extraOutput}
|
||
|
});
|
||
|
}
|
||
|
`;
|
||
|
}
|
||
|
|
||
|
return output;
|
||
|
}
|