debian-mirror-gitlab/config/webpack.config.js

838 lines
29 KiB
JavaScript
Raw Normal View History

2022-03-02 08:16:31 +05:30
// eslint-disable-next-line import/order
const crypto = require('./helpers/patched_crypto');
2020-01-01 13:55:28 +05:30
const fs = require('fs');
2021-06-08 01:23:25 +05:30
const path = require('path');
2021-04-17 20:07:23 +05:30
2021-09-30 23:02:18 +05:30
const BABEL_VERSION = require('@babel/core/package.json').version;
2021-04-17 20:07:23 +05:30
const SOURCEGRAPH_VERSION = require('@sourcegraph/code-host-integration/package.json').version;
2022-10-11 01:57:18 +05:30
const GITLAB_WEB_IDE_VERSION = require('@gitlab/web-ide/package.json').version;
2021-04-17 20:07:23 +05:30
2021-09-30 23:02:18 +05:30
const BABEL_LOADER_VERSION = require('babel-loader/package.json').version;
2018-03-27 19:54:05 +05:30
const CompressionPlugin = require('compression-webpack-plugin');
2019-10-12 21:52:04 +05:30
const CopyWebpackPlugin = require('copy-webpack-plugin');
2021-04-17 20:07:23 +05:30
const glob = require('glob');
const VueLoaderPlugin = require('vue-loader/lib/plugin');
const VUE_LOADER_VERSION = require('vue-loader/package.json').version;
const VUE_VERSION = require('vue/package.json').version;
2021-09-30 23:02:18 +05:30
2022-11-25 23:54:43 +05:30
const { ESBuildMinifyPlugin } = require('esbuild-loader');
2021-04-17 20:07:23 +05:30
const webpack = require('webpack');
const { BundleAnalyzerPlugin } = require('webpack-bundle-analyzer');
const { StatsWriterPlugin } = require('webpack-stats-plugin');
const WEBPACK_VERSION = require('webpack/package.json').version;
2022-11-25 23:54:43 +05:30
const MonacoWebpackPlugin = require('monaco-editor-webpack-plugin');
const esbuildConfiguration = require('./esbuild.config');
2021-04-17 20:07:23 +05:30
2021-03-11 19:13:27 +05:30
const createIncrementalWebpackCompiler = require('./helpers/incremental_webpack_compiler');
2021-04-17 20:07:23 +05:30
const IS_EE = require('./helpers/is_ee_env');
2021-10-27 15:23:28 +05:30
const IS_JH = require('./helpers/is_jh_env');
2021-04-17 20:07:23 +05:30
const vendorDllHash = require('./helpers/vendor_dll_hash');
2021-12-11 22:18:48 +05:30
const GraphqlKnownOperationsPlugin = require('./plugins/graphql_known_operations_plugin');
2018-03-27 19:54:05 +05:30
const ROOT_PATH = path.resolve(__dirname, '..');
2021-09-30 23:02:18 +05:30
const SUPPORTED_BROWSERS = fs.readFileSync(path.join(ROOT_PATH, '.browserslistrc'), 'utf-8');
const SUPPORTED_BROWSERS_HASH = crypto
.createHash('sha256')
.update(SUPPORTED_BROWSERS)
.digest('hex');
2020-01-01 13:55:28 +05:30
const VENDOR_DLL = process.env.WEBPACK_VENDOR_DLL && process.env.WEBPACK_VENDOR_DLL !== 'false';
2018-12-13 13:39:08 +05:30
const CACHE_PATH = process.env.WEBPACK_CACHE_PATH || path.join(ROOT_PATH, 'tmp/cache');
2018-03-27 19:54:05 +05:30
const IS_PRODUCTION = process.env.NODE_ENV === 'production';
2022-04-04 11:22:00 +05:30
const IS_DEV_SERVER = process.env.WEBPACK_SERVE === 'true';
2022-11-25 23:54:43 +05:30
const WEBPACK_USE_ESBUILD_LOADER =
process.env.WEBPACK_USE_ESBUILD_LOADER && process.env.WEBPACK_USE_ESBUILD_LOADER !== 'false';
2021-04-17 20:07:23 +05:30
2022-04-04 11:22:00 +05:30
const { DEV_SERVER_HOST, DEV_SERVER_PUBLIC_ADDR } = process.env;
const DEV_SERVER_PORT = parseInt(process.env.DEV_SERVER_PORT, 10);
2021-02-22 17:27:13 +05:30
const DEV_SERVER_ALLOWED_HOSTS =
process.env.DEV_SERVER_ALLOWED_HOSTS && process.env.DEV_SERVER_ALLOWED_HOSTS.split(',');
2018-11-08 19:23:39 +05:30
const DEV_SERVER_LIVERELOAD = IS_DEV_SERVER && process.env.DEV_SERVER_LIVERELOAD !== 'false';
2021-03-11 19:13:27 +05:30
const INCREMENTAL_COMPILER_ENABLED =
IS_DEV_SERVER &&
process.env.DEV_SERVER_INCREMENTAL &&
2022-11-25 23:54:43 +05:30
process.env.DEV_SERVER_INCREMENTAL !== 'false' &&
!WEBPACK_USE_ESBUILD_LOADER;
2021-11-11 11:23:49 +05:30
const INCREMENTAL_COMPILER_TTL = Number(process.env.DEV_SERVER_INCREMENTAL_TTL) || Infinity;
2022-11-25 23:54:43 +05:30
const INCREMENTAL_COMPILER_RECORD_HISTORY =
IS_DEV_SERVER && !process.env.CI && !WEBPACK_USE_ESBUILD_LOADER;
2020-06-23 00:09:42 +05:30
const WEBPACK_REPORT = process.env.WEBPACK_REPORT && process.env.WEBPACK_REPORT !== 'false';
const WEBPACK_MEMORY_TEST =
process.env.WEBPACK_MEMORY_TEST && process.env.WEBPACK_MEMORY_TEST !== 'false';
2022-10-11 01:57:18 +05:30
let NO_COMPRESSION = process.env.NO_COMPRESSION && process.env.NO_COMPRESSION !== 'false';
let NO_SOURCEMAPS = process.env.NO_SOURCEMAPS && process.env.NO_SOURCEMAPS !== 'false';
let NO_HASHED_CHUNKS = process.env.NO_HASHED_CHUNKS && process.env.NO_HASHED_CHUNKS !== 'false';
if (WEBPACK_REPORT) {
console.log('Webpack report enabled. Running a "slim" production build.');
// For our webpack report we need no source maps, compression _or_ hashed file names.
NO_SOURCEMAPS = true;
NO_COMPRESSION = true;
NO_HASHED_CHUNKS = true;
}
2018-11-08 19:23:39 +05:30
2021-02-22 17:27:13 +05:30
const WEBPACK_OUTPUT_PATH = path.join(ROOT_PATH, 'public/assets/webpack');
const WEBPACK_PUBLIC_PATH = '/assets/webpack/';
const SOURCEGRAPH_PACKAGE = '@sourcegraph/code-host-integration';
2022-10-11 01:57:18 +05:30
const GITLAB_WEB_IDE_PACKAGE = '@gitlab/web-ide';
2021-02-22 17:27:13 +05:30
const SOURCEGRAPH_PATH = path.join('sourcegraph', SOURCEGRAPH_VERSION, '/');
const SOURCEGRAPH_OUTPUT_PATH = path.join(WEBPACK_OUTPUT_PATH, SOURCEGRAPH_PATH);
const SOURCEGRAPH_PUBLIC_PATH = path.join(WEBPACK_PUBLIC_PATH, SOURCEGRAPH_PATH);
2018-11-08 19:23:39 +05:30
2022-10-11 01:57:18 +05:30
const GITLAB_WEB_IDE_PATH = path.join('gitlab-vscode', GITLAB_WEB_IDE_VERSION, '/');
const GITLAB_WEB_IDE_OUTPUT_PATH = path.join(WEBPACK_OUTPUT_PATH, GITLAB_WEB_IDE_PATH);
const GITLAB_WEB_IDE_PUBLIC_PATH = path.join(WEBPACK_PUBLIC_PATH, GITLAB_WEB_IDE_PATH);
2018-11-08 19:23:39 +05:30
const devtool = IS_PRODUCTION ? 'source-map' : 'cheap-module-eval-source-map';
2018-03-27 19:54:05 +05:30
let autoEntriesCount = 0;
let watchAutoEntries = [];
2018-10-15 14:42:47 +05:30
const defaultEntries = ['./main'];
2018-03-27 19:54:05 +05:30
2021-03-11 19:13:27 +05:30
const incrementalCompiler = createIncrementalWebpackCompiler(
2021-11-11 11:23:49 +05:30
INCREMENTAL_COMPILER_RECORD_HISTORY,
2021-03-11 19:13:27 +05:30
INCREMENTAL_COMPILER_ENABLED,
path.join(CACHE_PATH, 'incremental-webpack-compiler-history.json'),
2021-11-11 11:23:49 +05:30
INCREMENTAL_COMPILER_TTL,
2021-03-11 19:13:27 +05:30
);
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
2021-04-17 20:07:23 +05:30
function generateAutoEntries(entryPath, prefix = '.') {
const chunkPath = entryPath.replace(/\/index\.js$/, '');
2018-03-27 19:54:05 +05:30
const chunkName = chunkPath.replace(/\//g, '.');
2021-04-17 20:07:23 +05:30
autoEntriesMap[chunkName] = `${prefix}/${entryPath}`;
2018-03-17 18:26:18 +05:30
}
2021-04-17 20:07:23 +05:30
pageEntries.forEach((entryPath) => generateAutoEntries(entryPath));
2018-03-17 18:26:18 +05:30
2019-07-07 11:18:12 +05:30
if (IS_EE) {
const eePageEntries = glob.sync('pages/**/index.js', {
cwd: path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
});
2021-04-17 20:07:23 +05:30
eePageEntries.forEach((entryPath) => generateAutoEntries(entryPath, 'ee'));
2019-07-07 11:18:12 +05:30
watchAutoEntries.push(path.join(ROOT_PATH, 'ee/app/assets/javascripts/pages/'));
}
2021-10-27 15:23:28 +05:30
if (IS_JH) {
const eePageEntries = glob.sync('pages/**/index.js', {
cwd: path.join(ROOT_PATH, 'jh/app/assets/javascripts'),
});
eePageEntries.forEach((entryPath) => generateAutoEntries(entryPath, 'jh'));
watchAutoEntries.push(path.join(ROOT_PATH, 'jh/app/assets/javascripts/pages/'));
}
2018-10-15 14:42:47 +05:30
const autoEntryKeys = Object.keys(autoEntriesMap);
autoEntriesCount = autoEntryKeys.length;
// import ancestor entrypoints within their children
2021-03-08 18:12:59 +05:30
autoEntryKeys.forEach((entry) => {
2018-10-15 14:42:47 +05:30
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
2021-09-04 01:27:46 +05:30
/*
If you create manual entries, ensure that these import `app/assets/javascripts/webpack.js` right at
the top of the entry in order to ensure that the public path is correctly determined for loading
assets async. See: https://webpack.js.org/configuration/output/#outputpublicpath
Note: WebPack 5 has an 'auto' option for the public path which could allow us to remove this option
Note 2: If you are using web-workers, you might need to reset the public path, see:
https://gitlab.com/gitlab-org/gitlab/-/issues/321656
*/
2018-03-27 19:54:05 +05:30
const manualEntries = {
2018-10-15 14:42:47 +05:30
default: defaultEntries,
2021-04-17 20:07:23 +05:30
sentry: './sentry/index.js',
2020-11-24 15:15:51 +05:30
performance_bar: './performance_bar/index.js',
2021-10-27 15:23:28 +05:30
jira_connect_app: './jira_connect/subscriptions/index.js',
2022-03-02 08:16:31 +05:30
sandboxed_mermaid: './lib/mermaid.js',
2022-04-04 11:22:00 +05:30
redirect_listbox: './entrypoints/behaviors/redirect_listbox.js',
2022-11-25 23:54:43 +05:30
sandboxed_swagger: './lib/swagger.js',
2018-03-27 19:54:05 +05:30
};
2021-03-11 19:13:27 +05:30
return Object.assign(manualEntries, incrementalCompiler.filterEntryPoints(autoEntries));
2018-03-27 19:54:05 +05:30
}
2019-07-07 11:18:12 +05:30
const alias = {
2022-04-04 11:22:00 +05:30
// Map Apollo client to apollo/client/core to prevent react related imports from being loaded
'@apollo/client$': '@apollo/client/core',
2019-07-07 11:18:12 +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'),
2021-04-29 21:17:54 +05:30
jquery$: 'jquery/dist/jquery.slim.js',
2021-01-03 14:25:43 +05:30
shared_queries: path.join(ROOT_PATH, 'app/graphql/queries'),
2019-07-07 11:18:12 +05:30
// the following resolves files which are different between CE and EE
ee_else_ce: path.join(ROOT_PATH, 'app/assets/javascripts'),
2019-10-12 21:52:04 +05:30
2021-11-11 11:23:49 +05:30
// the following resolves files which are different between CE and JH
jh_else_ce: path.join(ROOT_PATH, 'app/assets/javascripts'),
2022-01-26 12:08:38 +05:30
// the following resolves files which are different between CE/EE/JH
any_else_ce: path.join(ROOT_PATH, 'app/assets/javascripts'),
2019-10-12 21:52:04 +05:30
// override loader path for icons.svg so we do not duplicate this asset
'@gitlab/svgs/dist/icons.svg': path.join(
ROOT_PATH,
'app/assets/javascripts/lib/utils/icons_path.js',
),
2022-08-13 15:12:31 +05:30
// test-environment-only aliases duplicated from Jest config
'spec/test_constants$': path.join(ROOT_PATH, 'spec/frontend/__helpers__/test_constants'),
ee_else_ce_jest: path.join(ROOT_PATH, 'spec/frontend'),
helpers: path.join(ROOT_PATH, 'spec/frontend/__helpers__'),
jest: path.join(ROOT_PATH, 'spec/frontend'),
test_fixtures: path.join(ROOT_PATH, 'tmp/tests/frontend/fixtures'),
test_fixtures_static: path.join(ROOT_PATH, 'spec/frontend/fixtures/static'),
test_helpers: path.join(ROOT_PATH, 'spec/frontend_integration/test_helpers'),
public: path.join(ROOT_PATH, 'public'),
2019-07-07 11:18:12 +05:30
};
if (IS_EE) {
Object.assign(alias, {
ee: path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
2020-01-01 13:55:28 +05:30
ee_component: path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
2019-07-07 11:18:12 +05:30
ee_empty_states: path.join(ROOT_PATH, 'ee/app/views/shared/empty_states'),
ee_icons: path.join(ROOT_PATH, 'ee/app/views/shared/icons'),
ee_images: path.join(ROOT_PATH, 'ee/app/assets/images'),
ee_else_ce: path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
2022-01-26 12:08:38 +05:30
jh_else_ee: path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
any_else_ce: path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
2022-08-13 15:12:31 +05:30
// test-environment-only aliases duplicated from Jest config
ee_else_ce_jest: path.join(ROOT_PATH, 'ee/spec/frontend'),
ee_jest: path.join(ROOT_PATH, 'ee/spec/frontend'),
test_fixtures: path.join(ROOT_PATH, 'tmp/tests/frontend/fixtures-ee'),
2019-07-07 11:18:12 +05:30
});
}
2021-10-27 15:23:28 +05:30
if (IS_JH) {
Object.assign(alias, {
jh: path.join(ROOT_PATH, 'jh/app/assets/javascripts'),
2021-11-11 11:23:49 +05:30
jh_component: path.join(ROOT_PATH, 'jh/app/assets/javascripts'),
jh_empty_states: path.join(ROOT_PATH, 'jh/app/views/shared/empty_states'),
2021-10-27 15:23:28 +05:30
jh_icons: path.join(ROOT_PATH, 'jh/app/views/shared/icons'),
jh_images: path.join(ROOT_PATH, 'jh/app/assets/images'),
2022-01-26 12:08:38 +05:30
// jh path alias https://gitlab.com/gitlab-org/gitlab/-/merge_requests/74305#note_732793956
2021-11-11 11:23:49 +05:30
jh_else_ce: path.join(ROOT_PATH, 'jh/app/assets/javascripts'),
2022-01-26 12:08:38 +05:30
jh_else_ee: path.join(ROOT_PATH, 'jh/app/assets/javascripts'),
any_else_ce: path.join(ROOT_PATH, 'jh/app/assets/javascripts'),
2021-10-27 15:23:28 +05:30
2022-08-13 15:12:31 +05:30
// test-environment-only aliases duplicated from Jest config
jh_jest: path.join(ROOT_PATH, 'jh/spec/frontend'),
2020-10-24 23:57:45 +05:30
});
}
2020-01-01 13:55:28 +05:30
let dll;
if (VENDOR_DLL && !IS_PRODUCTION) {
const dllHash = vendorDllHash();
const dllCachePath = path.join(ROOT_PATH, `tmp/cache/webpack-dlls/${dllHash}`);
2020-04-08 14:13:33 +05:30
dll = {
manifestPath: path.join(dllCachePath, 'vendor.dll.manifest.json'),
cacheFrom: dllCachePath,
2021-02-22 17:27:13 +05:30
cacheTo: path.join(WEBPACK_OUTPUT_PATH, `dll.${dllHash}/`),
2020-04-08 14:13:33 +05:30
publicPath: `dll.${dllHash}/vendor.dll.bundle.js`,
exists: null,
};
2020-01-01 13:55:28 +05:30
}
2022-11-25 23:54:43 +05:30
const defaultJsOptions = {
cacheDirectory: path.join(CACHE_PATH, 'babel-loader'),
cacheIdentifier: [
process.env.BABEL_ENV || process.env.NODE_ENV || 'development',
webpack.version,
BABEL_VERSION,
BABEL_LOADER_VERSION,
// Ensure that changing supported browsers will refresh the cache
// in order to not pull in outdated files that import core-js
SUPPORTED_BROWSERS_HASH,
].join('|'),
cacheCompression: false,
};
if (WEBPACK_USE_ESBUILD_LOADER) {
console.log('esbuild-loader is active');
}
2018-11-08 19:23:39 +05:30
module.exports = {
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: {
2021-02-22 17:27:13 +05:30
path: WEBPACK_OUTPUT_PATH,
publicPath: WEBPACK_PUBLIC_PATH,
2022-10-11 01:57:18 +05:30
filename:
IS_PRODUCTION && !NO_HASHED_CHUNKS ? '[name].[contenthash:8].bundle.js' : '[name].bundle.js',
chunkFilename:
IS_PRODUCTION && !NO_HASHED_CHUNKS ? '[name].[contenthash:8].chunk.js' : '[name].chunk.js',
2018-10-15 14:42:47 +05:30
globalObject: 'this', // allow HMR and web workers to play nice
},
2018-11-08 19:23:39 +05:30
resolve: {
2021-06-08 01:23:25 +05:30
extensions: ['.js'],
2019-07-07 11:18:12 +05:30
alias,
2017-08-17 22:00:37 +05:30
},
module: {
2018-11-08 19:23:39 +05:30
strictExportPresence: true,
2017-08-17 22:00:37 +05:30
rules: [
2019-02-15 15:39:39 +05:30
{
type: 'javascript/auto',
test: /\.mjs$/,
use: [],
},
2022-11-25 23:54:43 +05:30
WEBPACK_USE_ESBUILD_LOADER && {
test: /\.js$/,
exclude: (modulePath) =>
/node_modules|vendor[\\/]assets/.test(modulePath) && !/\.vue\.js/.test(modulePath),
loader: 'esbuild-loader',
options: esbuildConfiguration,
},
!WEBPACK_USE_ESBUILD_LOADER && {
2017-08-17 22:00:37 +05:30
test: /\.js$/,
2021-04-17 20:07:23 +05:30
exclude: (modulePath) =>
2022-04-04 11:22:00 +05:30
/node_modules|vendor[\\/]assets/.test(modulePath) && !/\.vue\.js/.test(modulePath),
2017-08-17 22:00:37 +05:30
loader: 'babel-loader',
2022-11-25 23:54:43 +05:30
options: defaultJsOptions,
},
WEBPACK_USE_ESBUILD_LOADER && {
test: /\.js$/,
include: (modulePath) =>
/node_modules\/(monaco-worker-manager|monaco-marker-data-provider)\/index\.js/.test(
modulePath,
) || /node_modules\/yaml/.test(modulePath),
loader: 'esbuild-loader',
options: esbuildConfiguration,
},
!WEBPACK_USE_ESBUILD_LOADER && {
test: /\.js$/,
include: (modulePath) =>
/node_modules\/(monaco-worker-manager|monaco-marker-data-provider)\/index\.js/.test(
modulePath,
) || /node_modules\/yaml/.test(modulePath),
loader: 'babel-loader',
2018-10-15 14:42:47 +05:30
options: {
2022-11-25 23:54:43 +05:30
plugins: ['@babel/plugin-proposal-numeric-separator'],
...defaultJsOptions,
2018-10-15 14:42:47 +05:30
},
2017-08-17 22:00:37 +05:30
},
{
test: /\.vue$/,
loader: 'vue-loader',
2018-11-08 19:23:39 +05:30
options: {
cacheDirectory: path.join(CACHE_PATH, 'vue-loader'),
cacheIdentifier: [
process.env.NODE_ENV || 'development',
webpack.version,
VUE_VERSION,
VUE_LOADER_VERSION,
].join('|'),
},
2017-08-17 22:00:37 +05:30
},
2019-02-15 15:39:39 +05:30
{
test: /\.(graphql|gql)$/,
exclude: /node_modules/,
loader: 'graphql-tag/loader',
},
2019-10-12 21:52:04 +05:30
{
test: /icons\.svg$/,
loader: 'file-loader',
options: {
2020-05-24 23:13:21 +05:30
name: '[name].[contenthash:8].[ext]',
2019-10-12 21:52:04 +05:30
},
},
2017-08-17 22:00:37 +05:30
{
test: /\.svg$/,
2019-10-12 21:52:04 +05:30
exclude: /icons\.svg$/,
2017-08-17 22:00:37 +05:30
loader: 'raw-loader',
},
{
2020-03-13 15:44:24 +05:30
test: /\.(gif|png|mp4)$/,
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
{
2020-01-01 13:55:28 +05:30
test: /_worker\.js$/,
2018-03-17 18:26:18 +05:30
use: [
{
loader: 'worker-loader',
options: {
2020-05-24 23:13:21 +05:30
name: '[name].[contenthash:8].worker.js',
2019-07-07 11:18:12 +05:30
inline: IS_DEV_SERVER,
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
{
2022-06-21 17:19:12 +05:30
test: /\.(worker(\.min)?\.js|pdf)$/,
2017-08-17 22:00:37 +05:30
exclude: /node_modules/,
loader: 'file-loader',
2017-09-10 17:25:29 +05:30
options: {
2020-05-24 23:13:21 +05:30
name: '[name].[contenthash:8].[ext]',
2018-05-09 12:01:36 +05:30
},
2017-08-17 22:00:37 +05:30
},
{
2018-11-08 19:23:39 +05:30
test: /.css$/,
2018-03-17 18:26:18 +05:30
use: [
2018-11-08 19:23:39 +05:30
'vue-style-loader',
2018-03-27 19:54:05 +05:30
{
2018-03-17 18:26:18 +05:30
loader: 'css-loader',
options: {
2020-05-24 23:13:21 +05:30
modules: 'global',
localIdentName: '[name].[contenthash:8].[ext]',
2018-05-09 12:01:36 +05:30
},
2018-03-17 18:26:18 +05:30
},
],
},
{
test: /\.(eot|ttf|woff|woff2)$/,
2020-07-28 23:09:34 +05:30
include: /node_modules\/(katex\/dist\/fonts|monaco-editor)/,
2018-03-17 18:26:18 +05:30
loader: 'file-loader',
options: {
2020-05-24 23:13:21 +05:30
name: '[name].[contenthash:8].[ext]',
2020-06-23 00:09:42 +05:30
esModule: false,
2018-05-09 12:01:36 +05:30
},
2017-08-17 22:00:37 +05:30
},
2021-11-18 22:05:49 +05:30
{
test: /editor\/schema\/.+\.json$/,
type: 'javascript/auto',
loader: 'file-loader',
options: {
name: '[name].[contenthash:8].[ext]',
},
},
2022-05-07 20:08:51 +05:30
{
test: /\.(yml|yaml)$/,
loader: 'raw-loader',
},
2022-11-25 23:54:43 +05:30
].filter(Boolean),
2018-11-08 19:23:39 +05:30
},
2017-09-10 17:25:29 +05:30
2018-11-08 19:23:39 +05:30
optimization: {
2020-05-24 23:13:21 +05:30
// Replace 'hashed' with 'deterministic' in webpack 5
moduleIds: 'hashed',
2018-11-08 19:23:39 +05:30
runtimeChunk: 'single',
splitChunks: {
2020-06-23 00:09:42 +05:30
maxInitialRequests: 20,
2020-10-24 23:57:45 +05:30
// In order to prevent firewalls tripping up: https://gitlab.com/gitlab-org/gitlab/-/issues/22648
automaticNameDelimiter: '-',
2018-11-08 19:23:39 +05:30
cacheGroups: {
default: false,
common: () => ({
priority: 20,
name: 'main',
chunks: 'initial',
minChunks: autoEntriesCount * 0.9,
}),
2021-06-08 01:23:25 +05:30
prosemirror: {
2021-04-29 21:17:54 +05:30
priority: 17,
2021-06-08 01:23:25 +05:30
name: 'prosemirror',
2021-04-29 21:17:54 +05:30
chunks: 'all',
2021-06-08 01:23:25 +05:30
test: /[\\/]node_modules[\\/]prosemirror.*?[\\/]/,
2021-04-29 21:17:54 +05:30
minChunks: 2,
reuseExistingChunk: true,
},
2021-01-03 14:25:43 +05:30
graphql: {
priority: 16,
name: 'graphql',
chunks: 'all',
test: /[\\/]node_modules[\\/][^\\/]*(immer|apollo|graphql|zen-observable)[^\\/]*[\\/]/,
minChunks: 2,
reuseExistingChunk: true,
},
2020-05-24 23:13:21 +05:30
monaco: {
priority: 15,
name: 'monaco',
2020-10-24 23:57:45 +05:30
chunks: 'all',
2020-05-24 23:13:21 +05:30
test: /[\\/]node_modules[\\/]monaco-editor[\\/]/,
minChunks: 2,
reuseExistingChunk: true,
},
echarts: {
priority: 14,
name: 'echarts',
chunks: 'all',
test: /[\\/]node_modules[\\/](echarts|zrender)[\\/]/,
minChunks: 2,
reuseExistingChunk: true,
},
security_reports: {
priority: 13,
name: 'security_reports',
chunks: 'initial',
test: /[\\/](vue_shared[\\/](security_reports|license_compliance)|security_dashboard)[\\/]/,
minChunks: 2,
reuseExistingChunk: true,
},
2018-11-08 19:23:39 +05:30
vendors: {
priority: 10,
chunks: 'async',
test: /[\\/](node_modules|vendor[\\/]assets[\\/]javascripts)[\\/]/,
},
commons: {
chunks: 'all',
minChunks: 2,
reuseExistingChunk: true,
},
},
},
2022-11-25 23:54:43 +05:30
...(WEBPACK_USE_ESBUILD_LOADER
? { minimizer: [new ESBuildMinifyPlugin(esbuildConfiguration)] }
: {}),
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',
2021-04-17 20:07:23 +05:30
transform(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,
2020-11-24 15:15:51 +05:30
errors: !IS_PRODUCTION,
warnings: !IS_PRODUCTION,
2017-09-10 17:25:29 +05:30
});
2020-01-01 13:55:28 +05:30
// tell our rails helper where to find the DLL files
if (dll) {
stats.dllAssets = dll.publicPath;
}
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
}),
2018-11-08 19:23:39 +05:30
// enable vue-loader to use existing loader rules for other module types
new VueLoaderPlugin(),
// automatically configure monaco editor web workers
2022-10-11 01:57:18 +05:30
new MonacoWebpackPlugin({
filename: '[name].[contenthash:8].worker.js',
2022-11-25 23:54:43 +05:30
customLanguages: [
{
label: 'yaml',
entry: 'monaco-yaml',
worker: {
id: 'monaco-yaml/yamlWorker',
entry: 'monaco-yaml/yaml.worker',
},
},
],
2022-10-11 01:57:18 +05:30
}),
2018-11-08 19:23:39 +05:30
2021-12-11 22:18:48 +05:30
new GraphqlKnownOperationsPlugin({ filename: 'graphql_known_operations.yml' }),
2017-08-17 22:00:37 +05:30
// fix legacy jQuery plugins which depend on globals
new webpack.ProvidePlugin({
$: 'jquery',
jQuery: 'jquery',
}),
2020-04-08 14:13:33 +05:30
// if DLLs are enabled, detect whether the DLL exists and create it automatically if necessary
dll && {
apply(compiler) {
compiler.hooks.beforeCompile.tapAsync('DllAutoCompilePlugin', (params, callback) => {
if (dll.exists) {
callback();
} else if (fs.existsSync(dll.manifestPath)) {
console.log(`Using vendor DLL found at: ${dll.cacheFrom}`);
dll.exists = true;
callback();
} else {
console.log(
`Warning: No vendor DLL found at: ${dll.cacheFrom}. Compiling DLL automatically.`,
);
2021-04-17 20:07:23 +05:30
// eslint-disable-next-line global-require
2021-09-30 23:02:18 +05:30
const dllConfig = require('./webpack.vendor.config');
2020-04-08 14:13:33 +05:30
const dllCompiler = webpack(dllConfig);
dllCompiler.run((err, stats) => {
if (err) {
return callback(err);
}
const info = stats.toJson();
if (stats.hasErrors()) {
console.error(info.errors.join('\n\n'));
return callback('DLL not compiled successfully.');
}
if (stats.hasWarnings()) {
console.warn(info.warnings.join('\n\n'));
console.warn('DLL compiled with warnings.');
} else {
console.log('DLL compiled successfully.');
}
dll.exists = true;
2021-04-17 20:07:23 +05:30
return callback();
2020-04-08 14:13:33 +05:30
});
}
});
},
},
2020-01-01 13:55:28 +05:30
// reference our compiled DLL modules
dll &&
new webpack.DllReferencePlugin({
context: ROOT_PATH,
manifest: dll.manifestPath,
}),
dll &&
2021-06-08 01:23:25 +05:30
new CopyWebpackPlugin({
patterns: [
{
from: dll.cacheFrom,
to: dll.cacheTo,
},
],
}),
2020-01-01 13:55:28 +05:30
!IS_EE &&
2021-03-08 18:12:59 +05:30
new webpack.NormalModuleReplacementPlugin(/^ee_component\/(.*)\.vue/, (resource) => {
2021-04-17 20:07:23 +05:30
// eslint-disable-next-line no-param-reassign
2019-07-07 11:18:12 +05:30
resource.request = path.join(
ROOT_PATH,
'app/assets/javascripts/vue_shared/components/empty_component.js',
);
2020-01-01 13:55:28 +05:30
}),
2019-07-07 11:18:12 +05:30
2022-08-27 11:52:29 +05:30
new webpack.NormalModuleReplacementPlugin(/markdown-it/, (resource) => {
// eslint-disable-next-line no-param-reassign
resource.request = path.join(ROOT_PATH, 'app/assets/javascripts/lib/markdown_it.js');
}),
2021-11-11 11:23:49 +05:30
!IS_JH &&
new webpack.NormalModuleReplacementPlugin(/^jh_component\/(.*)\.vue/, (resource) => {
// eslint-disable-next-line no-param-reassign
resource.request = path.join(
ROOT_PATH,
'app/assets/javascripts/vue_shared/components/empty_component.js',
);
}),
2021-06-08 01:23:25 +05:30
new CopyWebpackPlugin({
patterns: [
{
from: path.join(ROOT_PATH, 'node_modules/pdfjs-dist/cmaps/'),
2022-11-25 23:54:43 +05:30
to: path.join(WEBPACK_OUTPUT_PATH, 'pdfjs/cmaps/'),
},
{
from: path.join(ROOT_PATH, 'node_modules/pdfjs-dist/legacy/build/pdf.worker.min.js'),
to: path.join(WEBPACK_OUTPUT_PATH, 'pdfjs/'),
2021-06-08 01:23:25 +05:30
},
{
from: path.join(ROOT_PATH, 'node_modules', SOURCEGRAPH_PACKAGE, '/'),
to: SOURCEGRAPH_OUTPUT_PATH,
globOptions: {
ignore: ['package.json'],
},
},
2022-10-11 01:57:18 +05:30
{
from: path.join(ROOT_PATH, 'node_modules', GITLAB_WEB_IDE_PACKAGE, 'dist', 'public'),
to: GITLAB_WEB_IDE_OUTPUT_PATH,
},
2021-06-08 01:23:25 +05:30
{
from: path.join(
ROOT_PATH,
'node_modules/@gitlab/visual-review-tools/dist/visual_review_toolbar.js',
),
to: WEBPACK_OUTPUT_PATH,
},
],
}),
2019-10-12 21:52:04 +05:30
2018-11-08 19:23:39 +05:30
// compression can require a lot of compute time and is disabled in CI
IS_PRODUCTION && !NO_COMPRESSION && new CompressionPlugin(),
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
// WatchForChangesPlugin
// TODO: publish this as a separate plugin
IS_DEV_SERVER && {
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(
2021-03-08 18:12:59 +05:30
(file) => file.indexOf(nodeModulesPath) !== -1,
2018-11-08 19:23:39 +05:30
);
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
// watch for changes to missing node_modules
if (hasMissingNodeModules) compilation.contextDependencies.add(nodeModulesPath);
2018-03-17 18:26:18 +05:30
2018-11-08 19:23:39 +05:30
// watch for changes to automatic entrypoints
2021-03-08 18:12:59 +05:30
watchAutoEntries.forEach((watchPath) => compilation.contextDependencies.add(watchPath));
2017-09-10 17:25:29 +05:30
2018-11-08 19:23:39 +05:30
// report our auto-generated bundle count
2021-03-11 19:13:27 +05:30
if (incrementalCompiler.enabled) {
incrementalCompiler.logStatus(autoEntriesCount);
} else {
console.log(
`${autoEntriesCount} entries from '/pages' automatically added to webpack output.`,
);
}
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
callback();
});
},
},
2019-12-04 20:38:33 +05:30
// output the in-memory heap size upon compilation and exit
WEBPACK_MEMORY_TEST && {
apply(compiler) {
2021-04-17 20:07:23 +05:30
compiler.hooks.emit.tapAsync('ReportMemoryConsumptionPlugin', () => {
2019-12-04 20:38:33 +05:30
console.log('Assets compiled...');
if (global.gc) {
console.log('Running garbage collection...');
global.gc();
} else {
console.error(
"WARNING: you must use the --expose-gc node option to accurately measure webpack's heap size",
);
}
const memoryUsage = process.memoryUsage().heapUsed;
2021-03-08 18:12:59 +05:30
const toMB = (bytes) => Math.floor(bytes / 1024 / 1024);
2019-12-04 20:38:33 +05:30
console.log(`Webpack heap size: ${toMB(memoryUsage)} MB`);
2020-01-01 13:55:28 +05:30
const webpackStatistics = {
memoryUsage,
date: Date.now(), // milliseconds
commitSHA: process.env.CI_COMMIT_SHA,
nodeVersion: process.versions.node,
webpackVersion: WEBPACK_VERSION,
};
console.log(webpackStatistics);
fs.writeFileSync(
path.join(ROOT_PATH, 'webpack-dev-server.json'),
JSON.stringify(webpackStatistics),
);
2019-12-04 20:38:33 +05:30
// exit in case we're running webpack-dev-server
2021-04-17 20:07:23 +05:30
if (IS_DEV_SERVER) {
process.exit();
}
2019-12-04 20:38:33 +05:30
});
},
},
2018-11-08 19:23:39 +05:30
// optionally generate webpack bundle analysis
WEBPACK_REPORT &&
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'),
2019-12-21 20:55:43 +05:30
statsOptions: {
source: false,
2022-10-11 01:57:18 +05:30
errors: false,
warnings: false,
2019-12-21 20:55:43 +05:30
},
2018-11-08 19:23:39 +05:30
}),
2019-07-07 11:18:12 +05:30
new webpack.DefinePlugin({
2021-11-11 11:23:49 +05:30
// These are used to define window.gon.ee, window.gon.jh and other things properly in tests:
2019-12-21 20:55:43 +05:30
'process.env.IS_EE': JSON.stringify(IS_EE),
2021-11-11 11:23:49 +05:30
'process.env.IS_JH': JSON.stringify(IS_JH),
// These are used to check against "EE" properly in application code
2019-09-30 21:07:59 +05:30
IS_EE: IS_EE ? 'window.gon && window.gon.ee' : JSON.stringify(false),
2021-11-11 11:23:49 +05:30
IS_JH: IS_JH ? 'window.gon && window.gon.jh' : JSON.stringify(false),
2021-02-22 17:27:13 +05:30
// This is used by Sourcegraph because these assets are loaded dnamically
'process.env.SOURCEGRAPH_PUBLIC_PATH': JSON.stringify(SOURCEGRAPH_PUBLIC_PATH),
2022-10-11 01:57:18 +05:30
'process.env.GITLAB_WEB_IDE_PUBLIC_PATH': JSON.stringify(GITLAB_WEB_IDE_PUBLIC_PATH),
2022-05-07 20:08:51 +05:30
...(IS_PRODUCTION ? {} : { LIVE_RELOAD: DEV_SERVER_LIVERELOAD }),
2019-07-07 11:18:12 +05:30
}),
2020-06-23 00:09:42 +05:30
/* Pikaday has a optional dependency to moment.
We are currently not utilizing moment.
Ignoring this import removes warning from our development build.
Upstream reference:
https://github.com/Pikaday/Pikaday/blob/5c1a7559be/pikaday.js#L14
*/
new webpack.IgnorePlugin(/moment/, /pikaday/),
2018-11-08 19:23:39 +05:30
].filter(Boolean),
2022-04-04 11:22:00 +05:30
2018-11-08 19:23:39 +05:30
devServer: {
2022-04-04 11:22:00 +05:30
setupMiddlewares: (middlewares, devServer) => {
if (!devServer) {
throw new Error('webpack-dev-server is not defined');
}
const incrementalCompilerMiddleware = incrementalCompiler.createMiddleware(devServer);
if (incrementalCompilerMiddleware) {
2022-06-21 17:19:12 +05:30
middlewares.unshift({
name: 'incremental-compiler',
middleware: incrementalCompilerMiddleware,
});
2022-04-04 11:22:00 +05:30
}
return middlewares;
2021-03-11 19:13:27 +05:30
},
2022-04-04 11:22:00 +05:30
// Only print errors to CLI
devMiddleware: {
stats: 'errors-only',
},
host: DEV_SERVER_HOST || 'localhost',
port: DEV_SERVER_PORT || 3808,
2022-05-07 20:08:51 +05:30
// Setting up hot module reloading
// HMR works by setting up a websocket server and injecting
// a client script which connects to that server.
// The server will push messages to the client to reload parts
// of the JavaScript or reload the page if necessary
webSocketServer: DEV_SERVER_LIVERELOAD ? 'ws' : false,
2017-09-10 17:25:29 +05:30
hot: DEV_SERVER_LIVERELOAD,
2022-05-07 20:08:51 +05:30
liveReload: DEV_SERVER_LIVERELOAD,
2022-04-04 11:22:00 +05:30
// The following settings are mainly needed for HMR support in gitpod.
// Per default only local hosts are allowed, but here we could
// allow different hosts (e.g. ['.gitpod'], all of gitpod),
// as the webpack server will run on a different subdomain than
// the rails application
...(DEV_SERVER_ALLOWED_HOSTS ? { allowedHosts: DEV_SERVER_ALLOWED_HOSTS } : {}),
client: {
...(DEV_SERVER_PUBLIC_ADDR ? { webSocketURL: DEV_SERVER_PUBLIC_ADDR } : {}),
},
2018-11-08 19:23:39 +05:30
},
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
devtool: NO_SOURCEMAPS ? false : devtool,
2017-08-17 22:00:37 +05:30
2019-09-30 21:07:59 +05:30
node: {
2022-06-21 17:19:12 +05:30
fs: 'empty', // editorconfig requires 'fs'
2019-09-30 21:07:59 +05:30
setImmediate: false,
},
2018-11-08 19:23:39 +05:30
};