embark/lib/pipeline/webpack.config.js

188 lines
6.1 KiB
JavaScript
Raw Normal View History

2018-08-16 15:49:19 -05:00
/* global module process require */
2018-08-17 12:32:58 -05:00
// NOTE: some packages, plugins, and presets referenced/required in this webpack
2018-08-16 15:49:19 -05:00
// config are deps of embark and will be transitive dapp deps unless specified
2018-08-16 19:51:56 -05:00
// in the dapp's own package.json, perhaps with a different version/range or
// other specifier
2018-08-16 15:49:19 -05:00
2018-08-17 12:32:58 -05:00
// NOTE: embark modifies process.env.NODE_PATH so that embark's own
// node_modules directory will be searched by node's require(). That allows an
// ejected version of this config, inside a dapp, to function correctly without
// embark being an explicit dependency of the dapp. However, webpack and babel
// do not directly support NODE_PATH, so in various parts of the config,
// modules such as plugins and presets must be resolved with
2018-08-17 16:45:25 -05:00
// require.resolve(). That is only necessary if a plugin/preset is in embark's
2018-08-17 12:32:58 -05:00
// node_modules; if it's in the dapp's node_modules then it can be specified
// directly
2018-08-16 15:49:19 -05:00
// there's a bug in pkg clone-deep re: regex; for now use lodash.clonedeep
// see: https://github.com/jonschlinkert/clone-deep/pull/14
const cloneDeep = require('lodash.clonedeep');
const CompressionPlugin = require('compression-webpack-plugin');
2018-08-16 15:49:19 -05:00
const glob = require('glob');
const HardSourceWebpackPlugin = require('hard-source-webpack-plugin');
const path = require('path');
const dappPath = process.env.DAPP_PATH;
const embarkPath = process.env.EMBARK_PATH;
2018-08-16 15:49:19 -05:00
const embarkAliases = require(path.join(dappPath, '.embark/embark-aliases.json'));
const embarkAssets = require(path.join(dappPath, '.embark/embark-assets.json'));
2018-08-17 17:55:59 -05:00
const embarkNodeModules = path.join(embarkPath, 'node_modules');
2018-08-17 16:45:35 -05:00
const embarkJson = require(path.join(dappPath, 'embark.json'));
2018-08-16 15:49:19 -05:00
2018-08-17 17:55:59 -05:00
const buildDir = path.join(dappPath, embarkJson.buildDir);
2018-08-16 15:49:19 -05:00
// it's important to `embark reset` if a pkg version is specified in
// embark.json and changed/removed later, otherwise pkg resolution may behave
// unexpectedly, e.g. a different version may get used owing to lexographical
// order or because .embark/versions is still in place even though defaults are
2018-08-16 19:51:56 -05:00
// desired; once embark and its dependencies are always resolvable (in the
// usual node sense) relative to a dapp, embark's pkg version-management
// functionality can probably be deprecated in favor of a dapp simply
// specifying dependencies in its own package.json, e.g. a different ipfs
// version than embark's default
2018-08-16 15:49:19 -05:00
let versions;
try {
versions = glob.sync(path.join(dappPath, '.embark/versions/*/*'));
} catch (e) {
versions = [];
}
2018-08-17 16:45:47 -05:00
const entry = Object.keys(embarkAssets)
.filter(key => key.match(/\.js?$/))
.reduce((obj, key) => {
obj[key] = embarkAssets[key]
.map(file => {
let file_path = file.path;
if (!file.path.match(/^\.\//)) {
file_path = './' + file_path;
}
return file_path;
});
return obj;
}, {});
2018-08-16 15:49:19 -05:00
2018-08-18 19:55:34 -05:00
function resolve(pkgName) {
if (Array.isArray(pkgName)) {
const _pkgName = pkgName[0];
pkgName[0] = require.resolve(_pkgName);
return pkgName;
} else {
return require.resolve(pkgName);
}
}
2018-08-16 15:49:19 -05:00
// base config
// -----------------------------------------------------------------------------
const base = {
context: dappPath,
entry: entry,
module: {
rules: [
{
test: /\.css$/,
use: [{loader: 'style-loader'}, {loader: 'css-loader'}]
},
{
test: /\.scss$/,
use: [{loader: 'style-loader'}, {loader: 'css-loader'}]
},
{
test: /\.(png|woff|woff2|eot|ttf|svg)$/,
loader: 'url-loader?limit=100000'
},
{
2018-08-16 16:49:32 -05:00
test: /\.jsx?$/,
2018-08-16 15:49:19 -05:00
loader: 'babel-loader',
exclude: /(node_modules|bower_components|\.embark[\\/]versions)/,
options: {
plugins: [
[
2018-08-18 19:55:34 -05:00
'babel-plugin-module-resolver', {
2018-08-16 15:49:19 -05:00
'alias': embarkAliases
}
],
[
2018-08-18 19:55:34 -05:00
'@babel/plugin-transform-runtime', {
2018-08-16 15:49:19 -05:00
corejs: 2,
useESModules: true
}
]
2018-08-18 19:55:34 -05:00
].map(resolve),
2018-08-16 15:49:19 -05:00
presets: [
[
2018-08-18 19:55:34 -05:00
'@babel/preset-env', {
2018-08-16 15:49:19 -05:00
modules: false,
targets: {
browsers: ['last 1 version', 'not dead', '> 0.2%']
}
}
],
2018-08-18 19:55:34 -05:00
'@babel/preset-react'
].map(resolve)
2018-08-16 15:49:19 -05:00
}
}
]
},
output: {
filename: (chunkData) => chunkData.chunk.name,
// globalObject workaround for node-compatible UMD builds with webpack 4
// see: https://github.com/webpack/webpack/issues/6522#issuecomment-371120689
globalObject: 'typeof self !== \'undefined\' ? self : this',
libraryTarget: 'umd',
2018-08-17 17:55:59 -05:00
path: buildDir
2018-08-16 15:49:19 -05:00
},
2018-08-17 17:14:28 -05:00
plugins: [new HardSourceWebpackPlugin()],
2018-08-16 15:49:19 -05:00
// profiling and generating verbose stats increases build time; if stats
// are generated embark will write the output to:
// path.join(dappPath, '.embark/stats.[json,report]')
// to visualize the stats info in a browser run:
// npx webpack-bundle-analyzer <dappPath>/.embark/stats.json
profile: true, stats: 'verbose',
resolve: {
alias: embarkAliases,
modules: [
...versions,
'node_modules',
2018-08-17 17:55:59 -05:00
embarkNodeModules
2018-08-16 15:49:19 -05:00
]
},
resolveLoader: {
modules: [
'node_modules',
2018-08-17 17:55:59 -05:00
embarkNodeModules
2018-08-17 17:14:28 -05:00
]
2018-08-16 15:49:19 -05:00
}
};
// development config
// -----------------------------------------------------------------------------
const development = cloneDeep(base);
// full source maps increase build time but are useful during dapp development
development.devtool = 'source-map';
2018-08-16 15:49:19 -05:00
development.mode = 'development';
2018-08-17 16:46:36 -05:00
// alternatively:
// development.mode = 'none';
2018-08-16 15:49:19 -05:00
development.name = 'development';
const devBabelLoader = development.module.rules[3];
devBabelLoader.options.compact = false;
// production config
// -----------------------------------------------------------------------------
const production = cloneDeep(base);
production.mode = 'production';
production.name = 'production';
production.plugins.push(new CompressionPlugin());
2018-08-16 15:49:19 -05:00
// export a list of named configs
// -----------------------------------------------------------------------------
module.exports = [
development,
production
];