-
-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Improve webpack configs (deterministic builds across devices)
- Loading branch information
1 parent
1af2739
commit 2b9e766
Showing
5 changed files
with
167 additions
and
74 deletions.
There are no files selected for viewing
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
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
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
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,131 @@ | ||
|
||
//@ts-check | ||
'use strict'; | ||
|
||
const webpack = require('webpack'); | ||
const { createHash } = require('crypto'); | ||
|
||
class WebpackPlugin { | ||
_formatIdCache = new Map(); | ||
/** @type {(id: string, rootPath: string) => string} */ | ||
formatId(id, rootPath) { | ||
// Make sure all paths use / | ||
id = id.replace(/\\/g, '/'); | ||
// For `[path]` we unwrap, format then rewrap | ||
if (id[0] === '[' && id.endsWith(']')) { | ||
return `[${this.formatId(id.slice(1, id.length - 1), rootPath)}]`; | ||
} | ||
// When dealing with `path1!path2`, format each segment separately | ||
if (id.includes('!')) { | ||
id = id.split('!').map(s => this.formatId(s, rootPath)).join('!'); | ||
} | ||
// Make the paths relative to the project's rooth path if possible | ||
if (id.startsWith(rootPath)) { | ||
id = id.slice(rootPath.length); | ||
id = (id[0] === '/' ? '.' : './') + id; | ||
} | ||
let formatted = this._formatIdCache.get(id); | ||
if (formatted) return formatted; | ||
// Check if we're dealing with a Yarn directory | ||
let match = id.match(/^.*\/(\.?Yarn\/Berry|\.yarn)\/(.*)$/i); | ||
if (!match) { | ||
this._formatIdCache.set(id, formatted = id); | ||
return formatted; | ||
} | ||
const [, yarn, filepath] = match; | ||
// Check if we can extract the package name/version from the path | ||
match = filepath.match(/^unplugged\/([^/]+?)\-[\da-f]{10}\/node_modules\/(.*)$/i) | ||
|| filepath.match(/^cache\/([^/]+?)\-[\da-f]{10}\-\d+\.zip\/node_modules\/(.*)$/i); | ||
if (!match) { | ||
formatted = `/${yarn.toLowerCase() === '.yarn' ? '.' : ''}yarn/${filepath}`; | ||
this._formatIdCache.set(id, formatted); | ||
return formatted; | ||
} | ||
const [, name, path] = match; | ||
formatted = `${yarn.toLowerCase() === '.yarn' ? '.' : '/'}yarn/${name}/${path}`; | ||
this._formatIdCache.set(id, formatted); | ||
return formatted; | ||
} | ||
_hashModuleCache = new Map(); | ||
/** @type {(mod: webpack.Module, rootPath: string) => string} */ | ||
hashModule(mod, rootPath) { | ||
// Prefer `nameForCondition()` as it usually gives the actual file path | ||
// while `identifier()` can have extra `!` or `|` suffixes, i.e. a hash that somehow differs between devices | ||
const identifier = this.formatId(mod.nameForCondition() || mod.identifier(), rootPath); | ||
let hash = this._hashModuleCache.get(identifier); | ||
if (hash) return hash; | ||
hash = createHash('sha1').update(identifier).digest('hex'); | ||
this._hashModuleCache.set(identifier, hash); | ||
return hash; | ||
} | ||
/** @param {webpack.Compiler} compiler */ | ||
apply(compiler) { | ||
// Output start/stop messages making the $ts-webpack-watch problemMatcher (provided by an extension) work | ||
compiler.hooks.beforeCompile.tap('WebpackPlugin-BeforeCompile', () => { | ||
console.log('Compilation starting'); | ||
}); | ||
compiler.hooks.afterCompile.tap('WebpackPlugin-AfterCompile', () => { | ||
console.log('Compilation finished'); | ||
}); | ||
compiler.hooks.compilation.tap('WebpackPlugin-Compilation', compilation => { | ||
const rootPath = (compilation.options.context || '').replace(/\\/g, '/'); | ||
compilation.options.optimization.chunkIds = false; | ||
// Format `../../../Yarn/Berry/` with all the `cache`/`unplugged`/`__virtual__` to be more readable | ||
// (i.e. `/yarn/package-npm-x.y.z/package/index.js` for global Yarn cache or `/.yarn/...` for local) | ||
compilation.hooks.statsPrinter.tap('WebpackPlugin-StatsPrinter', stats => { | ||
/** @type {(id: string | {}, context: any) => string} */ | ||
const tapModId = (id, context) => typeof id === 'string' ? this.formatId(context.formatModuleId(id), rootPath) : '???'; | ||
stats.hooks.print.for('module.name').tap('WebpackPlugin-ModuleName', tapModId); | ||
}); | ||
// Include an `excludeModules` to `options.stats` to exclude modules loaded by dependencies | ||
compilation.hooks.statsNormalize.tap('WebpackPlugin-StatsNormalize', stats => { | ||
(stats.excludeModules || (stats.excludeModules = [])).push((name, { issuerPath }) => { | ||
if (name.startsWith('external "')) return true; | ||
const issuer = issuerPath && (issuerPath[issuerPath.length - 1].name || '').replace(/\\/g, '/'); | ||
if (!issuer) return false; | ||
const lower = this.formatId(issuer, rootPath).toLowerCase(); | ||
if (lower.startsWith('/yarn/')) return true; | ||
if (lower.startsWith('.yarn/')) return true; | ||
return false; | ||
}); | ||
}); | ||
// Determines how chunk IDs are generated, which is now actually deterministic | ||
// (we make sure to clean Yarn paths to prevent issues with `../../Yarn/Berry` being different on devices) | ||
compilation.hooks.chunkIds.tap('WebpackPlugin-ChunkIds', chunks => { | ||
const chunkIds = new Map(); | ||
const overlapMap = new Set(); | ||
let minLength = 4; // show at least 3 characters | ||
// Calculate the hashes for all the chunks | ||
for (const chunk of chunks) { | ||
if (chunk.id) { | ||
console.log(`Chunk ${chunk.id} already has an ID`); | ||
} | ||
// We're kinda doing something similar to Webpack 5's DeterministicChunkIdsPlugin but different | ||
const modules = compilation.chunkGraph.getChunkRootModules(chunk); | ||
const hashes = modules.map(m => this.hashModule(m, rootPath)).sort(); | ||
const hasher = createHash('sha1'); | ||
for (const hash of hashes) hasher.update(hash); | ||
const hash = hasher.digest('hex'); | ||
// With a 160-bit value, a clash is very unlikely, but let's check anyway | ||
if (chunkIds.has(hash)) throw new Error('Hash collision for chunk IDs'); | ||
chunkIds.set(chunk, hash); | ||
chunk.id = hash; | ||
// Make sure the minLength remains high enough to avoid collisions | ||
for (let i = minLength; i < hash.length; i++) { | ||
const part = hash.slice(0, i); | ||
if (overlapMap.has(part)) continue; | ||
overlapMap.add(part); | ||
minLength = i; | ||
break; | ||
} | ||
} | ||
// Assign the shortened (collision-free) hashes for all the chunks | ||
for (const [chunk, hash] of chunkIds) { | ||
chunk.id = hash.slice(0, minLength); | ||
chunk.ids = [chunk.id]; | ||
} | ||
}); | ||
}); | ||
} | ||
} | ||
module.exports.WebpackPlugin = WebpackPlugin; |
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