中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

使用pkg打包Node.js應用的方法步驟

發布時間:2020-10-11 18:55:51 來源:腳本之家 閱讀:431 作者:jingsam 欄目:web開發

Node.js應用不需要經過編譯過程,可以直接把源代碼拷貝到部署機上執行,確實比C++、Java這類編譯型應用部署方便。然而,Node.js應用執行需要有運行環境,意味著你需要先在部署機器上安裝Node.js。雖說沒有麻煩到哪里去,但畢竟多了一個步驟,特別是對于離線環境下的部署機,麻煩程度還要上升一級。假設你用Node.js寫一些小的桌面級工具軟件,部署到客戶機上還要先安裝Node.js,有點“大炮打蚊子”的感覺。更嚴重的是,如果部署機器上游多個Node.js應用,而且這些應用要依賴于不同的Node.js版本,那就更難部署了。

理想的情況是將Node.js打包為一個單獨的可執行文件,部署的時候直接拷貝過去就行了。除了部署方便外,因為不需要再拷貝源代碼了,還有利于保護知識產權。

將Node.js打包為可執行文件的工具有pkg、nexe、node-packer、enclose等,從打包速度、使用便捷程度、功能完整性來說,pkg是最優秀的。這篇文章就來講一講半年來我使用pkg打包Node.js應用的一些經驗。

pkg的打包原理簡單來說,就是將js代碼以及相關的資源文件打包到可執行文件中,然后劫持fs里面的一些函數,使它能夠讀到可執行文件中的代碼和資源文件。例如,原來的require('./a.js')會被劫持到一個虛擬目錄require('/snapshot/a.js')。

安裝

pkg既可以全局安裝也可以局部安裝,推薦采用局部安裝:

npm install pkg --save-dev

用法

pkg使用比較簡單,執行下pkg -h就可以基本了解用法,基本語法是:

pkg [options] <input>

<input>可以通過三種方式指定:

1.一個腳本文件,例如pkg index.js;
2.package.json,例如pkg package.json,這時會使用package.json中的bin字段作為入口文件;
3.一個目錄,例如pkg .,這時會尋找指定目錄下的package.json文件,然后在找bin字段作為入口文件。

[options]中可以指定打包的參數:

1.-t指定打包的目標平臺和Node版本,如-t node6-win-x64,node6-linux-x64,node6-macos-x64可以同時打包3個平臺的可執行程序;
2.-o指定輸出可執行文件的名稱,但如果用-t指定了多個目標,那么就要用--out-path指定輸出的目錄;
3.-c指定一個JSON配置文件,用來指定需要額外打包腳本和資源文件,通常使用package.json配置。

使用pkg的最佳實踐是:在package.json中的pkg字段中指定打包參數,使用npm scripts來執行打包過程,例如:

{
 ...
 "bin": "./bin/www",
 "scripts": {
  "pkg": "pkg . --out-path=dist/"
 },
 "pkg": {
  "scripts": [...]
  "assets": [...],
  "targets": [...]
 },
 ...
}

scripts和assets用來配置未打包進可執行文件的腳本和資源文件,文件路徑可以使用glob通配符。這里就浮現出一個問題:為什么有的腳本和資源文件打包不進去呢?

要回答這個問題,就涉及到pkg打包文件的機制。按照pkg文檔的說法,pkg只會自動地打包相對于__dirname、__filename的文件,例如path.join(__dirname, '../path/to/asset')。至于require(),因為require本身就是相對于__dirname的,所以能夠自動打包。假設文件中有以下代碼:

require('./build/' + cmd + '.js')
path.join(__dirname, 'views/' + viewName)

這些路徑都不是常量,pkg沒辦法幫你自動識別要打包哪個文件,所以文件就丟失了,所以這時候就使用scripts和assets來告訴pkg,這些文件要打包進去。那么我們怎么知道哪些文件沒有被打包呢?難倒要一行行地去翻源代碼嗎?其實很簡單,只需要把打包好的文件運行下,報錯信息一般就會告訴你缺失哪些文件,并且pkg在打包過程中也會提示一些它不能自動打包的文件。

注意事項

如果說pkg還有哪兒還可以改進的地方,那就是無法自動打包二進制模塊*.node文件。如果你的項目中引用了二進制模塊,如sqlite3,那么你需要手動地將*.node文件復制到可執行文件同一目錄,我通常使用命令cp node_modules/**/*.node .一鍵完成。但是,如果你要跨平臺打包,例如在windows上打包linux版本,相應的二進制模塊也要換成linux版本,通常需要你手動的下載或者編譯。

那為什么pkg不能將二進制模塊打包進去呢?我猜想是require載入一個js文件和node文件,它們的機制是不一樣的。另外從設計來說,不自動打包二進制模塊也是合理的,因為二進制模塊都是平臺相關的。如果我在windows上生成一個linux文件,那么就需要拉取linux版本的.node文件,這是比較困難的。并且有些二進制模塊不提供預編譯版本,需要安裝的時候編譯,pkg再牛也不可能模擬一個其他平臺的編譯環境吧。nexe可以自動打包二進制模塊,但是只能打包當前平臺和當前版本的可執行文件。這意味著如果Node.js應用引用了二進制包,那么這個應用就不能跨平臺打包了,所以我認為這方面,nexe不能算是一個好的設計。

還有一點就是關于項目中的配置文件處理,比如數據庫連接參數、環境變量等。因為這些配置文件會跟著不同的部署環境進行更改,所以為了方便更改,一般不希望把配置文件打包到exe。為了避免pkg自動地將配置文件打包到exe中,代碼中不要采用以下方式讀取配置文件:

fs.readFile(path.join(__dirname, './config.json')), callback)

而是采用相對于process.CWD()的方法讀取:

fs.readFile(path.join(process.CWD(), './config.json'), callback)
// 或者
fs.readFile('./config.json', callback)

如果配置文件是js格式的,那么不要直接require('./config'),而是采用動態require:

const config = require(process.CWD() + './config')

另外要提及的是pkg打包之后動態載入js文件會有安全性問題,即用戶可以在js文件寫任何處理邏輯,注入到打包后的exe中。例如,可以讀取exe里面的虛擬文件系統,把源代碼導出來。所以,盡量不要采用JS作為配置文件,也不要動態載入js模塊。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

阳山县| 深水埗区| 长治市| 关岭| 衡阳市| 鹿泉市| 吴堡县| 冷水江市| 公安县| 长治市| 宜良县| 霍州市| 远安县| 吉木萨尔县| 邢台市| 广宁县| 佛坪县| 屏东县| 东城区| 磴口县| 南投市| 长阳| 镇沅| 阿拉尔市| 靖远县| 九江县| 绥宁县| 三亚市| 镇原县| 丰都县| 宁阳县| 山东| 循化| 高阳县| 洛南县| 如皋市| 武强县| 郸城县| 龙陵县| 县级市| 徐州市|