UNPKG

@selfage/bundler_cli

Version:

CLI for bundling and running bundled frontend or backend TypeScript files.

149 lines 21.9 kB
"use strict"; 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.copyFiles)([(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.copyFiles)([(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.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 { copyFiles } 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 copyFiles(\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 copyFiles(\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.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"]}