Webpack5 踩坑之(html-webpack-plugin)

2021 開工第一天就給我個大大的驚喜。

起因

今天起了個大早障陶,懷著無比激動的心情來到了公司滋恬,一切還是跟往常一樣,當(dāng)我打開電腦并打開項目抱究,就出現(xiàn)了下面的場景了:

1-1.png

ERROR in   Error: The loader "/home/project/vue3-demo/node_modules/html-webpack-plugin/lib/loader.js!/home/project/vue3-demo/public/index.html" didn't return html.
  
  - index.js:339 HtmlWebpackPlugin.evaluateCompilationResult
    [vue3-demo]/[html-webpack-plugin]/index.js:339:24
  
  - index.js:243 
    [vue3-demo]/[html-webpack-plugin]/index.js:243:22
  
  - task_queues.js:93 processTicksAndRejections
    internal/process/task_queues.js:93:5
  
  - async Promise.all
  
  - async Promise.all

心態(tài)崩了呀恢氯,之前還是好好的,咋就涼了呢鼓寺?

經(jīng)過

bug 的過程比較漫長勋拟,我就不在這里描述了~

原因

環(huán)境

  • webpack:v5.22.0
  • html-webpack-plugin:v4.5.1

由于 webpack5.22.0 版本中修改了 library 的生成方式,具體代碼在 webpack5.22.0 源碼文件 webpack/lib/library/AssignLibraryPlugin.js279 行:

...
    result.add(
                `${accessWithInit(
                    fullNameResolved,
                    this._getPrefix(compilation).length,
                    false
                )} = __webpack_exports__${exportAccess};\n`
            );
...

webpack5.22.0 源碼文件 webpack/lib/library/AssignLibraryPlugin.js212 行:

render(source, { chunk }, { options, compilation }) {
        ...
            source = new ConcatSource(`${this.declare} ${base};\n`, source);
}

這樣說可能有點抽象妈候,大家可以在 node 環(huán)境中運(yùn)行一下下面這段代碼:

// let source = `var HTML_WEBPACK_PLUGIN_RESULT;HTML_WEBPACK_PLUGIN_RESULT="hello"`; // webpack 5.22.0 以下版本
let source = `var HTML_WEBPACK_PLUGIN_RESULT;(function(){HTML_WEBPACK_PLUGIN_RESULT="hello"})()`; // webpack 5.22.0 版本
const vm = require("vm");
const vmContext = vm.createContext({ HTML_WEBPACK_PLUGIN: true, require: require, module, console: console ,...global});
const vmScript = new vm.Script(source);
// Evaluate code and cast to string
let newSource;
try {
    newSource = vmScript.runInContext(vmContext);
} catch (e) {
    return Promise.reject(e);
}
console.log(newSource);

這是模擬的 webpack 5.22.0 版本下的 html-webpack-plugin 插件敢靡,運(yùn)行后可以發(fā)現(xiàn)返回的是:

undefined

然后我們模擬一下 webpack 5.22.0 以下版本的 html-webpack-plugin 插件,運(yùn)行后可以發(fā)現(xiàn)返回的是:

hello

當(dāng)然苦银,我這只是模擬一下 html-webpack-plugin 插件啸胧,真正返回的可不是 hello 字符串。

報錯的原因正是因為 webpack 5.22.0 版本下的 html-webpack-plugin 插件返回的是 undifined 而不是 hello幔虏。

我們可以試著改一下上面的 webpack 5.22.0 版本下的 html-webpack-plugin 插件代碼:

// let source = `var HTML_WEBPACK_PLUGIN_RESULT;HTML_WEBPACK_PLUGIN_RESULT="hello"`; // webpack 5.22.0 以下版本
let source = `var HTML_WEBPACK_PLUGIN_RESULT;(function(){HTML_WEBPACK_PLUGIN_RESULT="hello"})()`; // webpack 5.22.0 版本
const vm = require("vm");
const vmContext = vm.createContext({ HTML_WEBPACK_PLUGIN: true, require: require, module, console: console ,...global});
const vmScript = new vm.Script(source);
// Evaluate code and cast to string
let newSource;
try {
    vmScript.runInContext(vmContext);
} catch (e) {
    return Promise.reject(e);
}
newSource = vmContext.HTML_WEBPACK_PLUGIN_RESULT;
console.log(newSource);

可以看到纺念,我們加了下面這一行代碼:

newSource = vmContext.HTML_WEBPACK_PLUGIN_RESULT;

我們再次運(yùn)行后會發(fā)現(xiàn),代碼正常返回了 hello 字符串想括。

解決方法

  1. 對項目的 webpack 版本進(jìn)行降級柠辞,降到 v5.22.0 以下。

  2. 修改本地的 html-webpack-plugin 源碼主胧,找到你本地的 xxx/node_modules/html-webpack-plugin/index.js325 行:

     // Evaluate code and cast to string
        let newSource;
        try {
          // fix issues: https://github.com/jantimon/html-webpack-plugin/issues/1603
          // newSource = vmScript.runInContext(vmContext);
          vmScript.runInContext(vmContext);
        } catch (e) {
          return Promise.reject(e);
        }
        // see issues: https://github.com/jantimon/html-webpack-plugin/issues/1603
        newSource = vmContext.HTML_WEBPACK_PLUGIN_RESULT;
    

結(jié)果

首先肯定是不能去罵 webpack叭首,畢竟別人是大哥大,他會說 “我是不會改的踪栋,你愛用不用”焙格,所以就只能委屈一下 html-webpack-plugin 了,于是我點開了html-webpack-plugingithubissues 夷都,太哇塞了呀眷唉,已經(jīng)有很多人在討論了 Webpack v5.22.0` breaks HtmlWebpackLoader #1603

于是我趕緊在底下貼上我的解決方案并附上(you can do this):

1-2.png

然后趕緊去給作者提個 pr 蹭個 contributor,哈哈冬阳,畢竟別人是幾千 k 的 star 項目蛤虐。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市肝陪,隨后出現(xiàn)的幾起案子驳庭,更是在濱河造成了極大的恐慌,老刑警劉巖氯窍,帶你破解...
    沈念sama閱讀 221,548評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件饲常,死亡現(xiàn)場離奇詭異,居然都是意外死亡狼讨,警方通過查閱死者的電腦和手機(jī)贝淤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來政供,“玉大人播聪,你說我怎么就攤上這事〔几簦” “怎么了犬耻?”我有些...
    開封第一講書人閱讀 167,990評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長执泰。 經(jīng)常有香客問我,道長渡蜻,這世上最難降的妖魔是什么术吝? 我笑而不...
    開封第一講書人閱讀 59,618評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮茸苇,結(jié)果婚禮上排苍,老公的妹妹穿的比我還像新娘。我一直安慰自己学密,他們只是感情好淘衙,可當(dāng)我...
    茶點故事閱讀 68,618評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著腻暮,像睡著了一般彤守。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上哭靖,一...
    開封第一講書人閱讀 52,246評論 1 308
  • 那天具垫,我揣著相機(jī)與錄音,去河邊找鬼试幽。 笑死筝蚕,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播起宽,決...
    沈念sama閱讀 40,819評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼洲胖,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了坯沪?” 一聲冷哼從身側(cè)響起绿映,我...
    開封第一講書人閱讀 39,725評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎屏箍,沒想到半個月后绘梦,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,268評論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡赴魁,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,356評論 3 340
  • 正文 我和宋清朗相戀三年卸奉,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片颖御。...
    茶點故事閱讀 40,488評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡榄棵,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出潘拱,到底是詐尸還是另有隱情疹鳄,我是刑警寧澤,帶...
    沈念sama閱讀 36,181評論 5 350
  • 正文 年R本政府宣布芦岂,位于F島的核電站瘪弓,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏禽最。R本人自食惡果不足惜腺怯,卻給世界環(huán)境...
    茶點故事閱讀 41,862評論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望川无。 院中可真熱鬧呛占,春花似錦、人聲如沸懦趋。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽仅叫。三九已至帜篇,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間诫咱,已是汗流浹背坠狡。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留遂跟,地道東北人逃沿。 一個月前我還...
    沈念sama閱讀 48,897評論 3 376
  • 正文 我出身青樓婴渡,卻偏偏與公主長得像,于是被迫代替她去往敵國和親凯亮。 傳聞我的和親對象是個殘疾皇子边臼,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,500評論 2 359

推薦閱讀更多精彩內(nèi)容