Created
May 14, 2018 22:42
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
'use strict' | |
require('./check-versions')() | |
process.env.NODE_ENV = 'production' | |
const ora = require('ora') | |
const rm = require('rimraf') | |
const path = require('path') | |
const chalk = require('chalk') | |
const webpack = require('webpack') | |
const config = require('../config') | |
const webpackConfig = require('./webpack.prod.conf') | |
const spinner = ora('building for production...') | |
spinner.start() | |
rm(path.join(config.build.assetsRoot, config.build.assetsSubDirectory), err => { | |
if (err) throw err | |
webpack(webpackConfig, (err, stats) => { | |
spinner.stop() | |
if (err) throw err | |
process.stdout.write(stats.toString({ | |
colors: true, | |
modules: false, | |
children: false, // if you are using ts-loader, setting this to true will make tyescript errors show up during build | |
chunks: false, | |
chunkModules: false | |
}) + '\n\n') | |
if (stats.hasErrors()) { | |
console.log(chalk.red(' Build failed with errors.\n')) | |
process.exit(1) | |
} | |
console.log(chalk.cyan(' Build complete.\n')) | |
console.log(chalk.yellow( | |
' Tip: built files are meant to be served over an HTTP server.\n' + | |
' Opening index.html over file:// won\'t work.\n' | |
)) | |
}) | |
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
'use strict' | |
const path = require('path') | |
const config = require('../config') | |
const ExtractTextPlugin = require('extract-text-webpack-plugin') | |
const packageConfig = require('../package.json') | |
exports.assetsPath = function (_path) { | |
const assetsSubDirectory = process.env.NODE_ENV === 'production' | |
? config.build.assetsSubDirectory | |
: config.dev.assetsSubDirectory | |
return path.posix.join(assetsSubDirectory, _path) | |
} | |
exports.cssLoaders = function (options) { | |
options = options || {} | |
const cssLoader = { | |
loader: 'css-loader', | |
options: { | |
sourceMap: options.sourceMap | |
} | |
} | |
const lessLoader = { | |
loader: 'less-loader', | |
options: { | |
sourceMap: options.sourceMap | |
} | |
} | |
const postcssLoader = { | |
loader: 'postcss-loader', | |
options: { | |
sourceMap: options.sourceMap | |
} | |
} | |
// generate loader string to be used with extract text plugin | |
function generateLoaders (loader, loaderOptions) { | |
const loaders = options.usePostCSS ? [cssLoader, postcssLoader, lessLoader] : [cssLoader] | |
if (loader) { | |
loaders.push({ | |
loader: loader + '-loader', | |
options: Object.assign({}, loaderOptions, { | |
sourceMap: options.sourceMap | |
}) | |
}) | |
} | |
// Extract CSS when that option is specified | |
// (which is the case during production build) | |
if (options.extract) { | |
return ExtractTextPlugin.extract({ | |
use: loaders, | |
fallback: 'vue-style-loader' | |
}) | |
} else { | |
return ['vue-style-loader'].concat(loaders) | |
} | |
} | |
// https://vue-loader.vuejs.org/en/configurations/extract-css.html | |
return { | |
css: generateLoaders(), | |
postcss: generateLoaders(), | |
less: generateLoaders('less'), | |
sass: generateLoaders('sass', { indentedSyntax: true }), | |
scss: generateLoaders('sass'), | |
stylus: generateLoaders('stylus'), | |
styl: generateLoaders('stylus') | |
} | |
} | |
// Generate loaders for standalone style files (outside of .vue) | |
exports.styleLoaders = function (options) { | |
const output = [] | |
const loaders = exports.cssLoaders(options) | |
for (const extension in loaders) { | |
const loader = loaders[extension] | |
output.push({ | |
test: new RegExp('\\.' + extension + '$'), | |
use: loader | |
}) | |
} | |
return output | |
} | |
exports.createNotifierCallback = () => { | |
const notifier = require('node-notifier') | |
return (severity, errors) => { | |
if (severity !== 'error') return | |
const error = errors[0] | |
const filename = error.file && error.file.split('!').pop() | |
notifier.notify({ | |
title: packageConfig.name, | |
message: severity + ': ' + error.name, | |
subtitle: filename || '', | |
icon: path.join(__dirname, 'logo.png') | |
}) | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
'use strict' | |
const utils = require('./utils') | |
const config = require('../config') | |
const isProduction = process.env.NODE_ENV === 'production' | |
const sourceMapEnabled = isProduction | |
? config.build.productionSourceMap | |
: config.dev.cssSourceMap | |
module.exports = { | |
loaders: utils.cssLoaders({ | |
sourceMap: sourceMapEnabled, | |
extract: isProduction | |
}), | |
cssSourceMap: sourceMapEnabled, | |
cacheBusting: config.dev.cacheBusting, | |
transformToRequire: { | |
video: ['src', 'poster'], | |
source: 'src', | |
img: 'src', | |
image: 'xlink:href' | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
'use strict' | |
const path = require('path') | |
const utils = require('./utils') | |
const config = require('../config') | |
const vueLoaderConfig = require('./vue-loader.conf') | |
function resolve (dir) { | |
return path.join(__dirname, '..', dir) | |
} | |
module.exports = { | |
context: path.resolve(__dirname, '../'), | |
entry: { | |
app: './src/main.js' | |
}, | |
output: { | |
path: config.build.assetsRoot, | |
filename: '[name].js', | |
publicPath: process.env.NODE_ENV === 'production' | |
? config.build.assetsPublicPath | |
: config.dev.assetsPublicPath | |
}, | |
resolve: { | |
extensions: ['.js', '.vue', '.json'], | |
alias: { | |
'vue$': 'vue/dist/vue.esm.js', | |
'@': resolve('src'), | |
} | |
}, | |
module: { | |
rules: [ | |
{ | |
test: /\.vue$/, | |
loader: 'vue-loader', | |
options: vueLoaderConfig | |
}, | |
{ | |
test: /\.js$/, | |
loader: 'babel-loader', | |
include: [resolve('src'), resolve('test')] | |
}, | |
{ | |
test: /\.(png|jpe?g|gif|svg)(\?.*)?$/, | |
loader: 'url-loader', | |
options: { | |
limit: 10000, | |
name: utils.assetsPath('img/[name].[hash:7].[ext]') | |
} | |
}, | |
{ | |
test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/, | |
loader: 'url-loader', | |
options: { | |
limit: 10000, | |
name: utils.assetsPath('media/[name].[hash:7].[ext]') | |
} | |
}, | |
{ | |
test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/, | |
loader: 'url-loader', | |
options: { | |
limit: 10000, | |
name: utils.assetsPath('fonts/[name].[hash:7].[ext]') | |
} | |
} | |
] | |
}, | |
node: { | |
// prevent webpack from injecting useless setImmediate polyfill because Vue | |
// source contains it (although only uses it if it's native). | |
setImmediate: false, | |
// prevent webpack from injecting mocks to Node native modules | |
// that does not make sense for the client | |
dgram: 'empty', | |
fs: 'empty', | |
net: 'empty', | |
tls: 'empty', | |
child_process: 'empty' | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
'use strict' | |
const utils = require('./utils') | |
const webpack = require('webpack') | |
const config = require('../config') | |
const merge = require('webpack-merge') | |
const baseWebpackConfig = require('./webpack.base.conf') | |
const HtmlWebpackPlugin = require('html-webpack-plugin') | |
const FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin') | |
const portfinder = require('portfinder') | |
const HOST = process.env.HOST | |
const PORT = process.env.PORT && Number(process.env.PORT) | |
const devWebpackConfig = merge(baseWebpackConfig, { | |
module: { | |
rules: utils.styleLoaders({ sourceMap: config.dev.cssSourceMap, usePostCSS: true }) | |
}, | |
// cheap-module-eval-source-map is faster for development | |
devtool: config.dev.devtool, | |
// these devServer options should be customized in /config/index.js | |
devServer: { | |
clientLogLevel: 'warning', | |
historyApiFallback: true, | |
hot: true, | |
compress: true, | |
host: HOST || config.dev.host, | |
port: PORT || config.dev.port, | |
open: config.dev.autoOpenBrowser, | |
overlay: config.dev.errorOverlay | |
? { warnings: false, errors: true } | |
: false, | |
publicPath: config.dev.assetsPublicPath, | |
proxy: config.dev.proxyTable, | |
quiet: true, // necessary for FriendlyErrorsPlugin | |
watchOptions: { | |
poll: config.dev.poll, | |
} | |
}, | |
plugins: [ | |
new webpack.DefinePlugin({ | |
'process.env': require('../config/dev.env') | |
}), | |
new webpack.HotModuleReplacementPlugin(), | |
new webpack.NamedModulesPlugin(), // HMR shows correct file names in console on update. | |
new webpack.NoEmitOnErrorsPlugin(), | |
// https://github.com/ampedandwired/html-webpack-plugin | |
new HtmlWebpackPlugin({ | |
filename: 'index.html', | |
template: 'index.html', | |
inject: true | |
}), | |
] | |
}) | |
module.exports = new Promise((resolve, reject) => { | |
portfinder.basePort = process.env.PORT || config.dev.port | |
portfinder.getPort((err, port) => { | |
if (err) { | |
reject(err) | |
} else { | |
// publish the new Port, necessary for e2e tests | |
process.env.PORT = port | |
// add port to devServer config | |
devWebpackConfig.devServer.port = port | |
// Add FriendlyErrorsPlugin | |
devWebpackConfig.plugins.push(new FriendlyErrorsPlugin({ | |
compilationSuccessInfo: { | |
messages: [`Your application is running here: http://${devWebpackConfig.devServer.host}:${port}`], | |
}, | |
onErrors: config.dev.notifyOnErrors | |
? utils.createNotifierCallback() | |
: undefined | |
})) | |
resolve(devWebpackConfig) | |
} | |
}) | |
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
'use strict' | |
const path = require('path') | |
const utils = require('./utils') | |
const webpack = require('webpack') | |
const config = require('../config') | |
const merge = require('webpack-merge') | |
const baseWebpackConfig = require('./webpack.base.conf') | |
const CopyWebpackPlugin = require('copy-webpack-plugin') | |
const HtmlWebpackPlugin = require('html-webpack-plugin') | |
const ExtractTextPlugin = require('extract-text-webpack-plugin') | |
const OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin') | |
const UglifyJsPlugin = require('uglifyjs-webpack-plugin') | |
const PrerenderSpaPlugin = require('prerender-spa-plugin') | |
module.exports = { | |
// ... | |
plugins: [ | |
] | |
} | |
const env = require('../config/prod.env') | |
const webpackConfig = merge(baseWebpackConfig, { | |
module: { | |
rules: utils.styleLoaders({ | |
sourceMap: config.build.productionSourceMap, | |
extract: true, | |
usePostCSS: true | |
}) | |
}, | |
devtool: config.build.productionSourceMap ? config.build.devtool : false, | |
output: { | |
path: config.build.assetsRoot, | |
filename: utils.assetsPath('js/[name].[chunkhash].js'), | |
chunkFilename: utils.assetsPath('js/[id].[chunkhash].js') | |
}, | |
plugins: [ | |
new PrerenderSpaPlugin({ | |
// Absolute path to compiled SPA | |
staticDir: path.join(__dirname, '../dist'), | |
// Optional - The location of index.html | |
indexPath: path.join(__dirname, '../dist', 'index.html'), | |
// Required - Routes to render. | |
routes:[ '/', '/about', '/engineer', '/thinker', '/journo', '/thinker/read/' ] | |
}), | |
// http://vuejs.github.io/vue-loader/en/workflow/production.html | |
new webpack.DefinePlugin({ | |
'process.env': env | |
}), | |
new UglifyJsPlugin({ | |
uglifyOptions: { | |
compress: { | |
warnings: false | |
} | |
}, | |
sourceMap: config.build.productionSourceMap, | |
parallel: true | |
}), | |
// extract css into its own file | |
new ExtractTextPlugin({ | |
filename: utils.assetsPath('css/[name].[contenthash].css'), | |
// Setting the following option to `false` will not extract CSS from codesplit chunks. | |
// Their CSS will instead be inserted dynamically with style-loader when the codesplit chunk has been loaded by webpack. | |
// It's currently set to `true` because we are seeing that sourcemaps are included in the codesplit bundle as well when it's `false`, | |
// increasing file size: https://github.com/vuejs-templates/webpack/issues/1110 | |
allChunks: true, | |
}), | |
// Compress extracted CSS. We are using this plugin so that possible | |
// duplicated CSS from different components can be deduped. | |
new OptimizeCSSPlugin({ | |
cssProcessorOptions: config.build.productionSourceMap | |
? { safe: true, map: { inline: false } } | |
: { safe: true } | |
}), | |
// generate dist index.html with correct asset hash for caching. | |
// you can customize output by editing /index.html | |
// see https://github.com/ampedandwired/html-webpack-plugin | |
new HtmlWebpackPlugin({ | |
filename: config.build.index, | |
template: 'index.html', | |
inject: true, | |
minify: { | |
removeComments: true, | |
collapseWhitespace: true, | |
removeAttributeQuotes: true | |
// more options: | |
// https://github.com/kangax/html-minifier#options-quick-reference | |
}, | |
// necessary to consistently work with multiple chunks via CommonsChunkPlugin | |
chunksSortMode: 'dependency' | |
}), | |
// keep module.id stable when vender modules does not change | |
new webpack.HashedModuleIdsPlugin(), | |
// enable scope hoisting | |
new webpack.optimize.ModuleConcatenationPlugin(), | |
// split vendor js into its own file | |
new webpack.optimize.CommonsChunkPlugin({ | |
name: 'vendor', | |
minChunks (module) { | |
// any required modules inside node_modules are extracted to vendor | |
return ( | |
module.resource && | |
/\.js$/.test(module.resource) && | |
module.resource.indexOf( | |
path.join(__dirname, '../node_modules') | |
) === 0 | |
) | |
} | |
}), | |
// extract webpack runtime and module manifest to its own file in order to | |
// prevent vendor hash from being updated whenever app bundle is updated | |
new webpack.optimize.CommonsChunkPlugin({ | |
name: 'manifest', | |
minChunks: Infinity | |
}), | |
// This instance extracts shared chunks from code splitted chunks and bundles them | |
// in a separate chunk, similar to the vendor chunk | |
// see: https://webpack.js.org/plugins/commons-chunk-plugin/#extra-async-commons-chunk | |
new webpack.optimize.CommonsChunkPlugin({ | |
name: 'app', | |
async: 'vendor-async', | |
children: true, | |
minChunks: 3 | |
}), | |
// copy custom static assets | |
new CopyWebpackPlugin([ | |
{ | |
from: path.resolve(__dirname, '../static'), | |
to: config.build.assetsSubDirectory, | |
ignore: ['.*'] | |
} | |
]) | |
] | |
}) | |
if (config.build.productionGzip) { | |
const CompressionWebpackPlugin = require('compression-webpack-plugin') | |
webpackConfig.plugins.push( | |
new CompressionWebpackPlugin({ | |
asset: '[path].gz[query]', | |
algorithm: 'gzip', | |
test: new RegExp( | |
'\\.(' + | |
config.build.productionGzipExtensions.join('|') + | |
')$' | |
), | |
threshold: 10240, | |
minRatio: 0.8 | |
}) | |
) | |
} | |
if (config.build.bundleAnalyzerReport) { | |
const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin | |
webpackConfig.plugins.push(new BundleAnalyzerPlugin()) | |
} | |
module.exports = webpackConfig |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment