Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix: ensure that assets are not in a hashed folder #366

Merged
merged 1 commit into from
Apr 9, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion index.js
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ if (process.env.NODE_ENV === 'production') {
const clientStats = require('./dist/client-stats.json');
const distPath = path.join(__dirname, 'dist');
app.use(
`${env.BASE_URL}/assets`,
`${env.BASE_URL}/`,
expressStaticGzip(distPath, {
maxAge: '1d',
}),
Expand Down
58 changes: 29 additions & 29 deletions webpack.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ export const serviceName = process.env.SERVICE_NAME || 'not set';
export const dist = path.join(__dirname, 'dist');

/** Webpack public path. All emitted assets will have relative path to this path */
export const publicPath = `${env.BASE_URL}/assets/`;
export const publicPath = `${env.BASE_URL}/`;

/** True if we are in development mode */
export const isDev = env.NODE_ENV === 'development';
Expand All @@ -53,7 +53,7 @@ export const resolve: webpack.ResolveOptions = {
/** Extension that are allowed to be omitted from import statements */
extensions: ['.ts', '.tsx', '.js', '.jsx'],
/** "main" fields in package.json files to resolve a CommonJS module for */
mainFields: ['browser', 'module', 'main']
mainFields: ['browser', 'module', 'main'],
};

/** Get clean version of a version string of package.json entry for a package by
Expand All @@ -68,10 +68,10 @@ export function absoluteVersion(version: string) {

/** CDN path in case we would use minified react and react-DOM */
const cdnReact = `https://unpkg.com/react@${absoluteVersion(
packageJson.devDependencies.react
packageJson.devDependencies.react,
)}/umd/react.production.min.js`;
const cdnReactDOM = `https://unpkg.com/react-dom@${absoluteVersion(
packageJson.devDependencies['react-dom']
packageJson.devDependencies['react-dom'],
)}/umd/react-dom.production.min.js`;

/** Minification options for HTMLWebpackPlugin */
Expand All @@ -80,20 +80,20 @@ export const htmlMinifyConfig: HTMLWebpackPlugin.MinifyOptions = {
minifyJS: false,
removeComments: true,
collapseInlineTagWhitespace: true,
collapseWhitespace: true
collapseWhitespace: true,
};

/** Adds sourcemap support */
export const sourceMapRule: webpack.RuleSetRule = {
test: /\.js$/,
enforce: 'pre',
use: ['source-map-loader']
use: ['source-map-loader'],
};

/** Rule for images, icons and fonts */
export const imageAndFontsRule: webpack.RuleSetRule = {
test: /\.(ico|jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2)(\?.*)?$/,
type: 'asset/resource'
type: 'asset/resource',
};

/** Generates HTML file that includes webpack assets */
Expand All @@ -102,26 +102,26 @@ export const htmlPlugin = new HTMLWebpackPlugin({
inject: 'body',
minify: isProd ? htmlMinifyConfig : false,
hash: false,
showErrors: isDev
showErrors: isDev,
});

export const favIconPlugin = new FavIconWebpackPlugin({
logo: path.resolve(__dirname, 'src/assets/favicon.png'), //'./src/assets/favicon.png', - narusina
prefix: '[fullhash:8]/'
logo: path.resolve(__dirname, 'src/assets/favicon.png'),
prefix: 'assets/', // we can add '[fullhash:8]/' to the end of the file in future
});

/** Write client stats to a JSON file for production */
export const statsWriterPlugin = new StatsWriterPlugin({
filename: 'client-stats.json',
fields: ['chunks', 'publicPath', 'assets', 'assetsByChunkName', 'assetsByChunkId']
fields: ['chunks', 'publicPath', 'assets', 'assetsByChunkName', 'assetsByChunkId'],
});

/** Gzip assets */
export const compressionPlugin = new CompressionWebpackPlugin({
algorithm: 'gzip',
test: /\.(js|css|html)$/,
threshold: 10240,
minRatio: 0.8
minRatio: 0.8,
});

/** Define "process.env" in client app. Only provide things that can be public */
Expand All @@ -132,25 +132,25 @@ export const getDefinePlugin = (isServer: boolean) =>
: Object.keys(env).reduce(
(result, key: string) => ({
...result,
[key]: JSON.stringify((env as any)[key])
[key]: JSON.stringify((env as any)[key]),
}),
{}
{},
),
__isServer: isServer
__isServer: isServer,
});

/** Enables Webpack HMR */
export const hmrPlugin = new webpack.HotModuleReplacementPlugin();

/** Limit server chunks to be only one. No need to split code in server */
export const limitChunksPlugin = new webpack.optimize.LimitChunkCountPlugin({
maxChunks: 1
maxChunks: 1,
});

const typescriptRule = {
test: /\.tsx?$/,
exclude: /node_modules/,
use: ['babel-loader', { loader: 'ts-loader', options: { transpileOnly: true } }]
use: ['babel-loader', { loader: 'ts-loader', options: { transpileOnly: true } }],
};

/**
Expand All @@ -174,7 +174,7 @@ export const clientConfig: webpack.Configuration = {
},
mode: isDev ? 'development' : 'production',
module: {
rules: [sourceMapRule, typescriptRule, imageAndFontsRule]
rules: [sourceMapRule, typescriptRule, imageAndFontsRule],
},
optimization: {
emitOnErrors: isProd,
Expand All @@ -185,17 +185,17 @@ export const clientConfig: webpack.Configuration = {
enforce: true,
name: 'vendor',
priority: 10,
test: /[\\/]node_modules/
}
}
}
test: /[\\/]node_modules/,
},
},
},
},
output: {
publicPath,
path: dist,
filename: '[name]-[fullhash:8].js',
chunkFilename: '[name]-[chunkhash].chunk.js',
crossOriginLoading: 'anonymous'
crossOriginLoading: 'anonymous',
},
get plugins() {
const plugins: webpack.WebpackPluginInstance[] = [
Expand All @@ -210,8 +210,8 @@ export const clientConfig: webpack.Configuration = {
ctx.setHeader('Content-Type', 'application/javascript; charset=UTF-8');
await next();
}),
port: 7777
})
port: 7777,
}),
];

// Apply production specific configs
Expand All @@ -220,7 +220,7 @@ export const clientConfig: webpack.Configuration = {
}

return plugins;
}
},
};

/**
Expand All @@ -236,16 +236,16 @@ export const serverConfig: webpack.Configuration = {
devtool: isProd ? devtool : undefined,
entry: ['babel-polyfill', './src/server'],
module: {
rules: [sourceMapRule, typescriptRule, imageAndFontsRule]
rules: [sourceMapRule, typescriptRule, imageAndFontsRule],
},
externals: [nodeExternals({ allowlist: /lyft/ })],
output: {
path: dist,
filename: 'server.js',
libraryTarget: 'commonjs2'
libraryTarget: 'commonjs2',
// assetModuleFilename: `${publicPath}/[fullhash:8].[ext]`
},
plugins: [limitChunksPlugin, new ForkTsCheckerWebpackPlugin(), getDefinePlugin(true)]
plugins: [limitChunksPlugin, new ForkTsCheckerWebpackPlugin(), getDefinePlugin(true)],
};

export default [clientConfig, serverConfig];