Node.js如何使用命令行工具檢查更新

這篇文章主要介紹Node.js如何使用命令行工具檢查更新,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

創(chuàng)新互聯(lián)建站堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都做網(wǎng)站、網(wǎng)站設(shè)計(jì)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的麻城網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

使用

我們先簡(jiǎn)單看看這個(gè) npm 包的使用方法:

const updater = require('pkg-updater');
const pkg = require('./package.json'); // 命令行工具自己的 package 信息

updater({'pkg': pkg}) .then(() => { /* 在這里啟動(dòng)命令行工具 */ });

updater({
 'pkg': pkg, 
 // 自定義 registry
 'registry': 'http://xxx.registry.com',
 // 自定義請(qǐng)求的 dist-tag,默認(rèn)是 latest
 'tag': 'next',
 // 自定義檢查間隔,默認(rèn)是 1h
 'checkInterval': 24 * 60 * 60 * 1000,
 // 自定義更新提示信息
 'updateMessage': 'package update from <%=current%> to <%=latest%>.',
 // 自定義強(qiáng)制更新的版本更新級(jí)別,默認(rèn)是 major
 'level': 'minor'
}).then(() => { /* 在這里啟動(dòng)命令行工具 */ });

updater({
 'pkg': pkg,
 // 完全自定義版本更新時(shí)的邏輯
 'onVersionChange': function* (opts) {

 }
}).then(() => { /* 在這里啟動(dòng)命令行工具 */ });

效果如圖:

Node.js如何使用命令行工具檢查更新

Node.js如何使用命令行工具檢查更新

實(shí)現(xiàn)

使用方法很簡(jiǎn)單,我們一起來看看其實(shí)現(xiàn)方法。

需求

我們先來梳理下需求,一個(gè)命令行檢查更新器應(yīng)該至少提供如下功能:

能從遠(yuǎn)程獲取最新版本

能根據(jù)檢查結(jié)果進(jìn)行提示

在版本不兼容時(shí)可以直接退出,強(qiáng)制用戶升級(jí)程序

獲取版本

獲取最新版本這個(gè)功能看起來很簡(jiǎn)單,就是發(fā)送一個(gè)請(qǐng)求從“某處”獲取信息。但是有一些問題需要我們考慮:

從哪里獲取版本信息?

獲取版本信息的策略是怎樣的?(什么時(shí)候獲?。揩@取的信息如何處理?)

從哪里獲取版本信息

我們的命令行工具一般都是使用 npm 進(jìn)行分發(fā),最簡(jiǎn)便的方法就是直接通過 registry 獲取。通過請(qǐng)求 https://registry.npmjs.org/{name}/{dist-tag} 就可以得到 package 對(duì)應(yīng) tag 的版本信息。結(jié)果類似下面這樣:

// https://registry.npmjs.org/co/latest
{
 "name": "co",
 "version": "4.5.0"
}

在實(shí)際實(shí)現(xiàn)時(shí),我們應(yīng)該允許調(diào)用者自定義 registry 地址、請(qǐng)求的 dist-tag 等,這樣可以有更多的定制性。

獲取版本信息的策略

首先想到的方法是用戶每次執(zhí)行命令時(shí)都去獲取一次版本信息,這樣的獲取策略應(yīng)該是最簡(jiǎn)單和實(shí)時(shí)的。

但是這個(gè)策略其實(shí)并不合適:

每次執(zhí)行命令都要去發(fā)請(qǐng)求進(jìn)行檢查,如果網(wǎng)絡(luò)延遲,會(huì)阻塞命令執(zhí)行,影響用戶體驗(yàn)

工具的版本更新其實(shí)并不會(huì)很頻繁,沒有必要進(jìn)行實(shí)時(shí)檢查

網(wǎng)絡(luò)請(qǐng)求的影響因素很多,不能保證每次都成功,應(yīng)該提供本地緩存機(jī)制來存儲(chǔ)請(qǐng)求成功的結(jié)果,避免版本信息的不可用

綜合上面的幾點(diǎn),我們?cè)O(shè)計(jì)如下的獲取策略:

將發(fā)送網(wǎng)絡(luò)請(qǐng)求獲取版本信息的邏輯放在一個(gè)獨(dú)立的后臺(tái)進(jìn)程去執(zhí)行,保證不阻塞主命令執(zhí)行

請(qǐng)求成功后將版本信息、檢查時(shí)間緩存到用戶機(jī)器

每次執(zhí)行命令時(shí),只是讀取本地緩存下來的版本信息,不去發(fā)送網(wǎng)絡(luò)請(qǐng)求

根據(jù)緩存下來的檢查時(shí)間和當(dāng)前時(shí)間,在一個(gè)間隔之內(nèi)不去額外創(chuàng)建后臺(tái)檢查進(jìn)程

將上面的策略翻譯成代碼大概就是下面這樣:

// 讀取本地緩存的檢查結(jié)果
const checkInfo = yield updater.readCheckInfo(opts);
const lastCheck = checkInfo.lastCheck;
const lastVersion = checkInfo.lastVersion;

// 根據(jù)版本信息提示用戶
// ...

// 在時(shí)間間隔內(nèi),直接返回
if (Date.now() - lastCheck < opts.checkInterval) {
 return;
}

// 創(chuàng)建后臺(tái)檢查進(jìn)程
try {
 require('child_process').spawn(
 process.execPath,
 [require('path').join(__dirname, '_check.js'), JSON.stringify({
  'pkg': opts.pkg, // package 信息
  'tag': opts.tag, // 檢查的 dist-tag
  'logFile': opts.logFile, // 緩存文件路徑
  'registry': opts.registry // registry 地址
 })],
 {'stdio': ['ignore', 'ignore', 'ignore'], 'detached': true}
 ).unref();
} catch(e) {}

后臺(tái)進(jìn)程執(zhí)行的 _check.js 文件也很簡(jiǎn)單,如下所示:

const opts = JSON.parse(process.argv[2]);

let lastVersion = '';
try {
 // 發(fā)送請(qǐng)求獲取最新版本
 const url = normalizeUrl(opts.registry + '/' + opts.pkg.name + '/' + (opts.tag || 'latest'));
 const res = yield got.get(url, {
 'json': true,
 'timeout': 60 * 1000
 });
 if (res && res.body && res.body.version) {
 lastVersion = res.body.version;
 }
} catch(e) {}

// 如果獲取失敗了,最新版本就是當(dāng)前版本(package.version)
if (!lastVersion) {
 lastVersion = opts.pkg.version;
}

let data = yield util.readJson(opts.logFile);
if (!data[opts.pkg.name]) {
 data[opts.pkg.name] = {};
}
data[opts.pkg.name].lastVersion = lastVersion; // 最新版本
data[opts.pkg.name].lastCheck = Date.now(); // 檢查時(shí)間

// 寫入緩存
yield util.writeJson(opts.logFile, data);

提示

當(dāng)版本更新了,我們應(yīng)該在終端提示用戶。這里有兩個(gè)問題:

提示文案的問題

提示文案顯示間隔的問題(一直顯示?每隔一段時(shí)間顯示?)

這里我們采取的策略是:

提供默認(rèn)提示文案,清晰的說明當(dāng)前版本、最新版本、更新方法,允許調(diào)用者自定義提示文案

只要有更新就一直顯示提示文案,因?yàn)槲覀兿M脩艚?jīng)常的進(jìn)行更新

實(shí)現(xiàn)代碼大概如下:

// 比對(duì)版本
const type = updater.diffType(opts.pkg.version, lastVersion, opts.level);
if (type) {
 // 根據(jù)模板渲染提示信息
 const str = updater.template(opts.updateMessage || updater.defaultOpts.updateMessage)({
 'colors': updater.colors,
 'name': opts.pkg.name,
 'current': opts.pkg.version,
 'latest': opts.lastVersion,
 'command': 'npm i -g ' + opts.pkg.name
 });
 // 進(jìn)行提示
 console.log(
 updater.boxen(str, {
  'padding': 1,
  'margin': 1,
  'borderStyle': 'classic'
 })
 );
}

強(qiáng)制升級(jí)

對(duì)于 npm 模塊來說,版本 a.b.c 的更新一般有三種情況:

patch:c 位,小版本更新,一般是 bug 修復(fù)

minor:b 位,中版本更新,一般增加新功能、bug 修復(fù)

major,a 位,大版本更新,一般是不兼容的升級(jí)

我們希望當(dāng)遠(yuǎn)程版本的更新如果是 major 形式,命令行工具將直接退出,強(qiáng)制用戶進(jìn)行升級(jí)后才能使用。這可以保證我們推送一個(gè)大版本后,所有的用戶都能夠馬上更新掉,而不是繼續(xù)使用老版本,造成版本碎片的問題。

實(shí)現(xiàn)代碼大致如下:

// 比對(duì)版本
const type = updater.diffType(opts.pkg.version, lastVersion, opts.level);
if (type) {
 // 根據(jù)模板渲染提示信息
 const str = updater.template(opts.updateMessage || updater.defaultOpts.updateMessage)({
 'colors': updater.colors,
 'name': opts.pkg.name,
 'current': opts.pkg.version,
 'latest': opts.lastVersion,
 'command': 'npm i -g ' + opts.pkg.name
 });
 // 進(jìn)行提示
 console.log(
 updater.boxen(str, {
  'padding': 1,
  'margin': 1,
  'borderStyle': 'classic'
 })
 );

 // 不兼容的更新,直接讓進(jìn)程退出
 if (type == 'incompatible') {
 process.exit(1);
 }
}

以上是“Node.js如何使用命令行工具檢查更新”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!

分享文章:Node.js如何使用命令行工具檢查更新
本文路徑:http://muchs.cn/article0/pgohoo.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站改版、服務(wù)器托管、品牌網(wǎng)站建設(shè)定制網(wǎng)站、全網(wǎng)營(yíng)銷推廣、小程序開發(fā)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

營(yíng)銷型網(wǎng)站建設(shè)