今天突然碰到了一個兼容性需求,需要根據(jù)不同 macOS 版本,進行不同的兼容性處理。
成都創(chuàng)新互聯(lián)自2013年創(chuàng)立以來,先為水富等服務建站,水富等地企業(yè),進行企業(yè)商務咨詢服務。為水富企業(yè)網站制作PC+手機+微官網三網同步一站式服務解決您的所有建站問題。
沒想到看似簡單的需求,中間也經歷了一番波折,好在最后解決了問題。
在此記錄一下解決問題的過程,也方便其他有類似需求的同學參考。
既然需要針對 mac 系統(tǒng)進行兼容性處理,首先需要區(qū)分系統(tǒng)類型,好在 node 提供了相關的 API,只需使用 os.platform()
即可直接獲取系統(tǒng)類型。
import { platform } from 'os';
console.log(platform()); // 'darwin'
值得一提的是,macOS 對應的名稱不是 macOS
而是 darwin
。
除了可以通過 os 這個包獲取系統(tǒng)類型,我們還可以通過 process.platform
來獲取。
import { platform } from 'process';
console.log(platform); // 'darwin'
很多熟悉 node 的同學這時候會說了,可以通過 os.release()
快速獲取系統(tǒng)版本。但是,當你執(zhí)行 os.release()
,你會發(fā)現(xiàn),它的結果并符合常規(guī)認知。
比如,我的系統(tǒng)版本是 12.0.1
,但是,os.release()
的執(zhí)行結果卻是 21.1.0
。
查看 node 官方文檔可以發(fā)現(xiàn),node 是通過 uname(3) 來確定操作系統(tǒng)的,所以執(zhí)行結果比較違背常規(guī)認知。
既然 os.release()
獲取的系統(tǒng)版本很難讓人理解,那么有沒有什么方案可以直接獲取實際的系統(tǒng)版本號呢?
mac 上存在 sw_vers
命令,可以直接獲取系統(tǒng)信息。
$ sw_vers
ProductName: macOS
ProductVersion: 12.0.1
BuildVersion: 21A559
可以看到 sw_vers
命令直接獲取到了系統(tǒng)版本:12.0.1
。
如果不需要其他系統(tǒng)信息,我們可以通過添加命令行參數(shù),過濾命令行結果,直接獲取系統(tǒng)版本。
$ sw_vers -ProductVersion
12.0.1
由于命令行不區(qū)分大小寫,所以命令行參數(shù)無所謂大小寫,你也可以寫成 -productversion
或者 -ProductVersion
。
既然知道命令行怎么寫,接下來我們只需要在 node 中調用命令行即可。
import { execSync } from 'child_process';
console.log(execSync('sw_vers -ProductVersion', { encoding: 'utf-8' })); // 12.0.1
通過 node 子進程執(zhí)行命令行命令,可以獲取正確的系統(tǒng)版本。但是這種方式有著性能方面的缺陷,需要創(chuàng)建子進程執(zhí)行命令行,相對于 os.release()
的方式,開銷較大。
之前使用 os.release()
的問題在于返回結果比較違反常規(guī)認知,但是它的結果其實是正確的,只不過需要進行一次映射。
所以,通過 os.release()
配合映射表即可優(yōu)雅地獲取系統(tǒng)版本。
import os from 'os';
const macVersionMap = new Map([
[21, ['Monterey', '12']],
[20, ['Big Sur', '11']],
[19, ['Catalina', '10.15']],
[18, ['Mojave', '10.14']],
[17, ['High Sierra', '10.13']],
[16, ['Sierra', '10.12']],
[15, ['El Capitan', '10.11']],
[14, ['Yosemite', '10.10']],
[13, ['Mavericks', '10.9']],
[12, ['Mountain Lion', '10.8']],
[11, ['Lion', '10.7']],
[10, ['Snow Leopard', '10.6']],
[9, ['Leopard', '10.5']],
[8, ['Tiger', '10.4']],
[7, ['Panther', '10.3']],
[6, ['Jaguar', '10.2']],
[5, ['Puma', '10.1']],
]);
const getMacRelease = (release: string) => {
const macRelease = release ?? os.release();
const firstReleaseVersion = Number(macRelease.split('.')[0]);
const [name, version] = macVersionMap.get(firstReleaseVersion) || [
'Unknown',
'',
];
return {
name,
version,
};
};
console.log(getMacRelease()) // 12
為了方便 mac 新發(fā)布系統(tǒng)后依然能夠識別,將新系統(tǒng)版本名稱識別為 Unknow
,新版本識別為空字符串。
此舉是學習移動端機型打分平臺,如果出現(xiàn)版本為空時,默認為高版本,不影響低版本的兼容性處理。
現(xiàn)在,我們就能夠通過 node 優(yōu)雅地獲取系統(tǒng)版本啦。