| Index: webpack_runner.js |
| diff --git a/webpack_runner.js b/webpack_runner.js |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..a3b5b9462b5123c16e1b7ff379efdd6b0c77b559 |
| --- /dev/null |
| +++ b/webpack_runner.js |
| @@ -0,0 +1,152 @@ |
| +/* |
| + * This file is part of Adblock Plus <https://adblockplus.org/>, |
| + * Copyright (C) 2006-present eyeo GmbH |
| + * |
| + * Adblock Plus is free software: you can redistribute it and/or modify |
| + * it under the terms of the GNU General Public License version 3 as |
| + * published by the Free Software Foundation. |
| + * |
| + * Adblock Plus is distributed in the hope that it will be useful, |
| + * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| + * GNU General Public License for more details. |
| + * |
| + * You should have received a copy of the GNU General Public License |
| + * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. |
| + */ |
| + |
| +"use strict"; |
| + |
| +const path = require("path"); |
| +const process = require("process"); |
| + |
| +const MemoryFS = require("memory-fs"); |
| +const webpack = require("webpack"); |
| + |
| +// We read the configuration from STDIN rather than as an argument to improve |
| +// the output on error. Otherwise the (fairly huge) configuration is printed |
| +// along with the actual error message. |
| +let inputChunks = []; |
| +process.stdin.setEncoding("utf-8"); |
| +process.stdin.on("data", chunk => { inputChunks.push(chunk); }); |
| +process.stdin.on("end", () => |
| +{ |
| + let {bundles, extension_path, |
| + info_module, resolve_paths} = JSON.parse(inputChunks.join("")); |
| + |
| + // The contents of the info module is passed to us as a string from the Python |
| + // packager and we pass it through to our custom loader now so it is available |
| + // at bundle time. |
| + require("./info-loader").setInfoModule(info_module); |
| + |
| + // Since the cost of starting Node.js and loading all the modules is hugely |
| + // larger than actually producing bundles we avoid paying it multiple times, |
| + // instead producing all the bundles in one go. |
| + let options = []; |
| + for (let {bundle_name, entry_points} of bundles) |
| + { |
| + options.push({ |
| + context: extension_path, |
| + devtool: "source-map", |
| + module: { |
| + rules: [{ |
| + include: path.join(__dirname, "info-loader.js"), |
| + use: ["info-loader"] |
| + }] |
| + }, |
| + entry: entry_points, |
| + output: { |
| + path: "/", |
| + filename: bundle_name |
| + }, |
| + resolve: { |
| + modules: resolve_paths, |
| + alias: { |
| + // To use our custom loader for the info module we must first set up |
| + // an alias to a file that actually exists. We use the loader's file |
| + // itself so it's self contained, but any existing file path would |
| + // work. |
| + info$: path.join(__dirname, "info-loader.js"), |
| + // Prevent builtin Node.js modules from being used instead of our own |
| + // when the names clash. Once relative paths are used this won't be |
| + // necessary. |
| + url$: "url.js", |
| + events$: "events.js", |
| + punycode$: "punycode.js" |
| + }, |
| + plugins: [ |
| + function() |
| + { |
| + // Our old module system in packagerChrome.py used to prefix |
| + // module names with the name of their parent directory and an |
| + // underscore - but only when that directory wasn't called |
| + // "lib". This plugin is for backwards compatability, but can |
| + // be removed once use of that deprecated syntax has been |
| + // replaced. |
| + this.plugin("described-resolve", (request, callback) => |
| + { |
| + let target = request.request; |
| + |
| + let prefixIndex = target.indexOf("_"); |
| + if (prefixIndex == -1) |
| + return callback(); |
| + |
| + let prefix = target.substring(0, prefixIndex); |
| + if (prefix == "lib") |
| + return callback(); |
| + |
| + let fixedTarget = path.join(prefix, |
| + target.substring(prefixIndex + 1)); |
| + return this.doResolve( |
| + "resolve", Object.assign({}, request, {request: fixedTarget}), |
| + "Changed prefixed path using legacy buildtools syntax from " + |
| + target + " to " + fixedTarget, |
| + callback |
| + ); |
| + }); |
| + } |
| + ] |
| + }, |
| + resolveLoader: { |
| + modules: [path.resolve(__dirname)] |
| + } |
| + }); |
| + } |
| + |
| + // Based on this example |
| + // https://webpack.js.org/api/node/#custom-file-systems |
| + let memoryFS = new MemoryFS(); |
| + let webpackCompiler = webpack(options); |
| + |
| + webpackCompiler.outputFileSystem = memoryFS; |
| + webpackCompiler.run((err, stats) => |
| + { |
| + // Error handling is based on this example |
| + // https://webpack.js.org/api/node/#error-handling |
| + if (err) |
| + { |
| + let reason = err.stack || err; |
| + if (err.details) |
| + reason += "\n" + err.details; |
| + throw new Error(reason); |
| + } |
| + else if (stats.hasErrors()) |
| + throw new Error(stats.toJson().errors.join("\n")); |
| + else |
| + { |
| + let files = {}; |
| + |
| + for (let config of options) |
| + { |
| + let filepath = path.join(config.output.path, config.output.filename); |
| + let mappath = filepath + ".map"; |
| + files[filepath] = memoryFS.readFileSync(filepath, "utf-8"); |
| + files[mappath] = memoryFS.readFileSync(mappath, "utf-8"); |
| + memoryFS.unlinkSync(filepath); |
| + memoryFS.unlinkSync(mappath); |
| + } |
| + |
| + console.log(JSON.stringify(files)); |
| + } |
| + }); |
| +}); |