PageRenderTime 1664ms CodeModel.GetById 1ms RepoModel.GetById 0ms app.codeStats 0ms

/node_modules/systeminformation/lib/system.js

https://gitlab.com/varunsonavne/node-hello
JavaScript | 829 lines | 676 code | 26 blank | 127 comment | 90 complexity | 4644c43e45d5025a7a8aa3c796fed1d4 MD5 | raw file
  1. 'use strict';
  2. // @ts-check
  3. // ==================================================================================
  4. // system.js
  5. // ----------------------------------------------------------------------------------
  6. // Description: System Information - library
  7. // for Node.js
  8. // Copyright: (c) 2014 - 2022
  9. // Author: Sebastian Hildebrandt
  10. // ----------------------------------------------------------------------------------
  11. // License: MIT
  12. // ==================================================================================
  13. // 2. System (Hardware, BIOS, Base Board)
  14. // ----------------------------------------------------------------------------------
  15. const fs = require('fs');
  16. const os = require('os');
  17. const util = require('./util');
  18. const exec = require('child_process').exec;
  19. const execSync = require('child_process').execSync;
  20. const execPromise = util.promisify(require('child_process').exec);
  21. let _platform = process.platform;
  22. const _linux = (_platform === 'linux' || _platform === 'android');
  23. const _darwin = (_platform === 'darwin');
  24. const _windows = (_platform === 'win32');
  25. const _freebsd = (_platform === 'freebsd');
  26. const _openbsd = (_platform === 'openbsd');
  27. const _netbsd = (_platform === 'netbsd');
  28. const _sunos = (_platform === 'sunos');
  29. function system(callback) {
  30. return new Promise((resolve) => {
  31. process.nextTick(() => {
  32. let result = {
  33. manufacturer: '',
  34. model: 'Computer',
  35. version: '',
  36. serial: '-',
  37. uuid: '-',
  38. sku: '-',
  39. virtual: false
  40. };
  41. if (_linux || _freebsd || _openbsd || _netbsd) {
  42. exec('export LC_ALL=C; dmidecode -t system 2>/dev/null; unset LC_ALL', function (error, stdout) {
  43. // if (!error) {
  44. let lines = stdout.toString().split('\n');
  45. result.manufacturer = util.getValue(lines, 'manufacturer');
  46. result.model = util.getValue(lines, 'product name');
  47. result.version = util.getValue(lines, 'version');
  48. result.serial = util.getValue(lines, 'serial number');
  49. result.uuid = util.getValue(lines, 'uuid').toLowerCase();
  50. result.sku = util.getValue(lines, 'sku number');
  51. // }
  52. // Non-Root values
  53. const cmd = `echo -n "product_name: "; cat /sys/devices/virtual/dmi/id/product_name 2>/dev/null; echo;
  54. echo -n "product_serial: "; cat /sys/devices/virtual/dmi/id/product_serial 2>/dev/null; echo;
  55. echo -n "product_uuid: "; cat /sys/devices/virtual/dmi/id/product_uuid 2>/dev/null; echo;
  56. echo -n "product_version: "; cat /sys/devices/virtual/dmi/id/product_version 2>/dev/null; echo;
  57. echo -n "sys_vendor: "; cat /sys/devices/virtual/dmi/id/sys_vendor 2>/dev/null; echo;`;
  58. try {
  59. lines = execSync(cmd).toString().split('\n');
  60. result.manufacturer = result.manufacturer === '' ? util.getValue(lines, 'sys_vendor') : result.manufacturer;
  61. result.model = result.model === '' ? util.getValue(lines, 'product_name') : result.model;
  62. result.version = result.version === '' ? util.getValue(lines, 'product_version') : result.version;
  63. result.serial = result.serial === '' ? util.getValue(lines, 'product_serial') : result.serial;
  64. result.uuid = result.uuid === '' ? util.getValue(lines, 'product_uuid').toLowerCase() : result.uuid;
  65. } catch (e) {
  66. util.noop();
  67. }
  68. if (!result.serial || result.serial.toLowerCase().indexOf('o.e.m.') !== -1) { result.serial = '-'; }
  69. if (!result.manufacturer || result.manufacturer.toLowerCase().indexOf('o.e.m.') !== -1) { result.manufacturer = ''; }
  70. if (!result.model || result.model.toLowerCase().indexOf('o.e.m.') !== -1) { result.model = 'Computer'; }
  71. if (!result.version || result.version.toLowerCase().indexOf('o.e.m.') !== -1) { result.version = ''; }
  72. if (!result.sku || result.sku.toLowerCase().indexOf('o.e.m.') !== -1) { result.sku = '-'; }
  73. // detect virtual (1)
  74. if (result.model.toLowerCase() === 'virtualbox' || result.model.toLowerCase() === 'kvm' || result.model.toLowerCase() === 'virtual machine' || result.model.toLowerCase() === 'bochs' || result.model.toLowerCase().startsWith('vmware') || result.model.toLowerCase().startsWith('droplet')) {
  75. result.virtual = true;
  76. switch (result.model.toLowerCase()) {
  77. case 'virtualbox':
  78. result.virtualHost = 'VirtualBox';
  79. break;
  80. case 'vmware':
  81. result.virtualHost = 'VMware';
  82. break;
  83. case 'kvm':
  84. result.virtualHost = 'KVM';
  85. break;
  86. case 'bochs':
  87. result.virtualHost = 'bochs';
  88. break;
  89. }
  90. }
  91. if (result.manufacturer.toLowerCase().startsWith('vmware') || result.manufacturer.toLowerCase() === 'xen') {
  92. result.virtual = true;
  93. switch (result.manufacturer.toLowerCase()) {
  94. case 'vmware':
  95. result.virtualHost = 'VMware';
  96. break;
  97. case 'xen':
  98. result.virtualHost = 'Xen';
  99. break;
  100. }
  101. }
  102. if (!result.virtual) {
  103. try {
  104. const disksById = execSync('ls -1 /dev/disk/by-id/ 2>/dev/null').toString();
  105. if (disksById.indexOf('_QEMU_') >= 0) {
  106. result.virtual = true;
  107. result.virtualHost = 'QEMU';
  108. }
  109. if (disksById.indexOf('_VBOX_') >= 0) {
  110. result.virtual = true;
  111. result.virtualHost = 'VirtualBox';
  112. }
  113. } catch (e) {
  114. util.noop();
  115. }
  116. }
  117. if (!result.virtual && (os.release().toLowerCase().indexOf('microsoft') >= 0 || os.release().toLowerCase().endsWith('wsl2'))) {
  118. const kernelVersion = parseFloat(os.release().toLowerCase());
  119. result.virtual = true;
  120. result.manufacturer = 'Microsoft';
  121. result.model = 'WSL';
  122. result.version = kernelVersion < 4.19 ? '1' : '2';
  123. }
  124. if ((_freebsd || _openbsd || _netbsd) && !result.virtualHost) {
  125. try {
  126. const procInfo = execSync('dmidecode -t 4');
  127. const procLines = procInfo.toString().split('\n');
  128. const procManufacturer = util.getValue(procLines, 'manufacturer', ':', true);
  129. switch (procManufacturer.toLowerCase()) {
  130. case 'virtualbox':
  131. result.virtualHost = 'VirtualBox';
  132. break;
  133. case 'vmware':
  134. result.virtualHost = 'VMware';
  135. break;
  136. case 'kvm':
  137. result.virtualHost = 'KVM';
  138. break;
  139. case 'bochs':
  140. result.virtualHost = 'bochs';
  141. break;
  142. }
  143. } catch (e) {
  144. util.noop();
  145. }
  146. }
  147. // detect docker
  148. if (fs.existsSync('/.dockerenv') || fs.existsSync('/.dockerinit')) {
  149. result.model = 'Docker Container';
  150. }
  151. try {
  152. const stdout = execSync('dmesg 2>/dev/null | grep -iE "virtual|hypervisor" | grep -iE "vmware|qemu|kvm|xen" | grep -viE "Nested Virtualization|/virtual/"');
  153. // detect virtual machines
  154. let lines = stdout.toString().split('\n');
  155. if (lines.length > 0) {
  156. if (result.model === 'Computer') { result.model = 'Virtual machine'; }
  157. result.virtual = true;
  158. if (stdout.toString().toLowerCase().indexOf('vmware') >= 0 && !result.virtualHost) {
  159. result.virtualHost = 'VMware';
  160. }
  161. if (stdout.toString().toLowerCase().indexOf('qemu') >= 0 && !result.virtualHost) {
  162. result.virtualHost = 'QEMU';
  163. }
  164. if (stdout.toString().toLowerCase().indexOf('xen') >= 0 && !result.virtualHost) {
  165. result.virtualHost = 'Xen';
  166. }
  167. if (stdout.toString().toLowerCase().indexOf('kvm') >= 0 && !result.virtualHost) {
  168. result.virtualHost = 'KVM';
  169. }
  170. }
  171. } catch (e) {
  172. util.noop();
  173. }
  174. if (result.manufacturer === '' && result.model === 'Computer' && result.version === '') {
  175. // Check Raspberry Pi
  176. fs.readFile('/proc/cpuinfo', function (error, stdout) {
  177. if (!error) {
  178. let lines = stdout.toString().split('\n');
  179. result.model = util.getValue(lines, 'hardware', ':', true).toUpperCase();
  180. result.version = util.getValue(lines, 'revision', ':', true).toLowerCase();
  181. result.serial = util.getValue(lines, 'serial', ':', true);
  182. const model = util.getValue(lines, 'model:', ':', true);
  183. // reference values: https://elinux.org/RPi_HardwareHistory
  184. // https://www.raspberrypi.org/documentation/hardware/raspberrypi/revision-codes/README.md
  185. if ((result.model === 'BCM2835' || result.model === 'BCM2708' || result.model === 'BCM2709' || result.model === 'BCM2710' || result.model === 'BCM2711' || result.model === 'BCM2836' || result.model === 'BCM2837') && model.toLowerCase().indexOf('raspberry') >= 0) {
  186. const rPIRevision = util.decodePiCpuinfo(lines);
  187. result.model = rPIRevision.model;
  188. result.version = rPIRevision.revisionCode;
  189. result.manufacturer = 'Raspberry Pi Foundation';
  190. result.raspberry = {
  191. manufacturer: rPIRevision.manufacturer,
  192. processor: rPIRevision.processor,
  193. type: rPIRevision.type,
  194. revision: rPIRevision.revision
  195. };
  196. }
  197. // if (result.model === 'BCM2835' || result.model === 'BCM2708' || result.model === 'BCM2709' || result.model === 'BCM2835' || result.model === 'BCM2837') {
  198. // // Pi 4
  199. // if (['d03114'].indexOf(result.version) >= 0) {
  200. // result.model = result.model + ' - Pi 4 Model B';
  201. // result.version = result.version + ' - Rev. 1.4';
  202. // }
  203. // if (['b03112', 'c03112'].indexOf(result.version) >= 0) {
  204. // result.model = result.model + ' - Pi 4 Model B';
  205. // result.version = result.version + ' - Rev. 1.2';
  206. // }
  207. // if (['a03111', 'b03111', 'c03111'].indexOf(result.version) >= 0) {
  208. // result.model = result.model + ' - Pi 4 Model B';
  209. // result.version = result.version + ' - Rev. 1.1';
  210. // }
  211. // // Pi 3
  212. // if (['a02082', 'a22082', 'a32082', 'a52082'].indexOf(result.version) >= 0) {
  213. // result.model = result.model + ' - Pi 3 Model B';
  214. // result.version = result.version + ' - Rev. 1.2';
  215. // }
  216. // if (['a22083'].indexOf(result.version) >= 0) {
  217. // result.model = result.model + ' - Pi 3 Model B';
  218. // result.version = result.version + ' - Rev. 1.3';
  219. // }
  220. // if (['a020d3'].indexOf(result.version) >= 0) {
  221. // result.model = result.model + ' - Pi 3 Model B+';
  222. // result.version = result.version + ' - Rev. 1.3';
  223. // }
  224. // if (['9020e0'].indexOf(result.version) >= 0) {
  225. // result.model = result.model + ' - Pi 3 Model A+';
  226. // result.version = result.version + ' - Rev. 1.3';
  227. // }
  228. // // Pi 2 Model B
  229. // if (['a01040'].indexOf(result.version) >= 0) {
  230. // result.model = result.model + ' - Pi 2 Model B';
  231. // result.version = result.version + ' - Rev. 1.0';
  232. // }
  233. // if (['a01041', 'a21041'].indexOf(result.version) >= 0) {
  234. // result.model = result.model + ' - Pi 2 Model B';
  235. // result.version = result.version + ' - Rev. 1.1';
  236. // }
  237. // if (['a22042', 'a02042'].indexOf(result.version) >= 0) {
  238. // result.model = result.model + ' - Pi 2 Model B';
  239. // result.version = result.version + ' - Rev. 1.2';
  240. // }
  241. // // Compute Model
  242. // if (['a02100'].indexOf(result.version) >= 0) {
  243. // result.model = result.model + ' - Pi CM3+';
  244. // result.version = result.version + ' - Rev 1.0';
  245. // }
  246. // if (['a020a0', 'a220a0'].indexOf(result.version) >= 0) {
  247. // result.model = result.model + ' - Pi CM3';
  248. // result.version = result.version + ' - Rev 1.0';
  249. // }
  250. // if (['900061'].indexOf(result.version) >= 0) {
  251. // result.model = result.model + ' - Pi CM';
  252. // result.version = result.version + ' - Rev 1.1';
  253. // }
  254. // // Pi Zero
  255. // if (['900092', '920092'].indexOf(result.version) >= 0) {
  256. // result.model = result.model + ' - Pi Zero';
  257. // result.version = result.version + ' - Rev 1.2';
  258. // }
  259. // if (['900093', '920093'].indexOf(result.version) >= 0) {
  260. // result.model = result.model + ' - Pi Zero';
  261. // result.version = result.version + ' - Rev 1.3';
  262. // }
  263. // if (['9000c1'].indexOf(result.version) >= 0) {
  264. // result.model = result.model + ' - Pi Zero W';
  265. // result.version = result.version + ' - Rev 1.1';
  266. // }
  267. // // A, B, A+ B+
  268. // if (['0002', '0003'].indexOf(result.version) >= 0) {
  269. // result.model = result.model + ' - Pi Model B';
  270. // result.version = result.version + ' - Rev 1.0';
  271. // }
  272. // if (['0004', '0005', '0006', '000d', '000e', '000f'].indexOf(result.version) >= 0) {
  273. // result.model = result.model + ' - Pi Model B';
  274. // result.version = result.version + ' - Rev 2.0';
  275. // }
  276. // if (['0007', '0008', '0009'].indexOf(result.version) >= 0) {
  277. // result.model = result.model + ' - Pi Model A';
  278. // result.version = result.version + ' - Rev 2.0';
  279. // }
  280. // if (['0010'].indexOf(result.version) >= 0) {
  281. // result.model = result.model + ' - Pi Model B+';
  282. // result.version = result.version + ' - Rev 1.0';
  283. // }
  284. // if (['0012'].indexOf(result.version) >= 0) {
  285. // result.model = result.model + ' - Pi Model A+';
  286. // result.version = result.version + ' - Rev 1.0';
  287. // }
  288. // if (['0013', '900032'].indexOf(result.version) >= 0) {
  289. // result.model = result.model + ' - Pi Model B+';
  290. // result.version = result.version + ' - Rev 1.2';
  291. // }
  292. // if (['0015', '900021'].indexOf(result.version) >= 0) {
  293. // result.model = result.model + ' - Pi Model A+';
  294. // result.version = result.version + ' - Rev 1.1';
  295. // }
  296. // if (result.model.indexOf('Pi') !== -1 && result.version) { // Pi, Pi Zero
  297. // result.manufacturer = 'Raspberry Pi Foundation';
  298. // }
  299. // }
  300. }
  301. if (callback) { callback(result); }
  302. resolve(result);
  303. });
  304. } else {
  305. if (callback) { callback(result); }
  306. resolve(result);
  307. }
  308. });
  309. }
  310. if (_darwin) {
  311. exec('ioreg -c IOPlatformExpertDevice -d 2', function (error, stdout) {
  312. if (!error) {
  313. let lines = stdout.toString().replace(/[<>"]/g, '').split('\n');
  314. result.manufacturer = util.getValue(lines, 'manufacturer', '=', true);
  315. result.model = util.getValue(lines, 'model', '=', true);
  316. result.version = util.getValue(lines, 'version', '=', true);
  317. result.serial = util.getValue(lines, 'ioplatformserialnumber', '=', true);
  318. result.uuid = util.getValue(lines, 'ioplatformuuid', '=', true).toLowerCase();
  319. result.sku = util.getValue(lines, 'board-id', '=', true);
  320. }
  321. if (callback) { callback(result); }
  322. resolve(result);
  323. });
  324. }
  325. if (_sunos) {
  326. if (callback) { callback(result); }
  327. resolve(result);
  328. }
  329. if (_windows) {
  330. try {
  331. util.powerShell('Get-WmiObject Win32_ComputerSystemProduct | select Name,Vendor,Version,IdentifyingNumber,UUID | fl').then((stdout, error) => {
  332. if (!error) {
  333. // let lines = stdout.split('\r\n').filter(line => line.trim() !== '').filter((line, idx) => idx > 0)[0].trim().split(/\s\s+/);
  334. let lines = stdout.split('\r\n');
  335. result.manufacturer = util.getValue(lines, 'vendor', ':');
  336. result.model = util.getValue(lines, 'name', ':');
  337. result.version = util.getValue(lines, 'version', ':');
  338. result.serial = util.getValue(lines, 'identifyingnumber', ':');
  339. result.uuid = util.getValue(lines, 'uuid', ':').toLowerCase();
  340. // detect virtual (1)
  341. const model = result.model.toLowerCase();
  342. if (model === 'virtualbox' || model === 'kvm' || model === 'virtual machine' || model === 'bochs' || model.startsWith('vmware') || model.startsWith('qemu')) {
  343. result.virtual = true;
  344. if (model.startsWith('virtualbox')) { result.virtualHost = 'VirtualBox'; }
  345. if (model.startsWith('vmware')) { result.virtualHost = 'VMware'; }
  346. if (model.startsWith('kvm')) { result.virtualHost = 'KVM'; }
  347. if (model.startsWith('bochs')) { result.virtualHost = 'bochs'; }
  348. if (model.startsWith('qemu')) { result.virtualHost = 'KVM'; }
  349. }
  350. const manufacturer = result.manufacturer.toLowerCase();
  351. if (manufacturer.startsWith('vmware') || manufacturer.startsWith('qemu') || manufacturer === 'xen') {
  352. result.virtual = true;
  353. if (manufacturer.startsWith('vmware')) { result.virtualHost = 'VMware'; }
  354. if (manufacturer.startsWith('xen')) { result.virtualHost = 'Xen'; }
  355. if (manufacturer.startsWith('qemu')) { result.virtualHost = 'KVM'; }
  356. }
  357. util.powerShell('Get-WmiObject MS_Systeminformation -Namespace "root/wmi" | select systemsku | fl ').then((stdout, error) => {
  358. if (!error) {
  359. let lines = stdout.split('\r\n');
  360. result.sku = util.getValue(lines, 'systemsku', ':');
  361. }
  362. if (!result.virtual) {
  363. util.powerShell('Get-WmiObject Win32_bios | select Version, SerialNumber, SMBIOSBIOSVersion').then((stdout, error) => {
  364. if (!error) {
  365. let lines = stdout.toString();
  366. if (lines.indexOf('VRTUAL') >= 0 || lines.indexOf('A M I ') >= 0 || lines.indexOf('VirtualBox') >= 0 || lines.indexOf('VMWare') >= 0 || lines.indexOf('Xen') >= 0) {
  367. result.virtual = true;
  368. if (lines.indexOf('VirtualBox') >= 0 && !result.virtualHost) {
  369. result.virtualHost = 'VirtualBox';
  370. }
  371. if (lines.indexOf('VMware') >= 0 && !result.virtualHost) {
  372. result.virtualHost = 'VMware';
  373. }
  374. if (lines.indexOf('Xen') >= 0 && !result.virtualHost) {
  375. result.virtualHost = 'Xen';
  376. }
  377. if (lines.indexOf('VRTUAL') >= 0 && !result.virtualHost) {
  378. result.virtualHost = 'Hyper-V';
  379. }
  380. if (lines.indexOf('A M I') >= 0 && !result.virtualHost) {
  381. result.virtualHost = 'Virtual PC';
  382. }
  383. }
  384. if (callback) { callback(result); }
  385. resolve(result);
  386. } else {
  387. if (callback) { callback(result); }
  388. resolve(result);
  389. }
  390. });
  391. } else {
  392. if (callback) { callback(result); }
  393. resolve(result);
  394. }
  395. });
  396. } else {
  397. if (callback) { callback(result); }
  398. resolve(result);
  399. }
  400. });
  401. } catch (e) {
  402. if (callback) { callback(result); }
  403. resolve(result);
  404. }
  405. }
  406. });
  407. });
  408. }
  409. exports.system = system;
  410. function bios(callback) {
  411. return new Promise((resolve) => {
  412. process.nextTick(() => {
  413. let result = {
  414. vendor: '',
  415. version: '',
  416. releaseDate: '',
  417. revision: '',
  418. };
  419. let cmd = '';
  420. if (_linux || _freebsd || _openbsd || _netbsd) {
  421. if (process.arch === 'arm') {
  422. cmd = 'cat /proc/cpuinfo | grep Serial';
  423. } else {
  424. cmd = 'export LC_ALL=C; dmidecode -t bios 2>/dev/null; unset LC_ALL';
  425. }
  426. exec(cmd, function (error, stdout) {
  427. let lines = stdout.toString().split('\n');
  428. result.vendor = util.getValue(lines, 'Vendor');
  429. result.version = util.getValue(lines, 'Version');
  430. let datetime = util.getValue(lines, 'Release Date');
  431. result.releaseDate = util.parseDateTime(datetime).date;
  432. result.revision = util.getValue(lines, 'BIOS Revision');
  433. result.serial = util.getValue(lines, 'SerialNumber');
  434. let language = util.getValue(lines, 'Currently Installed Language').split('|')[0];
  435. if (language) {
  436. result.language = language;
  437. }
  438. if (lines.length && stdout.toString().indexOf('Characteristics:') >= 0) {
  439. const features = [];
  440. lines.forEach(line => {
  441. if (line.indexOf(' is supported') >= 0) {
  442. const feature = line.split(' is supported')[0].trim();
  443. features.push(feature);
  444. }
  445. });
  446. result.features = features;
  447. }
  448. // Non-Root values
  449. const cmd = `echo -n "bios_date: "; cat /sys/devices/virtual/dmi/id/bios_date 2>/dev/null; echo;
  450. echo -n "bios_vendor: "; cat /sys/devices/virtual/dmi/id/bios_vendor 2>/dev/null; echo;
  451. echo -n "bios_version: "; cat /sys/devices/virtual/dmi/id/bios_version 2>/dev/null; echo;`;
  452. try {
  453. lines = execSync(cmd).toString().split('\n');
  454. result.vendor = !result.vendor ? util.getValue(lines, 'bios_vendor') : result.vendor;
  455. result.version = !result.version ? util.getValue(lines, 'bios_version') : result.version;
  456. datetime = util.getValue(lines, 'bios_date');
  457. result.releaseDate = !result.releaseDate ? util.parseDateTime(datetime).date : result.releaseDate;
  458. } catch (e) {
  459. util.noop();
  460. }
  461. if (callback) { callback(result); }
  462. resolve(result);
  463. });
  464. }
  465. if (_darwin) {
  466. result.vendor = 'Apple Inc.';
  467. exec(
  468. 'system_profiler SPHardwareDataType -json', function (error, stdout) {
  469. try {
  470. const hardwareData = JSON.parse(stdout.toString());
  471. if (hardwareData && hardwareData.SPHardwareDataType && hardwareData.SPHardwareDataType.length) {
  472. let bootRomVersion = hardwareData.SPHardwareDataType[0].boot_rom_version;
  473. bootRomVersion = bootRomVersion ? bootRomVersion.split('(')[0].trim() : null;
  474. result.version = bootRomVersion;
  475. }
  476. } catch (e) {
  477. util.noop();
  478. }
  479. if (callback) { callback(result); }
  480. resolve(result);
  481. });
  482. }
  483. if (_sunos) {
  484. result.vendor = 'Sun Microsystems';
  485. if (callback) { callback(result); }
  486. resolve(result);
  487. }
  488. if (_windows) {
  489. try {
  490. util.powerShell('Get-WmiObject Win32_bios | select Description,Version,Manufacturer,ReleaseDate,BuildNumber,SerialNumber | fl').then((stdout, error) => {
  491. if (!error) {
  492. let lines = stdout.toString().split('\r\n');
  493. const description = util.getValue(lines, 'description', ':');
  494. if (description.indexOf(' Version ') !== -1) {
  495. // ... Phoenix ROM BIOS PLUS Version 1.10 A04
  496. result.vendor = description.split(' Version ')[0].trim();
  497. result.version = description.split(' Version ')[1].trim();
  498. } else if (description.indexOf(' Ver: ') !== -1) {
  499. // ... BIOS Date: 06/27/16 17:50:16 Ver: 1.4.5
  500. result.vendor = util.getValue(lines, 'manufacturer', ':');
  501. result.version = description.split(' Ver: ')[1].trim();
  502. } else {
  503. result.vendor = util.getValue(lines, 'manufacturer', ':');
  504. result.version = util.getValue(lines, 'version', ':');
  505. }
  506. result.releaseDate = util.getValue(lines, 'releasedate', ':');
  507. if (result.releaseDate.length >= 10) {
  508. result.releaseDate = result.releaseDate.substr(0, 4) + '-' + result.releaseDate.substr(4, 2) + '-' + result.releaseDate.substr(6, 2);
  509. }
  510. result.revision = util.getValue(lines, 'buildnumber', ':');
  511. result.serial = util.getValue(lines, 'serialnumber', ':');
  512. }
  513. if (callback) { callback(result); }
  514. resolve(result);
  515. });
  516. } catch (e) {
  517. if (callback) { callback(result); }
  518. resolve(result);
  519. }
  520. }
  521. });
  522. });
  523. }
  524. exports.bios = bios;
  525. function baseboard(callback) {
  526. return new Promise((resolve) => {
  527. process.nextTick(() => {
  528. let result = {
  529. manufacturer: '',
  530. model: '',
  531. version: '',
  532. serial: '-',
  533. assetTag: '-',
  534. memMax: null,
  535. memSlots: null
  536. };
  537. let cmd = '';
  538. if (_linux || _freebsd || _openbsd || _netbsd) {
  539. if (process.arch === 'arm') {
  540. cmd = 'cat /proc/cpuinfo | grep Serial';
  541. // 'BCM2709', 'BCM2835', 'BCM2708' -->
  542. } else {
  543. cmd = 'export LC_ALL=C; dmidecode -t 2 2>/dev/null; unset LC_ALL';
  544. }
  545. const workload = [];
  546. workload.push(execPromise(cmd));
  547. workload.push(execPromise('export LC_ALL=C; dmidecode -t memory 2>/dev/null'));
  548. util.promiseAll(
  549. workload
  550. ).then(data => {
  551. let lines = data.results[0] ? data.results[0].toString().split('\n') : [''];
  552. result.manufacturer = util.getValue(lines, 'Manufacturer');
  553. result.model = util.getValue(lines, 'Product Name');
  554. result.version = util.getValue(lines, 'Version');
  555. result.serial = util.getValue(lines, 'Serial Number');
  556. result.assetTag = util.getValue(lines, 'Asset Tag');
  557. // Non-Root values
  558. const cmd = `echo -n "board_asset_tag: "; cat /sys/devices/virtual/dmi/id/board_asset_tag 2>/dev/null; echo;
  559. echo -n "board_name: "; cat /sys/devices/virtual/dmi/id/board_name 2>/dev/null; echo;
  560. echo -n "board_serial: "; cat /sys/devices/virtual/dmi/id/board_serial 2>/dev/null; echo;
  561. echo -n "board_vendor: "; cat /sys/devices/virtual/dmi/id/board_vendor 2>/dev/null; echo;
  562. echo -n "board_version: "; cat /sys/devices/virtual/dmi/id/board_version 2>/dev/null; echo;`;
  563. try {
  564. lines = execSync(cmd).toString().split('\n');
  565. result.manufacturer = !result.manufacturer ? util.getValue(lines, 'board_vendor') : result.manufacturer;
  566. result.model = !result.model ? util.getValue(lines, 'board_name') : result.model;
  567. result.version = !result.version ? util.getValue(lines, 'board_version') : result.version;
  568. result.serial = !result.serial ? util.getValue(lines, 'board_serial') : result.serial;
  569. result.assetTag = !result.assetTag ? util.getValue(lines, 'board_asset_tag') : result.assetTag;
  570. } catch (e) {
  571. util.noop();
  572. }
  573. if (result.serial.toLowerCase().indexOf('o.e.m.') !== -1) { result.serial = '-'; }
  574. if (result.assetTag.toLowerCase().indexOf('o.e.m.') !== -1) { result.assetTag = '-'; }
  575. // mem
  576. lines = data.results[1] ? data.results[1].toString().split('\n') : [''];
  577. result.memMax = util.toInt(util.getValue(lines, 'Maximum Capacity')) * 1024 * 1024 * 1024 || null;
  578. result.memSlots = util.toInt(util.getValue(lines, 'Number Of Devices')) || null;
  579. // raspberry
  580. let linesRpi = '';
  581. try {
  582. linesRpi = fs.readFileSync('/proc/cpuinfo').toString().split('\n');
  583. } catch (e) {
  584. util.noop();
  585. }
  586. const hardware = util.getValue(linesRpi, 'hardware');
  587. if (hardware.startsWith('BCM')) {
  588. const rpi = util.decodePiCpuinfo(linesRpi);
  589. result.manufacturer = rpi.manufacturer;
  590. result.model = 'Raspberry Pi';
  591. result.serial = rpi.serial;
  592. result.version = rpi.type + ' - ' + rpi.revision;
  593. result.memMax = os.totalmem();
  594. result.memSlots = 0;
  595. }
  596. if (callback) { callback(result); }
  597. resolve(result);
  598. });
  599. }
  600. if (_darwin) {
  601. const workload = [];
  602. workload.push(execPromise('ioreg -c IOPlatformExpertDevice -d 2'));
  603. workload.push(execPromise('system_profiler SPMemoryDataType'));
  604. util.promiseAll(
  605. workload
  606. ).then(data => {
  607. let lines = data.results[0] ? data.results[0].toString().replace(/[<>"]/g, '').split('\n') : [''];
  608. result.manufacturer = util.getValue(lines, 'manufacturer', '=', true);
  609. result.model = util.getValue(lines, 'model', '=', true);
  610. result.version = util.getValue(lines, 'version', '=', true);
  611. result.serial = util.getValue(lines, 'ioplatformserialnumber', '=', true);
  612. result.assetTag = util.getValue(lines, 'board-id', '=', true);
  613. // mem
  614. let devices = data.results[1] ? data.results[1].toString().split(' BANK ') : [''];
  615. if (devices.length === 1) {
  616. devices = data.results[1] ? data.results[1].toString().split(' DIMM') : [''];
  617. }
  618. devices.shift();
  619. result.memSlots = devices.length;
  620. if (os.arch() === 'arm64') {
  621. result.memSlots = 0;
  622. result.memMax = os.totalmem();
  623. }
  624. if (callback) { callback(result); }
  625. resolve(result);
  626. });
  627. }
  628. if (_sunos) {
  629. if (callback) { callback(result); }
  630. resolve(result);
  631. }
  632. if (_windows) {
  633. try {
  634. const workload = [];
  635. workload.push(util.powerShell('Get-WmiObject Win32_baseboard | select Model,Manufacturer,Product,Version,SerialNumber,PartNumber,SKU | fl'));
  636. workload.push(util.powerShell('Get-WmiObject Win32_physicalmemoryarray | select MaxCapacity, MemoryDevices | fl'));
  637. util.promiseAll(
  638. workload
  639. ).then(data => {
  640. let lines = data.results[0] ? data.results[0].toString().split('\r\n') : [''];
  641. result.manufacturer = util.getValue(lines, 'manufacturer', ':');
  642. result.model = util.getValue(lines, 'model', ':');
  643. if (!result.model) {
  644. result.model = util.getValue(lines, 'product', ':');
  645. }
  646. result.version = util.getValue(lines, 'version', ':');
  647. result.serial = util.getValue(lines, 'serialnumber', ':');
  648. result.assetTag = util.getValue(lines, 'partnumber', ':');
  649. if (!result.assetTag) {
  650. result.assetTag = util.getValue(lines, 'sku', ':');
  651. }
  652. // memphysical
  653. lines = data.results[1] ? data.results[1].toString().split('\r\n') : [''];
  654. result.memMax = util.toInt(util.getValue(lines, 'MaxCapacity', ':')) || null;
  655. result.memSlots = util.toInt(util.getValue(lines, 'MemoryDevices', ':')) || null;
  656. if (callback) { callback(result); }
  657. resolve(result);
  658. });
  659. } catch (e) {
  660. if (callback) { callback(result); }
  661. resolve(result);
  662. }
  663. }
  664. });
  665. });
  666. }
  667. exports.baseboard = baseboard;
  668. function chassis(callback) {
  669. const chassisTypes = ['Other',
  670. 'Unknown',
  671. 'Desktop',
  672. 'Low Profile Desktop',
  673. 'Pizza Box',
  674. 'Mini Tower',
  675. 'Tower',
  676. 'Portable',
  677. 'Laptop',
  678. 'Notebook',
  679. 'Hand Held',
  680. 'Docking Station',
  681. 'All in One',
  682. 'Sub Notebook',
  683. 'Space-Saving',
  684. 'Lunch Box',
  685. 'Main System Chassis',
  686. 'Expansion Chassis',
  687. 'SubChassis',
  688. 'Bus Expansion Chassis',
  689. 'Peripheral Chassis',
  690. 'Storage Chassis',
  691. 'Rack Mount Chassis',
  692. 'Sealed-Case PC',
  693. 'Multi-System Chassis',
  694. 'Compact PCI',
  695. 'Advanced TCA',
  696. 'Blade',
  697. 'Blade Enclosure',
  698. 'Tablet',
  699. 'Convertible',
  700. 'Detachable',
  701. 'IoT Gateway ',
  702. 'Embedded PC',
  703. 'Mini PC',
  704. 'Stick PC',
  705. ];
  706. return new Promise((resolve) => {
  707. process.nextTick(() => {
  708. let result = {
  709. manufacturer: '',
  710. model: '',
  711. type: '',
  712. version: '',
  713. serial: '-',
  714. assetTag: '-',
  715. sku: '',
  716. };
  717. if (_linux || _freebsd || _openbsd || _netbsd) {
  718. const cmd = `echo -n "chassis_asset_tag: "; cat /sys/devices/virtual/dmi/id/chassis_asset_tag 2>/dev/null; echo;
  719. echo -n "chassis_serial: "; cat /sys/devices/virtual/dmi/id/chassis_serial 2>/dev/null; echo;
  720. echo -n "chassis_type: "; cat /sys/devices/virtual/dmi/id/chassis_type 2>/dev/null; echo;
  721. echo -n "chassis_vendor: "; cat /sys/devices/virtual/dmi/id/chassis_vendor 2>/dev/null; echo;
  722. echo -n "chassis_version: "; cat /sys/devices/virtual/dmi/id/chassis_version 2>/dev/null; echo;`;
  723. exec(cmd, function (error, stdout) {
  724. let lines = stdout.toString().split('\n');
  725. result.manufacturer = util.getValue(lines, 'chassis_vendor');
  726. const ctype = parseInt(util.getValue(lines, 'chassis_type').replace(/\D/g, ''));
  727. result.type = (ctype && !isNaN(ctype) && ctype < chassisTypes.length) ? chassisTypes[ctype - 1] : '';
  728. result.version = util.getValue(lines, 'chassis_version');
  729. result.serial = util.getValue(lines, 'chassis_serial');
  730. result.assetTag = util.getValue(lines, 'chassis_asset_tag');
  731. if (result.manufacturer.toLowerCase().indexOf('o.e.m.') !== -1) { result.manufacturer = '-'; }
  732. if (result.version.toLowerCase().indexOf('o.e.m.') !== -1) { result.version = '-'; }
  733. if (result.serial.toLowerCase().indexOf('o.e.m.') !== -1) { result.serial = '-'; }
  734. if (result.assetTag.toLowerCase().indexOf('o.e.m.') !== -1) { result.assetTag = '-'; }
  735. if (callback) { callback(result); }
  736. resolve(result);
  737. });
  738. }
  739. if (_darwin) {
  740. exec('ioreg -c IOPlatformExpertDevice -d 2', function (error, stdout) {
  741. if (!error) {
  742. let lines = stdout.toString().replace(/[<>"]/g, '').split('\n');
  743. result.manufacturer = util.getValue(lines, 'manufacturer', '=', true);
  744. result.model = util.getValue(lines, 'model', '=', true);
  745. result.version = util.getValue(lines, 'version', '=', true);
  746. result.serial = util.getValue(lines, 'ioplatformserialnumber', '=', true);
  747. result.assetTag = util.getValue(lines, 'board-id', '=', true);
  748. }
  749. if (callback) { callback(result); }
  750. resolve(result);
  751. });
  752. }
  753. if (_sunos) {
  754. if (callback) { callback(result); }
  755. resolve(result);
  756. }
  757. if (_windows) {
  758. try {
  759. util.powerShell('Get-WmiObject Win32_SystemEnclosure | select Model,Manufacturer,ChassisTypes,Version,SerialNumber,PartNumber,SKU | fl').then((stdout, error) => {
  760. if (!error) {
  761. let lines = stdout.toString().split('\r\n');
  762. result.manufacturer = util.getValue(lines, 'manufacturer', ':');
  763. result.model = util.getValue(lines, 'model', ':');
  764. const ctype = parseInt(util.getValue(lines, 'ChassisTypes', ':').replace(/\D/g, ''));
  765. result.type = (ctype && !isNaN(ctype) && ctype < chassisTypes.length) ? chassisTypes[ctype - 1] : '';
  766. result.version = util.getValue(lines, 'version', ':');
  767. result.serial = util.getValue(lines, 'serialnumber', ':');
  768. result.assetTag = util.getValue(lines, 'partnumber', ':');
  769. result.sku = util.getValue(lines, 'sku', ':');
  770. if (result.manufacturer.toLowerCase().indexOf('o.e.m.') !== -1) { result.manufacturer = '-'; }
  771. if (result.version.toLowerCase().indexOf('o.e.m.') !== -1) { result.version = '-'; }
  772. if (result.serial.toLowerCase().indexOf('o.e.m.') !== -1) { result.serial = '-'; }
  773. if (result.assetTag.toLowerCase().indexOf('o.e.m.') !== -1) { result.assetTag = '-'; }
  774. }
  775. if (callback) { callback(result); }
  776. resolve(result);
  777. });
  778. } catch (e) {
  779. if (callback) { callback(result); }
  780. resolve(result);
  781. }
  782. }
  783. });
  784. });
  785. }
  786. exports.chassis = chassis;