2018-03-27 19:54:05 +05:30
|
|
|
const fs = require('fs');
|
|
|
|
const path = require('path');
|
|
|
|
const glob = require('glob');
|
|
|
|
const webpack = require('webpack');
|
|
|
|
const StatsWriterPlugin = require('webpack-stats-plugin').StatsWriterPlugin;
|
|
|
|
const CopyWebpackPlugin = require('copy-webpack-plugin');
|
|
|
|
const CompressionPlugin = require('compression-webpack-plugin');
|
|
|
|
const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin;
|
|
|
|
|
|
|
|
const ROOT_PATH = path.resolve(__dirname, '..');
|
|
|
|
const IS_PRODUCTION = process.env.NODE_ENV === 'production';
|
|
|
|
const IS_DEV_SERVER = process.argv.join(' ').indexOf('webpack-dev-server') !== -1;
|
|
|
|
const DEV_SERVER_HOST = process.env.DEV_SERVER_HOST || 'localhost';
|
|
|
|
const DEV_SERVER_PORT = parseInt(process.env.DEV_SERVER_PORT, 10) || 3808;
|
|
|
|
const DEV_SERVER_LIVERELOAD = process.env.DEV_SERVER_LIVERELOAD !== 'false';
|
|
|
|
const WEBPACK_REPORT = process.env.WEBPACK_REPORT;
|
|
|
|
const NO_COMPRESSION = process.env.NO_COMPRESSION;
|
|
|
|
|
|
|
|
let autoEntriesCount = 0;
|
|
|
|
let watchAutoEntries = [];
|
2018-10-15 14:42:47 +05:30
|
|
|
const defaultEntries = ['./main'];
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
function generateEntries() {
|
|
|
|
// generate automatic entry points
|
|
|
|
const autoEntries = {};
|
2018-10-15 14:42:47 +05:30
|
|
|
const autoEntriesMap = {};
|
2018-05-09 12:01:36 +05:30
|
|
|
const pageEntries = glob.sync('pages/**/index.js', {
|
|
|
|
cwd: path.join(ROOT_PATH, 'app/assets/javascripts'),
|
|
|
|
});
|
|
|
|
watchAutoEntries = [path.join(ROOT_PATH, 'app/assets/javascripts/pages/')];
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
function generateAutoEntries(path, prefix = '.') {
|
|
|
|
const chunkPath = path.replace(/\/index\.js$/, '');
|
|
|
|
const chunkName = chunkPath.replace(/\//g, '.');
|
2018-10-15 14:42:47 +05:30
|
|
|
autoEntriesMap[chunkName] = `${prefix}/${path}`;
|
2018-03-17 18:26:18 +05:30
|
|
|
}
|
|
|
|
|
2018-05-09 12:01:36 +05:30
|
|
|
pageEntries.forEach(path => generateAutoEntries(path));
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
const autoEntryKeys = Object.keys(autoEntriesMap);
|
|
|
|
autoEntriesCount = autoEntryKeys.length;
|
|
|
|
|
|
|
|
// import ancestor entrypoints within their children
|
|
|
|
autoEntryKeys.forEach(entry => {
|
|
|
|
const entryPaths = [autoEntriesMap[entry]];
|
|
|
|
const segments = entry.split('.');
|
|
|
|
while (segments.pop()) {
|
|
|
|
const ancestor = segments.join('.');
|
|
|
|
if (autoEntryKeys.includes(ancestor)) {
|
|
|
|
entryPaths.unshift(autoEntriesMap[ancestor]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
autoEntries[entry] = defaultEntries.concat(entryPaths);
|
|
|
|
});
|
2018-03-27 19:54:05 +05:30
|
|
|
|
|
|
|
const manualEntries = {
|
2018-10-15 14:42:47 +05:30
|
|
|
default: defaultEntries,
|
2018-05-09 12:01:36 +05:30
|
|
|
raven: './raven/index.js',
|
2018-03-27 19:54:05 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
return Object.assign(manualEntries, autoEntries);
|
|
|
|
}
|
|
|
|
|
|
|
|
const config = {
|
2018-10-15 14:42:47 +05:30
|
|
|
mode: IS_PRODUCTION ? 'production' : 'development',
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
context: path.join(ROOT_PATH, 'app/assets/javascripts'),
|
|
|
|
|
|
|
|
entry: generateEntries,
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
output: {
|
|
|
|
path: path.join(ROOT_PATH, 'public/assets/webpack'),
|
|
|
|
publicPath: '/assets/webpack/',
|
2018-10-15 14:42:47 +05:30
|
|
|
filename: IS_PRODUCTION ? '[name].[chunkhash:8].bundle.js' : '[name].bundle.js',
|
|
|
|
chunkFilename: IS_PRODUCTION ? '[name].[chunkhash:8].chunk.js' : '[name].chunk.js',
|
|
|
|
globalObject: 'this', // allow HMR and web workers to play nice
|
|
|
|
},
|
|
|
|
|
|
|
|
optimization: {
|
|
|
|
nodeEnv: false,
|
|
|
|
runtimeChunk: 'single',
|
|
|
|
splitChunks: {
|
|
|
|
maxInitialRequests: 4,
|
|
|
|
cacheGroups: {
|
|
|
|
default: false,
|
|
|
|
common: () => ({
|
|
|
|
priority: 20,
|
|
|
|
name: 'main',
|
|
|
|
chunks: 'initial',
|
|
|
|
minChunks: autoEntriesCount * 0.9,
|
|
|
|
}),
|
|
|
|
vendors: {
|
|
|
|
priority: 10,
|
|
|
|
chunks: 'async',
|
|
|
|
test: /[\\/](node_modules|vendor[\\/]assets[\\/]javascripts)[\\/]/,
|
|
|
|
},
|
|
|
|
commons: {
|
|
|
|
chunks: 'all',
|
|
|
|
minChunks: 2,
|
|
|
|
reuseExistingChunk: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
|
|
|
|
|
|
|
module: {
|
|
|
|
rules: [
|
|
|
|
{
|
|
|
|
test: /\.js$/,
|
|
|
|
exclude: /(node_modules|vendor\/assets)/,
|
|
|
|
loader: 'babel-loader',
|
2018-10-15 14:42:47 +05:30
|
|
|
options: {
|
|
|
|
cacheDirectory: path.join(ROOT_PATH, 'tmp/cache/babel-loader'),
|
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
test: /\.vue$/,
|
|
|
|
loader: 'vue-loader',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
test: /\.svg$/,
|
|
|
|
loader: 'raw-loader',
|
|
|
|
},
|
|
|
|
{
|
2017-09-10 17:25:29 +05:30
|
|
|
test: /\.(gif|png)$/,
|
2017-08-17 22:00:37 +05:30
|
|
|
loader: 'url-loader',
|
2017-09-10 17:25:29 +05:30
|
|
|
options: { limit: 2048 },
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
2018-03-17 18:26:18 +05:30
|
|
|
{
|
|
|
|
test: /\_worker\.js$/,
|
|
|
|
use: [
|
|
|
|
{
|
|
|
|
loader: 'worker-loader',
|
|
|
|
options: {
|
2018-10-15 14:42:47 +05:30
|
|
|
name: '[name].[hash:8].worker.js',
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
2018-03-17 18:26:18 +05:30
|
|
|
},
|
2018-10-15 14:42:47 +05:30
|
|
|
'babel-loader',
|
2018-03-17 18:26:18 +05:30
|
|
|
],
|
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
{
|
2017-09-10 17:25:29 +05:30
|
|
|
test: /\.(worker(\.min)?\.js|pdf|bmpr)$/,
|
2017-08-17 22:00:37 +05:30
|
|
|
exclude: /node_modules/,
|
|
|
|
loader: 'file-loader',
|
2017-09-10 17:25:29 +05:30
|
|
|
options: {
|
2018-10-15 14:42:47 +05:30
|
|
|
name: '[name].[hash:8].[ext]',
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
|
|
|
{
|
2018-05-09 12:01:36 +05:30
|
|
|
test: /katex.min.css$/,
|
2018-03-17 18:26:18 +05:30
|
|
|
include: /node_modules\/katex\/dist/,
|
|
|
|
use: [
|
|
|
|
{ loader: 'style-loader' },
|
2018-03-27 19:54:05 +05:30
|
|
|
{
|
2018-03-17 18:26:18 +05:30
|
|
|
loader: 'css-loader',
|
|
|
|
options: {
|
2018-10-15 14:42:47 +05:30
|
|
|
name: '[name].[hash:8].[ext]',
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
2018-03-17 18:26:18 +05:30
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
test: /\.(eot|ttf|woff|woff2)$/,
|
|
|
|
include: /node_modules\/katex\/dist\/fonts/,
|
|
|
|
loader: 'file-loader',
|
|
|
|
options: {
|
2018-10-15 14:42:47 +05:30
|
|
|
name: '[name].[hash:8].[ext]',
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
2017-09-10 17:25:29 +05:30
|
|
|
{
|
|
|
|
test: /monaco-editor\/\w+\/vs\/loader\.js$/,
|
|
|
|
use: [
|
|
|
|
{ loader: 'exports-loader', options: 'l.global' },
|
|
|
|
{ loader: 'imports-loader', options: 'l=>{},this=>l,AMDLoader=>this,module=>undefined' },
|
|
|
|
],
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
2017-09-10 17:25:29 +05:30
|
|
|
],
|
|
|
|
|
|
|
|
noParse: [/monaco-editor\/\w+\/vs\//],
|
2018-03-17 18:26:18 +05:30
|
|
|
strictExportPresence: true,
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
|
|
|
|
|
|
|
plugins: [
|
|
|
|
// manifest filename must match config.webpack.manifest_filename
|
|
|
|
// webpack-rails only needs assetsByChunkName to function properly
|
2017-09-10 17:25:29 +05:30
|
|
|
new StatsWriterPlugin({
|
|
|
|
filename: 'manifest.json',
|
|
|
|
transform: function(data, opts) {
|
2018-03-27 19:54:05 +05:30
|
|
|
const stats = opts.compiler.getStats().toJson({
|
2017-09-10 17:25:29 +05:30
|
|
|
chunkModules: false,
|
|
|
|
source: false,
|
|
|
|
chunks: false,
|
|
|
|
modules: false,
|
2018-05-09 12:01:36 +05:30
|
|
|
assets: true,
|
2017-09-10 17:25:29 +05:30
|
|
|
});
|
|
|
|
return JSON.stringify(stats, null, 2);
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
}),
|
|
|
|
|
|
|
|
// prevent pikaday from including moment.js
|
|
|
|
new webpack.IgnorePlugin(/moment/, /pikaday/),
|
|
|
|
|
|
|
|
// fix legacy jQuery plugins which depend on globals
|
|
|
|
new webpack.ProvidePlugin({
|
|
|
|
$: 'jquery',
|
|
|
|
jQuery: 'jquery',
|
|
|
|
}),
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
// copy pre-compiled vendor libraries verbatim
|
|
|
|
new CopyWebpackPlugin([
|
|
|
|
{
|
2018-05-09 12:01:36 +05:30
|
|
|
from: path.join(
|
|
|
|
ROOT_PATH,
|
|
|
|
`node_modules/monaco-editor/${IS_PRODUCTION ? 'min' : 'dev'}/vs`
|
|
|
|
),
|
2017-09-10 17:25:29 +05:30
|
|
|
to: 'monaco-editor/vs',
|
|
|
|
transform: function(content, path) {
|
2018-03-17 18:26:18 +05:30
|
|
|
if (/\.js$/.test(path) && !/worker/i.test(path) && !/typescript/i.test(path)) {
|
2017-09-10 17:25:29 +05:30
|
|
|
return (
|
|
|
|
'(function(){\n' +
|
|
|
|
'var define = this.define, require = this.require;\n' +
|
|
|
|
'window.define = define; window.require = require;\n' +
|
|
|
|
content +
|
|
|
|
'\n}.call(window.__monaco_context__ || (window.__monaco_context__ = {})));'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return content;
|
2018-05-09 12:01:36 +05:30
|
|
|
},
|
|
|
|
},
|
2017-09-10 17:25:29 +05:30
|
|
|
]),
|
2017-08-17 22:00:37 +05:30
|
|
|
],
|
|
|
|
|
|
|
|
resolve: {
|
|
|
|
extensions: ['.js'],
|
|
|
|
alias: {
|
2018-05-09 12:01:36 +05:30
|
|
|
'~': path.join(ROOT_PATH, 'app/assets/javascripts'),
|
|
|
|
emojis: path.join(ROOT_PATH, 'fixtures/emojis'),
|
|
|
|
empty_states: path.join(ROOT_PATH, 'app/views/shared/empty_states'),
|
|
|
|
icons: path.join(ROOT_PATH, 'app/views/shared/icons'),
|
|
|
|
images: path.join(ROOT_PATH, 'app/assets/images'),
|
|
|
|
vendor: path.join(ROOT_PATH, 'vendor/assets/javascripts'),
|
|
|
|
vue$: 'vue/dist/vue.esm.js',
|
|
|
|
spec: path.join(ROOT_PATH, 'spec/javascripts'),
|
|
|
|
},
|
2018-03-27 19:54:05 +05:30
|
|
|
},
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
// sqljs requires fs
|
|
|
|
node: {
|
|
|
|
fs: 'empty',
|
|
|
|
},
|
|
|
|
};
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
if (IS_PRODUCTION) {
|
|
|
|
config.devtool = 'source-map';
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
// compression can require a lot of compute time and is disabled in CI
|
2017-09-10 17:25:29 +05:30
|
|
|
if (!NO_COMPRESSION) {
|
2018-03-17 18:26:18 +05:30
|
|
|
config.plugins.push(new CompressionPlugin());
|
2017-09-10 17:25:29 +05:30
|
|
|
}
|
2017-08-17 22:00:37 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (IS_DEV_SERVER) {
|
|
|
|
config.devtool = 'cheap-module-eval-source-map';
|
|
|
|
config.devServer = {
|
|
|
|
host: DEV_SERVER_HOST,
|
|
|
|
port: DEV_SERVER_PORT,
|
2017-09-10 17:25:29 +05:30
|
|
|
disableHostCheck: true,
|
2017-08-17 22:00:37 +05:30
|
|
|
headers: { 'Access-Control-Allow-Origin': '*' },
|
|
|
|
stats: 'errors-only',
|
2017-09-10 17:25:29 +05:30
|
|
|
hot: DEV_SERVER_LIVERELOAD,
|
2018-05-09 12:01:36 +05:30
|
|
|
inline: DEV_SERVER_LIVERELOAD,
|
2017-08-17 22:00:37 +05:30
|
|
|
};
|
2018-10-15 14:42:47 +05:30
|
|
|
config.plugins.push({
|
|
|
|
apply(compiler) {
|
|
|
|
compiler.hooks.emit.tapAsync('WatchForChangesPlugin', (compilation, callback) => {
|
|
|
|
const missingDeps = Array.from(compilation.missingDependencies);
|
|
|
|
const nodeModulesPath = path.join(ROOT_PATH, 'node_modules');
|
|
|
|
const hasMissingNodeModules = missingDeps.some(
|
|
|
|
file => file.indexOf(nodeModulesPath) !== -1
|
|
|
|
);
|
|
|
|
|
|
|
|
// watch for changes to missing node_modules
|
|
|
|
if (hasMissingNodeModules) compilation.contextDependencies.add(nodeModulesPath);
|
|
|
|
|
|
|
|
// watch for changes to automatic entrypoints
|
|
|
|
watchAutoEntries.forEach(watchPath => compilation.contextDependencies.add(watchPath));
|
|
|
|
|
|
|
|
// report our auto-generated bundle count
|
|
|
|
console.log(
|
|
|
|
`${autoEntriesCount} entries from '/pages' automatically added to webpack output.`
|
|
|
|
);
|
|
|
|
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
});
|
2017-09-10 17:25:29 +05:30
|
|
|
if (DEV_SERVER_LIVERELOAD) {
|
|
|
|
config.plugins.push(new webpack.HotModuleReplacementPlugin());
|
|
|
|
}
|
2017-08-17 22:00:37 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (WEBPACK_REPORT) {
|
|
|
|
config.plugins.push(
|
|
|
|
new BundleAnalyzerPlugin({
|
|
|
|
analyzerMode: 'static',
|
|
|
|
generateStatsFile: true,
|
|
|
|
openAnalyzer: false,
|
|
|
|
reportFilename: path.join(ROOT_PATH, 'webpack-report/index.html'),
|
|
|
|
statsFilename: path.join(ROOT_PATH, 'webpack-report/stats.json'),
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = config;
|