2020-01-28 10:30:39 +03:00
const fastGlob = require ( 'fast-glob' ) ;
2020-06-12 14:26:37 +03:00
const wrapAnsi = require ( 'wrap-ansi' ) ;
2020-08-22 04:23:03 +03:00
const AddAssetPlugin = require ( 'add-asset-webpack-plugin' ) ;
2020-08-12 01:30:47 +03:00
const CssMinimizerPlugin = require ( 'css-minimizer-webpack-plugin' ) ;
2020-01-28 10:30:39 +03:00
const FixStyleOnlyEntriesPlugin = require ( 'webpack-fix-style-only-entries' ) ;
2020-01-25 11:41:34 +03:00
const MiniCssExtractPlugin = require ( 'mini-css-extract-plugin' ) ;
2020-05-14 19:06:01 +03:00
const MonacoWebpackPlugin = require ( 'monaco-editor-webpack-plugin' ) ;
2020-01-25 11:41:34 +03:00
const PostCSSPresetEnv = require ( 'postcss-preset-env' ) ;
2020-01-28 10:30:39 +03:00
const TerserPlugin = require ( 'terser-webpack-plugin' ) ;
const VueLoaderPlugin = require ( 'vue-loader/lib/plugin' ) ;
2020-03-11 22:34:54 +03:00
const { statSync } = require ( 'fs' ) ;
const { resolve , parse } = require ( 'path' ) ;
2020-06-12 14:26:37 +03:00
const { LicenseWebpackPlugin } = require ( 'license-webpack-plugin' ) ;
2020-03-11 22:34:54 +03:00
const { SourceMapDevToolPlugin } = require ( 'webpack' ) ;
2020-01-28 10:30:39 +03:00
2020-03-11 22:34:54 +03:00
const glob = ( pattern ) => fastGlob . sync ( pattern , { cwd : _ _dirname , absolute : true } ) ;
2020-02-11 20:02:41 +03:00
2020-01-28 10:30:39 +03:00
const themes = { } ;
2020-02-11 20:02:41 +03:00
for ( const path of glob ( 'web_src/less/themes/*.less' ) ) {
2020-01-28 10:30:39 +03:00
themes [ parse ( path ) . name ] = [ path ] ;
}
2019-11-13 17:52:13 +03:00
2020-02-23 11:47:42 +03:00
const isProduction = process . env . NODE _ENV !== 'development' ;
2020-07-28 00:01:25 +03:00
const filterCssImport = ( url , ... args ) => {
const cssFile = args [ 1 ] || args [ 0 ] ; // resourcePath is 2nd argument for url and 3rd for import
2020-06-28 05:59:56 +03:00
const importedFile = url . replace ( /[?#].+/ , '' ) . toLowerCase ( ) ;
2020-07-06 11:56:54 +03:00
2020-06-28 05:59:56 +03:00
if ( cssFile . includes ( 'fomantic' ) ) {
if ( /brand-icons/ . test ( importedFile ) ) return false ;
2020-07-12 12:10:56 +03:00
if ( /(eot|ttf|otf|woff|svg)$/ . test ( importedFile ) ) return false ;
2020-07-06 11:56:54 +03:00
}
if ( cssFile . includes ( 'font-awesome' ) ) {
if ( /(eot|ttf|otf|woff|svg)$/ . test ( importedFile ) ) return false ;
2020-06-28 05:59:56 +03:00
}
2020-07-06 11:56:54 +03:00
return true ;
2020-06-28 05:59:56 +03:00
} ;
2019-11-13 17:52:13 +03:00
module . exports = {
2020-02-23 11:47:42 +03:00
mode : isProduction ? 'production' : 'development' ,
2019-11-13 17:52:13 +03:00
entry : {
2020-01-28 10:30:39 +03:00
index : [
2020-06-28 05:59:56 +03:00
resolve ( _ _dirname , 'web_src/js/jquery.js' ) ,
resolve ( _ _dirname , 'web_src/fomantic/build/semantic.js' ) ,
2020-01-28 10:30:39 +03:00
resolve ( _ _dirname , 'web_src/js/index.js' ) ,
2020-06-28 05:59:56 +03:00
resolve ( _ _dirname , 'web_src/fomantic/build/semantic.css' ) ,
2020-01-28 10:30:39 +03:00
resolve ( _ _dirname , 'web_src/less/index.less' ) ,
] ,
swagger : [
2020-02-07 20:09:30 +03:00
resolve ( _ _dirname , 'web_src/js/standalone/swagger.js' ) ,
2020-01-28 10:30:39 +03:00
] ,
2020-05-22 04:45:34 +03:00
serviceworker : [
resolve ( _ _dirname , 'web_src/js/serviceworker.js' ) ,
] ,
2020-07-03 12:55:36 +03:00
'eventsource.sharedworker' : [
resolve ( _ _dirname , 'web_src/js/features/eventsource.sharedworker.js' ) ,
] ,
2020-01-28 10:30:39 +03:00
... themes ,
2019-11-13 17:52:13 +03:00
} ,
2020-01-14 21:02:08 +03:00
devtool : false ,
2019-11-13 17:52:13 +03:00
output : {
2020-01-28 10:30:39 +03:00
path : resolve ( _ _dirname , 'public' ) ,
2020-05-22 04:45:34 +03:00
filename : ( { chunk } ) => {
// serviceworker can only manage assets below it's script's directory so
// we have to put it in / instead of /js/
2020-05-23 16:58:58 +03:00
return chunk . name === 'serviceworker' ? '[name].js' : 'js/[name].js' ;
2020-05-22 04:45:34 +03:00
} ,
2020-01-25 11:41:34 +03:00
chunkFilename : 'js/[name].js' ,
2019-11-13 17:52:13 +03:00
} ,
optimization : {
2020-02-23 11:47:42 +03:00
minimize : isProduction ,
2020-01-25 11:41:34 +03:00
minimizer : [
new TerserPlugin ( {
sourceMap : true ,
extractComments : false ,
terserOptions : {
output : {
comments : false ,
} ,
2019-11-18 00:39:06 +03:00
} ,
2020-01-25 11:41:34 +03:00
} ) ,
2020-08-12 01:30:47 +03:00
new CssMinimizerPlugin ( {
2020-06-28 05:59:56 +03:00
sourceMap : true ,
2020-08-12 01:30:47 +03:00
minimizerOptions : {
2020-01-25 11:41:34 +03:00
preset : [
'default' ,
{
discardComments : {
removeAll : true ,
} ,
} ,
] ,
} ,
} ) ,
] ,
2020-01-29 00:57:20 +03:00
splitChunks : {
chunks : 'async' ,
name : ( _ , chunks ) => chunks . map ( ( item ) => item . name ) . join ( '-' ) ,
2020-06-28 05:59:56 +03:00
} ,
2019-11-13 17:52:13 +03:00
} ,
2019-11-15 00:39:51 +03:00
module : {
rules : [
2020-01-20 13:07:30 +03:00
{
test : /\.vue$/ ,
exclude : /node_modules/ ,
2020-01-25 11:41:34 +03:00
loader : 'vue-loader' ,
2020-01-20 13:07:30 +03:00
} ,
2020-04-13 16:02:31 +03:00
{
test : /\.worker\.js$/ ,
2020-05-14 19:06:01 +03:00
exclude : /monaco/ ,
2020-04-13 16:02:31 +03:00
use : [
{
loader : 'worker-loader' ,
options : {
2020-08-12 01:30:47 +03:00
inline : 'no-fallback' ,
2020-04-13 16:02:31 +03:00
} ,
} ,
] ,
} ,
2019-11-15 00:39:51 +03:00
{
test : /\.js$/ ,
exclude : /node_modules/ ,
2020-01-22 09:35:29 +03:00
use : [
{
loader : 'babel-loader' ,
options : {
2020-02-01 18:12:41 +03:00
cacheDirectory : true ,
cacheCompression : false ,
cacheIdentifier : [
resolve ( _ _dirname , 'package.json' ) ,
resolve ( _ _dirname , 'package-lock.json' ) ,
resolve ( _ _dirname , 'webpack.config.js' ) ,
] . map ( ( path ) => statSync ( path ) . mtime . getTime ( ) ) . join ( ':' ) ,
2020-02-23 11:47:42 +03:00
sourceMaps : true ,
2020-01-22 09:35:29 +03:00
presets : [
[
'@babel/preset-env' ,
{
useBuiltIns : 'usage' ,
corejs : 3 ,
2020-01-25 11:41:34 +03:00
} ,
] ,
2020-01-20 13:07:30 +03:00
] ,
2020-01-22 09:35:29 +03:00
plugins : [
[
'@babel/plugin-transform-runtime' ,
{
regenerator : true ,
}
] ,
] ,
2020-06-28 05:59:56 +03:00
generatorOpts : {
compact : false ,
} ,
2020-01-25 11:41:34 +03:00
} ,
2020-01-22 09:35:29 +03:00
} ,
] ,
2019-11-18 00:39:06 +03:00
} ,
{
2020-06-28 05:59:56 +03:00
test : /.css$/i ,
use : [
{
loader : MiniCssExtractPlugin . loader ,
} ,
{
loader : 'css-loader' ,
options : {
importLoaders : 1 ,
url : filterCssImport ,
import : filterCssImport ,
sourceMap : true ,
} ,
} ,
{
loader : 'postcss-loader' ,
options : {
plugins : ( ) => [
PostCSSPresetEnv ( ) ,
] ,
sourceMap : true ,
} ,
} ,
] ,
} ,
{
test : /.less$/i ,
2020-01-25 11:41:34 +03:00
use : [
{
loader : MiniCssExtractPlugin . loader ,
} ,
{
loader : 'css-loader' ,
options : {
2020-01-28 10:30:39 +03:00
importLoaders : 2 ,
2020-06-28 05:59:56 +03:00
url : filterCssImport ,
import : filterCssImport ,
sourceMap : true ,
} ,
2020-01-25 11:41:34 +03:00
} ,
{
loader : 'postcss-loader' ,
options : {
plugins : ( ) => [
PostCSSPresetEnv ( ) ,
] ,
2020-06-28 05:59:56 +03:00
sourceMap : true ,
2020-01-25 11:41:34 +03:00
} ,
} ,
2020-01-28 10:30:39 +03:00
{
loader : 'less-loader' ,
2020-06-28 05:59:56 +03:00
options : {
sourceMap : true ,
} ,
2020-01-28 10:30:39 +03:00
} ,
2020-01-25 11:41:34 +03:00
] ,
2019-11-18 00:39:06 +03:00
} ,
2020-02-11 20:02:41 +03:00
{
test : /\.svg$/ ,
2020-07-12 12:10:56 +03:00
include : resolve ( _ _dirname , 'public/img/svg' ) ,
2020-02-11 20:02:41 +03:00
use : [
{
2020-07-12 12:10:56 +03:00
loader : 'raw-loader' ,
2020-02-11 20:02:41 +03:00
} ,
] ,
} ,
2020-05-14 19:06:01 +03:00
{
test : /\.(ttf|woff2?)$/ ,
use : [
{
loader : 'file-loader' ,
options : {
name : '[name].[ext]' ,
outputPath : 'fonts/' ,
2020-07-29 21:44:23 +03:00
publicPath : ( url ) => ` ../fonts/ ${ url } ` , // required to remove css/ path segment
} ,
} ,
] ,
} ,
{
test : /\.png$/i ,
use : [
{
loader : 'file-loader' ,
options : {
name : '[name].[ext]' ,
outputPath : 'img/webpack/' ,
publicPath : ( url ) => ` ../img/webpack/ ${ url } ` , // required to remove css/ path segment
2020-05-14 19:06:01 +03:00
} ,
} ,
] ,
} ,
2020-01-25 11:41:34 +03:00
] ,
2020-01-14 21:02:08 +03:00
} ,
plugins : [
2020-01-20 13:07:30 +03:00
new VueLoaderPlugin ( ) ,
2020-07-12 12:10:56 +03:00
// avoid generating useless js output files for css--only chunks
2020-01-28 10:30:39 +03:00
new FixStyleOnlyEntriesPlugin ( {
2020-07-12 12:10:56 +03:00
extensions : [ 'less' , 'scss' , 'css' ] ,
2020-01-28 10:30:39 +03:00
silent : true ,
} ) ,
2020-01-25 11:41:34 +03:00
new MiniCssExtractPlugin ( {
filename : 'css/[name].css' ,
chunkFilename : 'css/[name].css' ,
} ) ,
2020-01-14 21:02:08 +03:00
new SourceMapDevToolPlugin ( {
2020-06-28 05:59:56 +03:00
filename : '[file].map' ,
2020-01-29 00:57:20 +03:00
include : [
'js/index.js' ,
2020-06-28 05:59:56 +03:00
'css/index.css' ,
2020-01-14 21:02:08 +03:00
] ,
} ) ,
2020-05-14 19:06:01 +03:00
new MonacoWebpackPlugin ( {
filename : 'js/monaco-[name].worker.js' ,
} ) ,
2020-08-22 04:23:03 +03:00
isProduction ? new LicenseWebpackPlugin ( {
2020-06-12 14:26:37 +03:00
outputFilename : 'js/licenses.txt' ,
perChunkOutput : false ,
addBanner : false ,
skipChildCompilers : true ,
modulesDirectories : [
resolve ( _ _dirname , 'node_modules' ) ,
] ,
2020-08-22 04:23:03 +03:00
additionalModules : [
'@primer/octicons' ,
] . map ( ( name ) => ( { name , directory : resolve ( _ _dirname , ` node_modules/ ${ name } ` ) } ) ) ,
2020-06-12 14:26:37 +03:00
renderLicenses : ( modules ) => {
const line = '-' . repeat ( 80 ) ;
return modules . map ( ( module ) => {
const { name , version } = module . packageJson ;
const { licenseId , licenseText } = module ;
const body = wrapAnsi ( licenseText || '' , 80 ) ;
return ` ${ line } \n ${ name } @ ${ version } - ${ licenseId } \n ${ line } \n ${ body } ` ;
} ) . join ( '\n' ) ;
} ,
stats : {
warnings : false ,
errors : true ,
} ,
2020-08-22 04:23:03 +03:00
} ) : new AddAssetPlugin ( 'js/licenses.txt' , ` Licenses are disabled during development ` ) ,
2020-01-14 21:02:08 +03:00
] ,
performance : {
2020-03-18 01:57:17 +03:00
hints : false ,
2020-05-14 19:06:01 +03:00
maxEntrypointSize : Infinity ,
maxAssetSize : Infinity ,
2020-01-14 21:02:08 +03:00
} ,
2020-01-22 09:35:29 +03:00
resolve : {
symlinks : false ,
2020-02-24 00:34:28 +03:00
alias : {
vue$ : 'vue/dist/vue.esm.js' , // needed because vue's default export is the runtime only
} ,
2020-01-25 11:41:34 +03:00
} ,
2020-02-23 11:47:42 +03:00
watchOptions : {
ignored : [
'node_modules/**' ,
] ,
} ,
2020-05-14 19:06:01 +03:00
stats : {
children : false ,
2020-08-02 17:06:06 +03:00
excludeAssets : [
// exclude monaco's language chunks in stats output for brevity
// https://github.com/microsoft/monaco-editor-webpack-plugin/issues/113
/^js\/[0-9]+\.js$/ ,
] ,
2020-05-14 19:06:01 +03:00
} ,
2019-11-13 17:52:13 +03:00
} ;