@selfage/bundler_cli
Version:
CLI for bundling and running bundled frontend or backend TypeScript files.
152 lines • 22.3 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.bundleForNode = bundleForNode;
exports.bundleForNodeReturnAssetFiles = bundleForNodeReturnAssetFiles;
exports.bundleForBrowser = bundleForBrowser;
exports.bundleForBrowserReturnAssetFiles = bundleForBrowserReturnAssetFiles;
exports.bundle = bundle;
const browserifyConstructor = require("browserify");
const fs = require("fs");
const path = require("path");
const stream = require("stream");
const UglifyJS = require("uglify-js");
const file_extension_stripper_1 = require("./file_extension_stripper");
const files_copier_1 = require("./files_copier");
const get_stream_1 = require("./get_stream");
const compiler_1 = require("@selfage/cli/build/compiler");
// Export as a commonjs module.
let TEMP_DECLARATION_FILE_CONTENT_TEMPLATE = `
declare module "*{ext}" {
let path: string;
export = path;
}
`;
let ASSET_EXT_DECLARATION_FILE = `selfage_asset_exts_declaration.d.ts`;
async function bundleForNode(sourceFile, outputFile, fromDir = ".", toDir = fromDir, options) {
let assetFiles = await bundleForNodeReturnAssetFiles(sourceFile, outputFile, options);
if (path.posix.normalize(fromDir) === path.posix.normalize(toDir)) {
return;
}
await (0, files_copier_1.copyFilesToDir)([(0, file_extension_stripper_1.stripFileExtension)(outputFile) + ".js", ...assetFiles], fromDir, toDir);
}
async function bundleForNodeReturnAssetFiles(sourceFile, outputFile, options) {
let baseDir = path.posix.dirname(outputFile);
return await bundle(sourceFile, outputFile, baseDir, true, false, options);
}
async function bundleForBrowser(sourceFile, outputFile, baseDir = ".", outDir = baseDir, options) {
let assetFiles = await bundleForBrowserReturnAssetFiles(sourceFile, outputFile, baseDir, options);
if (path.posix.normalize(baseDir) === path.posix.normalize(outDir)) {
return;
}
await (0, files_copier_1.copyFilesToDir)([(0, file_extension_stripper_1.stripFileExtension)(outputFile) + ".js", ...assetFiles], baseDir, outDir);
}
async function bundleForBrowserReturnAssetFiles(sourceFile, outputFile, baseDir = ".", options) {
return bundle(sourceFile, outputFile, baseDir, false, true, options);
}
async function bundle(sourceFile, outputFile, baseDir, inNode, bundleExternal, options = {}) {
let assetExts;
if (options.assetExts) {
assetExts = options.assetExts;
}
else {
let packageJsonFile = options.packageJsonFile ?? "./package.json";
assetExts = JSON.parse((await fs.promises.readFile(packageJsonFile)).toString()).assetExts;
}
if (!assetExts) {
await (0, compiler_1.compile)(sourceFile, options.tsconfigFile, options.extraFiles);
}
else {
await compileWithAssets(sourceFile, assetExts, options.extraFiles, options.tsconfigFile);
}
try {
await fs.promises.stat(baseDir);
}
catch (e) {
e.message = "Base directory needs to be a valid directory. " + e.message;
throw e;
}
let entriesToBeBrowserified = new Array();
if (options.inlineJs) {
entriesToBeBrowserified.push(...options.inlineJs.map((jsCode) => {
return stream.Readable.from([jsCode]);
}));
}
if (options.extraFiles) {
// If exists, put them before the main file such that variables will be
// defined and functions will be called before executing the main file.
for (let extraFile of options.extraFiles) {
entriesToBeBrowserified.push(path.posix.relative(baseDir, (0, file_extension_stripper_1.stripFileExtension)(extraFile) + ".js"));
}
}
entriesToBeBrowserified.push(path.posix.relative(baseDir, (0, file_extension_stripper_1.stripFileExtension)(sourceFile) + ".js"));
let browserifyHandler = browserifyConstructor(entriesToBeBrowserified, {
node: inNode,
bundleExternal: bundleExternal,
basedir: baseDir,
debug: options.debug,
});
let outputAssetFiles = new Array();
browserifyHandler.transform((file) => {
if (assetExts && assetExts.includes(path.posix.extname(file))) {
return new AssetTransformer(file, outputAssetFiles);
}
else {
return new stream.PassThrough();
}
});
let bundledCode = await (0, get_stream_1.getStream)(browserifyHandler.bundle());
let outputCode;
if (options.skipMinify) {
outputCode = bundledCode;
}
else {
let minifyOptions = {};
if (options.debug) {
minifyOptions.sourceMap = {
content: "inline",
includeSources: true,
url: "inline",
};
}
let minifiedRes = UglifyJS.minify(bundledCode, minifyOptions);
if (minifiedRes.error) {
throw minifiedRes.error;
}
outputCode = minifiedRes.code;
}
await fs.promises.mkdir(path.posix.dirname(outputFile), {
recursive: true,
});
await fs.promises.writeFile((0, file_extension_stripper_1.stripFileExtension)(outputFile) + ".js", outputCode);
return outputAssetFiles;
}
async function compileWithAssets(sourceFile, assetExts, extraFiles = [], tsconfigFile) {
let tempFileContent = new Array();
for (let ext of assetExts) {
tempFileContent.push(TEMP_DECLARATION_FILE_CONTENT_TEMPLATE.replace("{ext}", ext));
}
await fs.promises.writeFile(ASSET_EXT_DECLARATION_FILE, tempFileContent.join(""));
await (0, compiler_1.compile)(sourceFile, tsconfigFile, [
ASSET_EXT_DECLARATION_FILE,
...extraFiles,
]);
}
class AssetTransformer extends stream.Transform {
constructor(absoluteFile, outputAssetFilePaths) {
super({
transform: (chunk, encoding, callback) => {
callback();
},
flush: (callback) => {
let relativePath = path.posix.relative(".", this.absoluteFile);
this.outputAssetFilePaths.push(relativePath);
callback(undefined,
// __filename will be transformed later by Browserify.
`module.exports = __filename;`);
},
});
this.absoluteFile = absoluteFile;
this.outputAssetFilePaths = outputAssetFilePaths;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundler.js","sourceRoot":"","sources":["bundler.ts"],"names":[],"mappings":"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","sourcesContent":["import browserifyConstructor = require(\"browserify\");\nimport fs = require(\"fs\");\nimport path = require(\"path\");\nimport stream = require(\"stream\");\nimport UglifyJS = require(\"uglify-js\");\nimport { stripFileExtension } from \"./file_extension_stripper\";\nimport { copyFilesToDir } from \"./files_copier\";\nimport { getStream } from \"./get_stream\";\nimport { compile } from \"@selfage/cli/build/compiler\";\n\n// Export as a commonjs module.\nlet TEMP_DECLARATION_FILE_CONTENT_TEMPLATE = `\ndeclare module \"*{ext}\" {\n   let path: string;\n   export = path;\n}\n`;\n\nlet ASSET_EXT_DECLARATION_FILE = `selfage_asset_exts_declaration.d.ts`;\n\nexport interface CommonBundleOptions {\n  tsconfigFile?: string;\n  extraFiles?: Array<string>;\n  inlineJs?: Array<string>;\n  assetExts?: Array<string>;\n  packageJsonFile?: string;\n  skipMinify?: boolean;\n  debug?: boolean;\n}\n\nexport async function bundleForNode(\n  sourceFile: string,\n  outputFile: string,\n  fromDir = \".\",\n  toDir = fromDir,\n  options?: CommonBundleOptions,\n): Promise<void> {\n  let assetFiles = await bundleForNodeReturnAssetFiles(\n    sourceFile,\n    outputFile,\n    options,\n  );\n\n  if (path.posix.normalize(fromDir) === path.posix.normalize(toDir)) {\n    return;\n  }\n  await copyFilesToDir(\n    [stripFileExtension(outputFile) + \".js\", ...assetFiles],\n    fromDir,\n    toDir,\n  );\n}\n\nexport async function bundleForNodeReturnAssetFiles(\n  sourceFile: string,\n  outputFile: string,\n  options?: CommonBundleOptions,\n): Promise<Array<string>> {\n  let baseDir = path.posix.dirname(outputFile);\n  return await bundle(sourceFile, outputFile, baseDir, true, false, options);\n}\n\nexport async function bundleForBrowser(\n  sourceFile: string,\n  outputFile: string,\n  baseDir = \".\",\n  outDir = baseDir,\n  options?: CommonBundleOptions,\n): Promise<void> {\n  let assetFiles = await bundleForBrowserReturnAssetFiles(\n    sourceFile,\n    outputFile,\n    baseDir,\n    options,\n  );\n\n  if (path.posix.normalize(baseDir) === path.posix.normalize(outDir)) {\n    return;\n  }\n  await copyFilesToDir(\n    [stripFileExtension(outputFile) + \".js\", ...assetFiles],\n    baseDir,\n    outDir,\n  );\n}\n\nexport async function bundleForBrowserReturnAssetFiles(\n  sourceFile: string,\n  outputFile: string,\n  baseDir = \".\",\n  options?: CommonBundleOptions,\n): Promise<Array<string>> {\n  return bundle(sourceFile, outputFile, baseDir, false, true, options);\n}\n\nexport async function bundle(\n  sourceFile: string,\n  outputFile: string,\n  baseDir: string,\n  inNode: boolean,\n  bundleExternal: boolean,\n  options: CommonBundleOptions = {},\n): Promise<Array<string>> {\n  let assetExts: Array<string>;\n  if (options.assetExts) {\n    assetExts = options.assetExts;\n  } else {\n    let packageJsonFile = options.packageJsonFile ?? \"./package.json\";\n    assetExts = JSON.parse(\n      (await fs.promises.readFile(packageJsonFile)).toString(),\n    ).assetExts;\n  }\n\n  if (!assetExts) {\n    await compile(sourceFile, options.tsconfigFile, options.extraFiles);\n  } else {\n    await compileWithAssets(\n      sourceFile,\n      assetExts,\n      options.extraFiles,\n      options.tsconfigFile,\n    );\n  }\n\n  try {\n    await fs.promises.stat(baseDir);\n  } catch (e) {\n    e.message = \"Base directory needs to be a valid directory. \" + e.message;\n    throw e;\n  }\n  let entriesToBeBrowserified = new Array<string | stream.Readable>();\n  if (options.inlineJs) {\n    entriesToBeBrowserified.push(\n      ...options.inlineJs.map((jsCode) => {\n        return stream.Readable.from([jsCode]);\n      }),\n    );\n  }\n  if (options.extraFiles) {\n    // If exists, put them before the main file such that variables will be\n    // defined and functions will be called before executing the main file.\n    for (let extraFile of options.extraFiles) {\n      entriesToBeBrowserified.push(\n        path.posix.relative(baseDir, stripFileExtension(extraFile) + \".js\"),\n      );\n    }\n  }\n  entriesToBeBrowserified.push(\n    path.posix.relative(baseDir, stripFileExtension(sourceFile) + \".js\"),\n  );\n  let browserifyHandler = browserifyConstructor(entriesToBeBrowserified, {\n    node: inNode,\n    bundleExternal: bundleExternal,\n    basedir: baseDir,\n    debug: options.debug,\n  });\n  let outputAssetFiles = new Array<string>();\n  browserifyHandler.transform((file) => {\n    if (assetExts && assetExts.includes(path.posix.extname(file))) {\n      return new AssetTransformer(file, outputAssetFiles);\n    } else {\n      return new stream.PassThrough();\n    }\n  });\n  let bundledCode = await getStream(browserifyHandler.bundle());\n\n  let outputCode: string;\n  if (options.skipMinify) {\n    outputCode = bundledCode;\n  } else {\n    let minifyOptions: UglifyJS.MinifyOptions = {};\n    if (options.debug) {\n      minifyOptions.sourceMap = {\n        content: \"inline\",\n        includeSources: true,\n        url: \"inline\",\n      };\n    }\n    let minifiedRes = UglifyJS.minify(bundledCode, minifyOptions);\n    if (minifiedRes.error) {\n      throw minifiedRes.error;\n    }\n    outputCode = minifiedRes.code;\n  }\n\n  await fs.promises.mkdir(path.posix.dirname(outputFile), {\n    recursive: true,\n  });\n  await fs.promises.writeFile(\n    stripFileExtension(outputFile) + \".js\",\n    outputCode,\n  );\n  return outputAssetFiles;\n}\n\nasync function compileWithAssets(\n  sourceFile: string,\n  assetExts: Array<string>,\n  extraFiles: Array<string> = [],\n  tsconfigFile?: string,\n): Promise<void> {\n  let tempFileContent = new Array<string>();\n  for (let ext of assetExts) {\n    tempFileContent.push(\n      TEMP_DECLARATION_FILE_CONTENT_TEMPLATE.replace(\"{ext}\", ext),\n    );\n  }\n  await fs.promises.writeFile(\n    ASSET_EXT_DECLARATION_FILE,\n    tempFileContent.join(\"\"),\n  );\n  await compile(sourceFile, tsconfigFile, [\n    ASSET_EXT_DECLARATION_FILE,\n    ...extraFiles,\n  ]);\n}\n\nclass AssetTransformer extends stream.Transform {\n  public constructor(\n    private absoluteFile: string,\n    private outputAssetFilePaths: Array<string>,\n  ) {\n    super({\n      transform: (chunk, encoding, callback) => {\n        callback();\n      },\n      flush: (callback) => {\n        let relativePath = path.posix.relative(\".\", this.absoluteFile);\n        this.outputAssetFilePaths.push(relativePath);\n        callback(\n          undefined,\n          // __filename will be transformed later by Browserify.\n          `module.exports = __filename;`,\n        );\n      },\n    });\n  }\n}\n"]}