// 检查 Node 和 npm 版本 require('./check-versions')() //使用了 config/index.js var config = require('../config') // 如果 Node 的环境无法判断当前是 dev / product 环境 if (!process.env.NODE_ENV) { // 使用 config.dev.env.NODE_ENV 作为当前的环境 process.env.NODE_ENV = JSON.parse(config.dev.env.NODE_ENV) } // 一个可以强制打开浏览器并跳转到指定 url 的插件 //(可以调用默认软件打开网址、图片、文件等内容的插件, //这里用它来调用默认浏览器打开dev-server监听的端口,例如:localhost:8080) var opn = require('opn') // 使用 NodeJS 自带的文件路径工具 var path = require('path') // 使用 express var express = require('express') // 使用 webpack var webpack = require('webpack') // http-proxy可以实现转发所有请求代理到后端真实API地址,以实现前后端开发完全分离 // 在config/index.js中可以对proxyTable想进行配置 var proxyMiddleware = require('http-proxy-middleware') // 根据 Node 环境来引入相应的 webpack 配置 var webpackConfig = process.env.NODE_ENV === 'testing' ? require('./webpack.prod.conf') : require('./webpack.dev.conf')
// default port where dev server listens for incoming traffic // 如果没有指定运行端口,使用 config.dev.port 作为运行端口 var port = process.env.PORT || config.dev.port // automatically open browser, if not set will be false // 用于判断是否要自动打开浏览器的布尔变量,当配置文件中没有设置自动打开浏览器的时候其值为 false var autoOpenBrowser = !!config.dev.autoOpenBrowser
// Define HTTP proxies to your custom API backend // https://github.com/chimurai/http-proxy-middleware // 定义 HTTP 代理表,代理到 API 服务器 // 使用 config.dev.proxyTable 的配置作为 proxyTable 的代理配置 var proxyTable = config.dev.proxyTable // 使用 express 启动一个服务 var app = express() // 启动 webpack 进行编译 var compiler = webpack(webpackConfig) // 启动 webpack-dev-middleware,将 编译后的文件暂存到内存中 //(webpack-dev-middleware使用compiler对象来对相应的文件进行编译和绑定 // 编译绑定后将得到的产物存放在内存中而没有写进磁盘 // 将这个中间件交给express使用之后即可访问这些编译后的产品文件) var devMiddleware = require('webpack-dev-middleware')(compiler, { publicPath: webpackConfig.output.publicPath, quiet: true }) // 启动 webpack-hot-middleware,也就是我们常说的 Hot-reload用于实现热重载功能的中间件 var hotMiddleware = require('webpack-hot-middleware')(compiler, { log: () => {}, heartbeat: 2000 }) // force page reload when html-webpack-plugin template changes // 当html-webpack-plugin提交之后通过热重载中间件发布重载动作使得页面重载 compiler.plugin('compilation', function (compilation) { compilation.plugin('html-webpack-plugin-after-emit', function (data, cb) { hotMiddleware.publish({ action: 'reload' }) cb() }) })
// handle fallback for HTML5 history API // 使用 connect-history-api-fallback 匹配资源,如果不匹配就可以重定向到指定地址,常用于SPA app.use(require('connect-history-api-fallback')())
// serve pure static assets // 拼接 static 文件夹的静态资源路径 var staticPath = path.posix.join(config.dev.assetsPublicPath, config.dev.assetsSubDirectory) // 为静态资源提供响应服务 app.use(staticPath, express.static('./static')) // 应用的地址信息,例如:http://localhost:8080 var uri = 'http://localhost:' + port
var _resolve var readyPromise = new Promise(resolve => { _resolve = resolve })
console.log('> Starting dev server...')
// webpack开发中间件合法(valid)之后输出提示语到控制台,表明服务器已启动 devMiddleware.waitUntilValid(() => { console.log('> Listening at ' + uri + '\n') // when env is testing, don't need open it // 如果不是测试环境,自动打开浏览器并跳到我们的开发地址 if (autoOpenBrowser && process.env.NODE_ENV !== 'testing') { opn(uri) } _resolve() }) //监听服务器端口 var server = app.listen(port)
// 检查 Node 和 npm 版本 require('./check-versions')() //生产环境 process.env.NODE_ENV = 'production' // 一个很好看的 loading 插件 var ora = require('ora') var rm = require('rimraf') // 使用 NodeJS 自带的文件路径插件 var path = require('path') // 用于在控制台输出带颜色字体的插件 var chalk = require('chalk') //加载webpack var webpack = require('webpack') //加载config中index.js var config = require('../config') //加载webpack.prod.conf var webpackConfig = require('./webpack.prod.conf')
var spinner = ora('building for production...') spinner.start()// 开启loading动画 // 拼接编译输出文件路径 var assetsPath = path.join(config.build.assetsRoot, config.build.assetsSubDirectory); // 删除这个文件夹 (递归删除) rm(assetsPath, err => { if (err) throw err // 开始 webpack 的编译 webpack(webpackConfig, function (err, stats) { // 编译成功的回调函数 spinner.stop() if (err) throw err process.stdout.write(stats.toString({ colors: true, modules: false, children: false, chunks: false, chunkModules: false }) + '\n\n')
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' )) }) })
module.exports = function () { var warnings = [] // 依次判断版本是否符合要求 for (var i = 0; i < versionRequirements.length; i++) { var mod = versionRequirements[i] if (!semver.satisfies(mod.currentVersion, mod.versionRequirement)) { warnings.push(mod.name + ': ' + chalk.red(mod.currentVersion) + ' should be ' + chalk.green(mod.versionRequirement) ) } } // 如果有警告则将其输出到控制台 if (warnings.length) { console.log(chalk.yellow('To use this template, you must update following to modules:')) for (var i = 0; i < warnings.length; i++) { var warning = warnings[i] console.log(' ' + warning) } process.exit(1) } }
// 使用 NodeJS 自带的文件路径插件 var path = require('path') // 使用一些小工具 var utils = require('./utils') // 加载 webpack var webpack = require('webpack') // 加载 confi.index.js var config = require('../config') // 加载 webpack 配置合并工具 var merge = require('webpack-merge') // 加载 webpack.base.conf.js var baseWebpackConfig = require('./webpack.base.conf') //使用copy-webpack-plugin插件 var CopyWebpackPlugin = require('copy-webpack-plugin') // 使用 html-webpack-plugin 插件,这个插件可以帮我们自动生成 html 并且注入到 .html 文件中 var HtmlWebpackPlugin = require('html-webpack-plugin')
// 用于从webpack生成的bundle中提取文本到特定文件中的插件 // 可以抽取出css,js文件将其与webpack输出的bundle分离 var ExtractTextPlugin = require('extract-text-webpack-plugin')
//使用js,css压缩插件 var OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin')
//判断当前环境是否为测试环境,如是加载测试环境配置文件,否则使用config.build.env var env = process.env.NODE_ENV === 'testing' ? require('../config/test.env') : config.build.env // 合并 webpack.base.conf.js var webpackConfig = merge(baseWebpackConfig, { module: { // 使用的 loader rules: utils.styleLoaders({ sourceMap: config.build.productionSourceMap, extract: true }) }, // 是否使用 #source-map 开发工具 devtool: config.build.productionSourceMap ? '#source-map' : false, output: { // 编译输出目录 path: config.build.assetsRoot, // 编译输出文件名 // 我们可以在 hash 后加 :6 决定使用几位 hash 值 filename: utils.assetsPath('js/[name].[chunkhash].js'), // 没有指定输出名的文件输出的文件名 chunkFilename: utils.assetsPath('js/[id].[chunkhash].js') }, // 使用的插件 plugins: [ // http://vuejs.github.io/vue-loader/en/workflow/production.html // definePlugin 接收字符串插入到代码当中, 所以你需要的话可以写上 JS 的字符串 new webpack.DefinePlugin({ 'process.env': env }), // 压缩 js (同样可以压缩 css) new webpack.optimize.UglifyJsPlugin({ compress: { warnings: false }, sourceMap: true }), // extract css into its own file // 将 css 文件分离出来 new ExtractTextPlugin({ filename: utils.assetsPath('css/[name].[contenthash].css') }), // Compress extracted CSS. We are using this plugin so that possible // duplicated CSS from different components can be deduped. new OptimizeCSSPlugin({ cssProcessorOptions: { 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 // 输入输出的 .html 文件 new HtmlWebpackPlugin({ filename: process.env.NODE_ENV === 'testing' ? 'index.html' : config.build.index, template: 'index.html', // 是否注入 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' }), // split vendor js into its own file // 没有指定输出文件名的文件输出的静态文件名 new webpack.optimize.CommonsChunkPlugin({ name: 'vendor', minChunks: function (module, count) { // 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', chunks: ['vendor'] }), // copy custom static assets new CopyWebpackPlugin([ { from: path.resolve(__dirname, '../static'), to: config.build.assetsSubDirectory, ignore: ['.*'] } ]) ] }) // 开启 gzip 的情况下使用下方的配置,引入compression插件进行压缩 if (config.build.productionGzip) { var CompressionWebpackPlugin = require('compression-webpack-plugin') // 加载 compression-webpack-plugin 插件 var reProductionGzipExtensions = '\\.(' + config.build.productionGzipExtensions.join('|') + ')$'; // 使用 compression-webpack-plugin 插件进行压缩 webpackConfig.plugins.push( new CompressionWebpackPlugin({ asset: '[path].gz[query]', algorithm: 'gzip', test: new RegExp(reProductionGzipExtensions), threshold: 10240, minRatio: 0.8 }) ) }
if (config.build.bundleAnalyzerReport) { var BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin webpackConfig.plugins.push(new BundleAnalyzerPlugin()) }
module.exports = { // production 环境 build: { // 使用 config/prod.env.js 中定义的编译环境 env: require('./prod.env'), // 编译输入的 index.html 文件 index: path.resolve(__dirname, '../dist/index.html'), // 编译输出的静态资源根路径 assetsRoot: path.resolve(__dirname, '../dist'), // 编译输出的二级目录 assetsSubDirectory: 'static', // 编译发布上线路径的根目录,可配置为资源服务器域名或 CDN 域名 assetsPublicPath: './', // 是否开启 cssSourceMap productionSourceMap: true, // Gzip off by default as many popular static hosts such as // Surge or Netlify already gzip all static assets for you. // Before setting to `true`, make sure to: // npm install --save-dev compression-webpack-plugin // 是否开启 gzip,默认不开启 productionGzip: false, // gzip模式下需要压缩的文件的扩展名 productionGzipExtensions: ['js', 'css'], // Run the build command with an extra argument to // View the bundle analyzer report after build finishes: // `npm run build --report` // Set to `true` or `false` to always turn it on or off bundleAnalyzerReport: process.env.npm_config_report }, // dev 环境 dev: { // 使用 config/dev.env.js 中定义的编译环境 env: require('./dev.env'), // 运行测试页面的端口 port: 8087, // 启动dev-server之后自动打开浏览器 autoOpenBrowser: true, // 编译输出的二级目录 assetsSubDirectory: 'static', // 编译发布上线路径的根目录,可配置为资源服务器域名或 CDN 域名 assetsPublicPath: '/', // 需要 proxyTable 代理的接口(可跨域) proxyTable: {}, //静态网址 /*proxyTable: {
// CSS Sourcemaps off by default because relative paths are "buggy" // with this option, according to the CSS-Loader README // (https://github.com/webpack/css-loader#sourcemaps) // In our experience, they generally work as expected, // just be aware of this issue when enabling this option. // 是否开启 cssSourceMap cssSourceMap: false } }