2018-01-30 11:53:12 -05:00
|
|
|
const defaults = require('lodash.defaults');
|
|
|
|
const gitsha = require('git-bundle-sha');
|
|
|
|
const path = require('path');
|
|
|
|
const webpack = require('webpack');
|
2015-10-13 15:40:51 -04:00
|
|
|
|
2021-02-19 12:51:39 -05:00
|
|
|
// Plugins
|
|
|
|
const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin;
|
|
|
|
const CopyWebpackPlugin = require('copy-webpack-plugin');
|
2024-01-18 15:37:11 -05:00
|
|
|
const EmitFilePlugin = require('emit-file-webpack-plugin');
|
2021-02-19 12:51:39 -05:00
|
|
|
const HtmlWebpackPlugin = require('html-webpack-plugin');
|
2021-07-09 11:09:51 -04:00
|
|
|
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
|
2022-01-28 16:56:46 -05:00
|
|
|
const TerserPlugin = require('terser-webpack-plugin');
|
2021-02-19 12:51:39 -05:00
|
|
|
|
|
|
|
// PostCss
|
|
|
|
const autoprefixer = require('autoprefixer');
|
|
|
|
|
2024-01-17 15:40:27 -05:00
|
|
|
/** @type {Array} */
|
2018-01-30 11:53:12 -05:00
|
|
|
let routes = require('./src/routes.json');
|
2024-01-17 15:40:27 -05:00
|
|
|
const templateConfig = require('./src/template-config.js');
|
2016-04-12 08:23:43 -04:00
|
|
|
|
2016-05-11 18:44:54 -04:00
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
2018-01-30 11:53:12 -05:00
|
|
|
routes = routes.concat(require('./src/routes-dev.json')); // eslint-disable-line global-require
|
2016-05-11 18:44:54 -04:00
|
|
|
}
|
|
|
|
|
2018-03-07 15:16:46 -05:00
|
|
|
routes = routes.filter(route => !process.env.VIEW || process.env.VIEW === route.view);
|
|
|
|
|
2024-01-17 15:40:27 -05:00
|
|
|
const pageRoutes = routes.filter(route => !route.redirect);
|
2018-01-30 11:53:12 -05:00
|
|
|
|
2024-01-18 15:37:11 -05:00
|
|
|
/**
|
|
|
|
* Retrieve a version ID string for the current build, to be emitted into `version.txt`.
|
|
|
|
* @returns {Promise<string>} A promise that resolves to a version ID string.
|
|
|
|
*/
|
|
|
|
const getVersionId = () => {
|
|
|
|
if (process.env.WWW_VERSION) {
|
|
|
|
return Promise.resolve(process.env.WWW_VERSION);
|
|
|
|
}
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
gitsha({length: 5}, (err, sha) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else {
|
|
|
|
resolve(sha);
|
2018-01-30 11:53:12 -05:00
|
|
|
}
|
2024-01-18 15:37:11 -05:00
|
|
|
});
|
2016-04-28 15:40:50 -04:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-09-04 01:26:56 -04:00
|
|
|
// Prepare all entry points
|
2024-01-17 15:40:27 -05:00
|
|
|
const entry = {};
|
2021-02-19 12:51:39 -05:00
|
|
|
|
2024-01-17 15:40:27 -05:00
|
|
|
pageRoutes.forEach(route => {
|
2021-02-19 12:51:39 -05:00
|
|
|
entry[route.name] = [
|
|
|
|
'./src/init.js',
|
|
|
|
`./src/views/${route.view}.jsx`
|
|
|
|
];
|
2015-09-04 01:26:56 -04:00
|
|
|
});
|
|
|
|
|
2024-01-18 14:54:47 -05:00
|
|
|
// HtmlWebpackPlugin v4 removed 'chunks' info that we need for our custom template.
|
|
|
|
// This plugin is a quick-and-dirty partial implementation of that information.
|
|
|
|
// Adapted from https://github.com/jantimon/html-webpack-plugin/issues/1369#issuecomment-1049968234
|
|
|
|
// Thanks, @daniel-nagy!
|
|
|
|
class HtmlWebpackBackwardsCompatibilityPlugin {
|
|
|
|
apply (compiler) {
|
|
|
|
compiler
|
|
|
|
.hooks
|
|
|
|
.compilation
|
|
|
|
.tap('HtmlWebpackBackwardsCompatibilityPlugin', compilation => {
|
|
|
|
HtmlWebpackPlugin
|
|
|
|
.getHooks(compilation)
|
|
|
|
.beforeAssetTagGeneration
|
|
|
|
.tapAsync(
|
|
|
|
'HtmlWebpackBackwardsCompatibilityPlugin',
|
|
|
|
(data, callback) => {
|
|
|
|
const {publicPath} = data.assets;
|
|
|
|
const chunks = {};
|
|
|
|
|
|
|
|
for (const entryPoint of compilation.entrypoints.values()) {
|
|
|
|
for (const chunk of entryPoint.chunks) {
|
2024-01-18 19:03:30 -05:00
|
|
|
const files = Array.from(chunk.files); // convert from Set
|
2024-01-18 14:54:47 -05:00
|
|
|
chunks[chunk.name] = {
|
2024-01-18 19:03:30 -05:00
|
|
|
entry: publicPath + files.find(file => file.endsWith('.js')),
|
|
|
|
css: files
|
2024-01-18 14:54:47 -05:00
|
|
|
.filter(file => file.endsWith('.css'))
|
|
|
|
.map(file => publicPath + file)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
data.assets.chunks = chunks;
|
|
|
|
|
|
|
|
callback(null, data);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-04 01:26:56 -04:00
|
|
|
// Config
|
2015-09-02 15:08:58 -04:00
|
|
|
module.exports = {
|
2015-09-04 01:26:56 -04:00
|
|
|
entry: entry,
|
2024-01-18 16:03:34 -05:00
|
|
|
devtool: process.env.NODE_ENV === 'production' ? false : 'eval',
|
2021-02-19 12:51:39 -05:00
|
|
|
mode: process.env.NODE_ENV === 'production' ? 'production' : 'development',
|
2015-09-02 15:08:58 -04:00
|
|
|
output: {
|
2015-11-10 13:38:28 -05:00
|
|
|
path: path.resolve(__dirname, 'build'),
|
2021-07-09 09:35:50 -04:00
|
|
|
filename: 'js/[name].bundle.js',
|
|
|
|
publicPath: '/'
|
2015-09-02 15:08:58 -04:00
|
|
|
},
|
2021-02-19 10:43:15 -05:00
|
|
|
resolve: {
|
2024-01-18 16:03:34 -05:00
|
|
|
fallback: {
|
2024-02-26 23:21:30 -05:00
|
|
|
// Node modules are no longer polyfilled by default in Webpack 5, so we need to add these here
|
|
|
|
buffer: require.resolve('buffer/'),
|
|
|
|
stream: require.resolve('stream-browserify') // jszip
|
2024-01-18 16:03:34 -05:00
|
|
|
},
|
2021-02-19 10:43:15 -05:00
|
|
|
symlinks: false // Fix local development with `npm link` packages
|
|
|
|
},
|
2015-09-02 15:08:58 -04:00
|
|
|
module: {
|
2018-01-30 11:53:12 -05:00
|
|
|
rules: [
|
2015-09-02 15:08:58 -04:00
|
|
|
{
|
2024-01-22 13:43:07 -05:00
|
|
|
test: /\.(?:js|mjs|cjs)x?$/,
|
2018-01-30 11:53:12 -05:00
|
|
|
loader: 'babel-loader',
|
2019-03-07 09:22:53 -05:00
|
|
|
include: [
|
|
|
|
path.resolve(__dirname, 'src'),
|
|
|
|
/node_modules[\\/]scratch-[^\\/]+[\\/]src/,
|
2019-07-12 11:56:34 -04:00
|
|
|
/node_modules[\\/]pify/,
|
|
|
|
/node_modules[\\/]async/
|
2019-03-07 09:22:53 -05:00
|
|
|
]
|
2015-09-02 15:08:58 -04:00
|
|
|
},
|
|
|
|
{
|
|
|
|
test: /\.scss$/,
|
2018-01-30 11:53:12 -05:00
|
|
|
use: [
|
2021-07-09 11:09:51 -04:00
|
|
|
MiniCssExtractPlugin.loader,
|
2022-03-02 10:55:33 -05:00
|
|
|
{
|
|
|
|
loader: 'css-loader',
|
|
|
|
options: {
|
|
|
|
url: false
|
|
|
|
}
|
|
|
|
},
|
2018-01-30 11:53:12 -05:00
|
|
|
{
|
|
|
|
loader: 'postcss-loader',
|
|
|
|
options: {
|
2022-01-26 15:50:40 -05:00
|
|
|
postcssOptions: {
|
|
|
|
plugins: function () {
|
2022-02-14 10:17:39 -05:00
|
|
|
return [autoprefixer()];
|
2022-01-26 15:50:40 -05:00
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'sass-loader'
|
|
|
|
]
|
2015-09-08 14:56:28 -04:00
|
|
|
},
|
2016-06-02 15:25:02 -04:00
|
|
|
{
|
|
|
|
test: /\.css$/,
|
2018-01-30 11:53:12 -05:00
|
|
|
use: [
|
2021-07-09 11:09:51 -04:00
|
|
|
MiniCssExtractPlugin.loader,
|
2022-03-02 10:55:33 -05:00
|
|
|
{
|
|
|
|
loader: 'css-loader',
|
|
|
|
options: {
|
|
|
|
url: false
|
|
|
|
}
|
|
|
|
},
|
2018-01-30 11:53:12 -05:00
|
|
|
{
|
|
|
|
loader: 'postcss-loader',
|
|
|
|
options: {
|
2022-01-26 15:50:40 -05:00
|
|
|
postcssOptions: {
|
|
|
|
plugins: function () {
|
2022-02-14 10:17:39 -05:00
|
|
|
return [autoprefixer()];
|
2022-01-26 15:50:40 -05:00
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
2016-06-02 15:25:02 -04:00
|
|
|
},
|
2015-09-08 14:56:28 -04:00
|
|
|
{
|
|
|
|
test: /\.(png|jpg|gif|eot|svg|ttf|woff)$/,
|
|
|
|
loader: 'url-loader'
|
2015-09-02 15:08:58 -04:00
|
|
|
}
|
2016-06-02 15:25:02 -04:00
|
|
|
],
|
|
|
|
noParse: /node_modules\/google-libphonenumber\/dist/
|
2015-09-02 16:33:31 -04:00
|
|
|
},
|
2021-02-19 12:51:39 -05:00
|
|
|
optimization: {
|
|
|
|
splitChunks: {
|
|
|
|
cacheGroups: {
|
|
|
|
common: {
|
|
|
|
chunks: 'all',
|
|
|
|
name: 'common',
|
2024-01-18 16:03:34 -05:00
|
|
|
minSize: 1024,
|
2021-02-19 12:51:39 -05:00
|
|
|
minChunks: pageRoutes.length // Extract only chunks common to all html pages
|
|
|
|
}
|
|
|
|
}
|
2022-01-28 16:56:46 -05:00
|
|
|
},
|
|
|
|
minimizer: [
|
|
|
|
new TerserPlugin({
|
|
|
|
parallel: 4
|
|
|
|
})
|
|
|
|
]
|
2021-02-19 12:51:39 -05:00
|
|
|
},
|
2015-09-02 16:33:31 -04:00
|
|
|
plugins: [
|
2021-07-09 11:09:51 -04:00
|
|
|
new MiniCssExtractPlugin(),
|
2024-01-18 14:54:47 -05:00
|
|
|
new HtmlWebpackBackwardsCompatibilityPlugin(),
|
2024-01-18 15:37:11 -05:00
|
|
|
new EmitFilePlugin({
|
|
|
|
filename: 'version.txt',
|
|
|
|
content: getVersionId
|
|
|
|
})
|
2021-02-19 12:51:39 -05:00
|
|
|
].concat(pageRoutes
|
2024-01-17 15:40:27 -05:00
|
|
|
.map(route => new HtmlWebpackPlugin(defaults({}, {
|
|
|
|
title: route.title,
|
|
|
|
filename: `${route.name}.html`,
|
|
|
|
route: route,
|
|
|
|
dynamicMetaTags: route.dynamicMetaTags
|
|
|
|
}, templateConfig)))
|
2018-03-19 11:58:03 -04:00
|
|
|
).concat([
|
2022-02-02 16:58:48 -05:00
|
|
|
new CopyWebpackPlugin({
|
|
|
|
patterns: [
|
|
|
|
{from: 'static'},
|
2023-06-26 12:12:05 -04:00
|
|
|
{from: 'intl', to: 'js'},
|
2022-02-02 16:58:48 -05:00
|
|
|
{
|
|
|
|
from: 'node_modules/scratch-gui/dist/static/blocks-media',
|
|
|
|
to: 'static/blocks-media'
|
2023-06-26 12:12:05 -04:00
|
|
|
},
|
2022-02-02 16:58:48 -05:00
|
|
|
{
|
|
|
|
from: 'node_modules/scratch-gui/dist/chunks',
|
|
|
|
to: 'static/chunks'
|
2023-06-26 12:12:05 -04:00
|
|
|
},
|
2022-02-02 16:58:48 -05:00
|
|
|
{
|
|
|
|
from: 'node_modules/scratch-gui/dist/extension-worker.js'
|
2023-06-26 12:12:05 -04:00
|
|
|
},
|
2022-02-02 16:58:48 -05:00
|
|
|
{
|
|
|
|
from: 'node_modules/scratch-gui/dist/extension-worker.js.map'
|
2023-06-26 12:12:05 -04:00
|
|
|
},
|
2022-02-02 16:58:48 -05:00
|
|
|
{
|
|
|
|
from: 'node_modules/scratch-gui/dist/static/assets',
|
|
|
|
to: 'static/assets'
|
2023-06-26 12:12:46 -04:00
|
|
|
},
|
|
|
|
{
|
|
|
|
from: 'node_modules/scratch-gui/dist/*.hex',
|
|
|
|
to: 'static',
|
|
|
|
flatten: true
|
2022-02-02 16:58:48 -05:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}),
|
2021-02-19 12:51:39 -05:00
|
|
|
new webpack.DefinePlugin({
|
2024-01-17 15:40:27 -05:00
|
|
|
'process.env.NODE_ENV': `"${process.env.NODE_ENV || 'development'}"`,
|
|
|
|
'process.env.API_HOST': `"${process.env.API_HOST || 'https://api.scratch.mit.edu'}"`,
|
|
|
|
'process.env.RECAPTCHA_SITE_KEY': `"${
|
|
|
|
process.env.RECAPTCHA_SITE_KEY || '6Lf6kK4UAAAAABKTyvdSqgcSVASEnMrCquiAkjVW'}"`,
|
|
|
|
'process.env.ASSET_HOST': `"${process.env.ASSET_HOST || 'https://assets.scratch.mit.edu'}"`,
|
|
|
|
'process.env.BACKPACK_HOST': `"${process.env.BACKPACK_HOST || 'https://backpack.scratch.mit.edu'}"`,
|
|
|
|
'process.env.CLOUDDATA_HOST': `"${process.env.CLOUDDATA_HOST || 'clouddata.scratch.mit.edu'}"`,
|
|
|
|
'process.env.PROJECT_HOST': `"${process.env.PROJECT_HOST || 'https://projects.scratch.mit.edu'}"`,
|
|
|
|
'process.env.STATIC_HOST': `"${process.env.STATIC_HOST || 'https://uploads.scratch.mit.edu'}"`,
|
2024-05-10 05:23:07 -04:00
|
|
|
'process.env.SCRATCH_ENV': `"${process.env.SCRATCH_ENV || 'development'}"`,
|
|
|
|
'process.env.THUMBNAIL_URI': `"${process.env.THUMBNAIL_URI || '/internalapi/project/thumbnail/{}/set/'}"`,
|
|
|
|
'process.env.THUMBNAIL_HOST': `"${process.env.THUMBNAIL_HOST || ''}"`
|
2021-02-19 12:51:39 -05:00
|
|
|
})
|
2018-02-08 12:01:07 -05:00
|
|
|
])
|
2021-02-19 12:51:39 -05:00
|
|
|
.concat(process.env.ANALYZE_BUNDLE === 'true' ? [
|
|
|
|
new BundleAnalyzerPlugin()
|
2018-03-19 11:58:03 -04:00
|
|
|
] : [])
|
2015-09-02 15:08:58 -04:00
|
|
|
};
|