| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292 | const webpack = require('webpack')const path = require('path')const fs = require('fs-extra')const yargs = require('yargs').argvconst _ = require('lodash')const Fiber = require('fibers')const { VueLoaderPlugin } = require('vue-loader')const { CleanWebpackPlugin } = require('clean-webpack-plugin')const CopyWebpackPlugin = require('copy-webpack-plugin')const HtmlWebpackPlugin = require('html-webpack-plugin')const HtmlWebpackPugPlugin = require('html-webpack-pug-plugin')const MiniCssExtractPlugin = require('mini-css-extract-plugin')const OptimizeCssAssetsPlugin = require('optimize-css-assets-webpack-plugin')const ScriptExtHtmlWebpackPlugin = require('script-ext-html-webpack-plugin')const VuetifyLoaderPlugin = require('vuetify-loader/lib/plugin')const WebpackBarPlugin = require('webpackbar')const SriWebpackPlugin = require('webpack-subresource-integrity')const babelConfig = fs.readJsonSync(path.join(process.cwd(), '.babelrc'))const cacheDir = '.webpack-cache/cache'const babelDir = path.join(process.cwd(), '.webpack-cache/babel')process.noDeprecation = truefs.emptyDirSync(path.join(process.cwd(), 'assets'))module.exports = {  mode: 'production',  entry: {    app: './client/index-app.js',    legacy: './client/index-legacy.js',    setup: './client/index-setup.js'  },  output: {    path: path.join(process.cwd(), 'assets'),    publicPath: '/',    filename: 'js/[name].[hash].js',    chunkFilename: 'js/[name].[chunkhash].js',    globalObject: 'this',    crossOriginLoading: 'use-credentials'  },  module: {    rules: [      {        test: /\.js$/,        exclude: /node_modules/,        use: [          {            loader: 'cache-loader',            options: {              cacheDirectory: cacheDir            }          },          {            loader: 'babel-loader',            options: {              ...babelConfig,              cacheDirectory: babelDir            }          }        ]      },      {        test: /\.css$/,        use: [          'style-loader',          MiniCssExtractPlugin.loader,          'css-loader',          'postcss-loader'        ]      },      {        test: /\.sass$/,        use: [          {            loader: 'cache-loader',            options: {              cacheDirectory: cacheDir            }          },          'style-loader',          'css-loader',          'postcss-loader',          {            loader: 'sass-loader',            options: {              implementation: require('sass'),              fiber: Fiber,              sourceMap: false            }          }        ]      },      {        test: /\.scss$/,        use: [          {            loader: 'cache-loader',            options: {              cacheDirectory: cacheDir            }          },          'style-loader',          MiniCssExtractPlugin.loader,          'css-loader',          'postcss-loader',          {            loader: 'sass-loader',            options: {              implementation: require('sass'),              fiber: Fiber,              sourceMap: false            }          },          {            loader: 'sass-resources-loader',            options: {              resources: path.join(process.cwd(), '/client/scss/global.scss')            }          }        ]      },      {        test: /\.vue$/,        loader: 'vue-loader'      },      {        test: /\.pug$/,        exclude: [          path.join(process.cwd(), 'dev')        ],        loader: 'pug-plain-loader'      },      {        test: /\.(png|jpg|gif)$/,        use: [          {            loader: 'url-loader',            options: {              limit: 8192            }          }        ]      },      {        test: /\.svg$/,        exclude: [          path.join(process.cwd(), 'node_modules/grapesjs')        ],        use: [          {            loader: 'file-loader',            options: {              name: '[name].[ext]',              outputPath: 'svg/'            }          }        ]      },      {        test: /\.(graphql|gql)$/,        exclude: /node_modules/,        use: [          { loader: 'graphql-persisted-document-loader' },          { loader: 'graphql-tag/loader' }        ]      },      {        test: /\.(woff(2)?|ttf|eot|svg)(\?v=\d+\.\d+\.\d+)?$/,        exclude: [          path.join(process.cwd(), 'client')        ],        use: [{          loader: 'file-loader',          options: {            name: '[name].[ext]',            outputPath: 'fonts/'          }        }]      },      {        test: /.jsx$/,        loader: 'babel-loader',        exclude: /node_modules/,        options: {          presets: ['es2015', 'react']        }      },      {        test: /\.flow$/,        loader: 'ignore-loader'      }    ]  },  plugins: [    new VueLoaderPlugin(),    new VuetifyLoaderPlugin(),    new webpack.BannerPlugin('Wiki.js - wiki.js.org - Licensed under AGPL'),    new CopyWebpackPlugin([      { from: 'client/static' },      { from: './node_modules/prismjs/components', to: 'js/prism' },      { from: './node_modules/graphql-voyager/dist/voyager.worker.js', to: 'js/' }    ], {}),    new MiniCssExtractPlugin({      filename: 'css/bundle.[hash].css',      chunkFilename: 'css/[name].[chunkhash].css'    }),    new HtmlWebpackPlugin({      template: 'dev/templates/master.pug',      filename: '../server/views/master.pug',      hash: false,      inject: false,      excludeChunks: ['setup', 'legacy']    }),    new HtmlWebpackPlugin({      template: 'dev/templates/legacy.pug',      filename: '../server/views/legacy/master.pug',      hash: false,      inject: false,      excludeChunks: ['setup', 'app']    }),    new HtmlWebpackPlugin({      template: 'dev/templates/setup.pug',      filename: '../server/views/setup.pug',      hash: false,      inject: false,      excludeChunks: ['app', 'legacy']    }),    new HtmlWebpackPugPlugin(),    new ScriptExtHtmlWebpackPlugin({      sync: 'runtime.js',      defaultAttribute: 'async'    }),    new SriWebpackPlugin({      hashFuncNames: ['sha256', 'sha512'],      enabled: true    }),    new WebpackBarPlugin({      name: 'Client Assets'    }),    new CleanWebpackPlugin(),    new OptimizeCssAssetsPlugin({      cssProcessorOptions: { discardComments: { removeAll: true } },      canPrint: true    }),    new webpack.DefinePlugin({      'process.env.NODE_ENV': JSON.stringify('production'),      'process.env.CURRENT_THEME': JSON.stringify(_.defaultTo(yargs.theme, 'default')),      '__REACT_DEVTOOLS_GLOBAL_HOOK__': '({ isDisabled: true })'    })  ],  optimization: {    namedModules: true,    namedChunks: true,    splitChunks: {      name: 'vendor',      minChunks: 2    },    runtimeChunk: 'single'  },  resolve: {    mainFields: ['browser', 'main', 'module'],    symlinks: true,    alias: {      '@': path.join(process.cwd(), 'client'),      'vue$': 'vue/dist/vue.esm.js',      'gql': path.join(process.cwd(), 'client/graph'),      // Duplicates fixes:      'apollo-link': path.join(process.cwd(), 'node_modules/apollo-link'),      'apollo-utilities': path.join(process.cwd(), 'node_modules/apollo-utilities'),      'uc.micro': path.join(process.cwd(), 'node_modules/uc.micro')    },    extensions: [      '.js',      '.json',      'jsx',      '.vue'    ],    modules: [      'node_modules'    ]  },  node: {    fs: 'empty'  },  stats: {    children: false,    entrypoints: false  },  target: 'web'}
 |