host_linux.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669
  1. // +build linux
  2. package host
  3. import (
  4. "bytes"
  5. "context"
  6. "encoding/binary"
  7. "fmt"
  8. "io/ioutil"
  9. "os"
  10. "os/exec"
  11. "path/filepath"
  12. "regexp"
  13. "runtime"
  14. "strconv"
  15. "strings"
  16. "sync/atomic"
  17. "time"
  18. "github.com/shirou/gopsutil/internal/common"
  19. )
  20. type LSB struct {
  21. ID string
  22. Release string
  23. Codename string
  24. Description string
  25. }
  26. // from utmp.h
  27. const USER_PROCESS = 7
  28. func Info() (*InfoStat, error) {
  29. return InfoWithContext(context.Background())
  30. }
  31. func InfoWithContext(ctx context.Context) (*InfoStat, error) {
  32. ret := &InfoStat{
  33. OS: runtime.GOOS,
  34. }
  35. hostname, err := os.Hostname()
  36. if err == nil {
  37. ret.Hostname = hostname
  38. }
  39. platform, family, version, err := PlatformInformation()
  40. if err == nil {
  41. ret.Platform = platform
  42. ret.PlatformFamily = family
  43. ret.PlatformVersion = version
  44. }
  45. kernelVersion, err := KernelVersion()
  46. if err == nil {
  47. ret.KernelVersion = kernelVersion
  48. }
  49. system, role, err := Virtualization()
  50. if err == nil {
  51. ret.VirtualizationSystem = system
  52. ret.VirtualizationRole = role
  53. }
  54. boot, err := BootTime()
  55. if err == nil {
  56. ret.BootTime = boot
  57. ret.Uptime = uptime(boot)
  58. }
  59. if numProcs, err := common.NumProcs(); err == nil {
  60. ret.Procs = numProcs
  61. }
  62. sysProductUUID := common.HostSys("class/dmi/id/product_uuid")
  63. switch {
  64. case common.PathExists(sysProductUUID):
  65. lines, err := common.ReadLines(sysProductUUID)
  66. if err == nil && len(lines) > 0 && lines[0] != "" {
  67. ret.HostID = strings.ToLower(lines[0])
  68. break
  69. }
  70. fallthrough
  71. default:
  72. values, err := common.DoSysctrl("kernel.random.boot_id")
  73. if err == nil && len(values) == 1 && values[0] != "" {
  74. ret.HostID = strings.ToLower(values[0])
  75. }
  76. }
  77. return ret, nil
  78. }
  79. // cachedBootTime must be accessed via atomic.Load/StoreUint64
  80. var cachedBootTime uint64
  81. // BootTime returns the system boot time expressed in seconds since the epoch.
  82. func BootTime() (uint64, error) {
  83. return BootTimeWithContext(context.Background())
  84. }
  85. func BootTimeWithContext(ctx context.Context) (uint64, error) {
  86. t := atomic.LoadUint64(&cachedBootTime)
  87. if t != 0 {
  88. return t, nil
  89. }
  90. system, role, err := Virtualization()
  91. if err != nil {
  92. return 0, err
  93. }
  94. statFile := "stat"
  95. if system == "lxc" && role == "guest" {
  96. // if lxc, /proc/uptime is used.
  97. statFile = "uptime"
  98. } else if system == "docker" && role == "guest" {
  99. // also docker, guest
  100. statFile = "uptime"
  101. }
  102. filename := common.HostProc(statFile)
  103. lines, err := common.ReadLines(filename)
  104. if err != nil {
  105. return 0, err
  106. }
  107. if statFile == "stat" {
  108. for _, line := range lines {
  109. if strings.HasPrefix(line, "btime") {
  110. f := strings.Fields(line)
  111. if len(f) != 2 {
  112. return 0, fmt.Errorf("wrong btime format")
  113. }
  114. b, err := strconv.ParseInt(f[1], 10, 64)
  115. if err != nil {
  116. return 0, err
  117. }
  118. t = uint64(b)
  119. atomic.StoreUint64(&cachedBootTime, t)
  120. return t, nil
  121. }
  122. }
  123. } else if statFile == "uptime" {
  124. if len(lines) != 1 {
  125. return 0, fmt.Errorf("wrong uptime format")
  126. }
  127. f := strings.Fields(lines[0])
  128. b, err := strconv.ParseFloat(f[0], 64)
  129. if err != nil {
  130. return 0, err
  131. }
  132. t = uint64(time.Now().Unix()) - uint64(b)
  133. atomic.StoreUint64(&cachedBootTime, t)
  134. return t, nil
  135. }
  136. return 0, fmt.Errorf("could not find btime")
  137. }
  138. func uptime(boot uint64) uint64 {
  139. return uint64(time.Now().Unix()) - boot
  140. }
  141. func Uptime() (uint64, error) {
  142. return UptimeWithContext(context.Background())
  143. }
  144. func UptimeWithContext(ctx context.Context) (uint64, error) {
  145. boot, err := BootTime()
  146. if err != nil {
  147. return 0, err
  148. }
  149. return uptime(boot), nil
  150. }
  151. func Users() ([]UserStat, error) {
  152. return UsersWithContext(context.Background())
  153. }
  154. func UsersWithContext(ctx context.Context) ([]UserStat, error) {
  155. utmpfile := common.HostVar("run/utmp")
  156. file, err := os.Open(utmpfile)
  157. if err != nil {
  158. return nil, err
  159. }
  160. defer file.Close()
  161. buf, err := ioutil.ReadAll(file)
  162. if err != nil {
  163. return nil, err
  164. }
  165. count := len(buf) / sizeOfUtmp
  166. ret := make([]UserStat, 0, count)
  167. for i := 0; i < count; i++ {
  168. b := buf[i*sizeOfUtmp : (i+1)*sizeOfUtmp]
  169. var u utmp
  170. br := bytes.NewReader(b)
  171. err := binary.Read(br, binary.LittleEndian, &u)
  172. if err != nil {
  173. continue
  174. }
  175. if u.Type != USER_PROCESS {
  176. continue
  177. }
  178. user := UserStat{
  179. User: common.IntToString(u.User[:]),
  180. Terminal: common.IntToString(u.Line[:]),
  181. Host: common.IntToString(u.Host[:]),
  182. Started: int(u.Tv.Sec),
  183. }
  184. ret = append(ret, user)
  185. }
  186. return ret, nil
  187. }
  188. func getOSRelease() (platform string, version string, err error) {
  189. contents, err := common.ReadLines(common.HostEtc("os-release"))
  190. if err != nil {
  191. return "", "", nil // return empty
  192. }
  193. for _, line := range contents {
  194. field := strings.Split(line, "=")
  195. if len(field) < 2 {
  196. continue
  197. }
  198. switch field[0] {
  199. case "ID": // use ID for lowercase
  200. platform = field[1]
  201. case "VERSION":
  202. version = field[1]
  203. }
  204. }
  205. return platform, version, nil
  206. }
  207. func getLSB() (*LSB, error) {
  208. ret := &LSB{}
  209. if common.PathExists(common.HostEtc("lsb-release")) {
  210. contents, err := common.ReadLines(common.HostEtc("lsb-release"))
  211. if err != nil {
  212. return ret, err // return empty
  213. }
  214. for _, line := range contents {
  215. field := strings.Split(line, "=")
  216. if len(field) < 2 {
  217. continue
  218. }
  219. switch field[0] {
  220. case "DISTRIB_ID":
  221. ret.ID = field[1]
  222. case "DISTRIB_RELEASE":
  223. ret.Release = field[1]
  224. case "DISTRIB_CODENAME":
  225. ret.Codename = field[1]
  226. case "DISTRIB_DESCRIPTION":
  227. ret.Description = field[1]
  228. }
  229. }
  230. } else if common.PathExists("/usr/bin/lsb_release") {
  231. lsb_release, err := exec.LookPath("/usr/bin/lsb_release")
  232. if err != nil {
  233. return ret, err
  234. }
  235. out, err := invoke.Command(lsb_release)
  236. if err != nil {
  237. return ret, err
  238. }
  239. for _, line := range strings.Split(string(out), "\n") {
  240. field := strings.Split(line, ":")
  241. if len(field) < 2 {
  242. continue
  243. }
  244. switch field[0] {
  245. case "Distributor ID":
  246. ret.ID = field[1]
  247. case "Release":
  248. ret.Release = field[1]
  249. case "Codename":
  250. ret.Codename = field[1]
  251. case "Description":
  252. ret.Description = field[1]
  253. }
  254. }
  255. }
  256. return ret, nil
  257. }
  258. func PlatformInformation() (platform string, family string, version string, err error) {
  259. return PlatformInformationWithContext(context.Background())
  260. }
  261. func PlatformInformationWithContext(ctx context.Context) (platform string, family string, version string, err error) {
  262. lsb, err := getLSB()
  263. if err != nil {
  264. lsb = &LSB{}
  265. }
  266. if common.PathExists(common.HostEtc("oracle-release")) {
  267. platform = "oracle"
  268. contents, err := common.ReadLines(common.HostEtc("oracle-release"))
  269. if err == nil {
  270. version = getRedhatishVersion(contents)
  271. }
  272. } else if common.PathExists(common.HostEtc("enterprise-release")) {
  273. platform = "oracle"
  274. contents, err := common.ReadLines(common.HostEtc("enterprise-release"))
  275. if err == nil {
  276. version = getRedhatishVersion(contents)
  277. }
  278. } else if common.PathExists(common.HostEtc("slackware-version")) {
  279. platform = "slackware"
  280. contents, err := common.ReadLines(common.HostEtc("slackware-version"))
  281. if err == nil {
  282. version = getSlackwareVersion(contents)
  283. }
  284. } else if common.PathExists(common.HostEtc("debian_version")) {
  285. if lsb.ID == "Ubuntu" {
  286. platform = "ubuntu"
  287. version = lsb.Release
  288. } else if lsb.ID == "LinuxMint" {
  289. platform = "linuxmint"
  290. version = lsb.Release
  291. } else {
  292. if common.PathExists("/usr/bin/raspi-config") {
  293. platform = "raspbian"
  294. } else {
  295. platform = "debian"
  296. }
  297. contents, err := common.ReadLines(common.HostEtc("debian_version"))
  298. if err == nil {
  299. version = contents[0]
  300. }
  301. }
  302. } else if common.PathExists(common.HostEtc("redhat-release")) {
  303. contents, err := common.ReadLines(common.HostEtc("redhat-release"))
  304. if err == nil {
  305. version = getRedhatishVersion(contents)
  306. platform = getRedhatishPlatform(contents)
  307. }
  308. } else if common.PathExists(common.HostEtc("system-release")) {
  309. contents, err := common.ReadLines(common.HostEtc("system-release"))
  310. if err == nil {
  311. version = getRedhatishVersion(contents)
  312. platform = getRedhatishPlatform(contents)
  313. }
  314. } else if common.PathExists(common.HostEtc("gentoo-release")) {
  315. platform = "gentoo"
  316. contents, err := common.ReadLines(common.HostEtc("gentoo-release"))
  317. if err == nil {
  318. version = getRedhatishVersion(contents)
  319. }
  320. } else if common.PathExists(common.HostEtc("SuSE-release")) {
  321. contents, err := common.ReadLines(common.HostEtc("SuSE-release"))
  322. if err == nil {
  323. version = getSuseVersion(contents)
  324. platform = getSusePlatform(contents)
  325. }
  326. // TODO: slackware detecion
  327. } else if common.PathExists(common.HostEtc("arch-release")) {
  328. platform = "arch"
  329. version = lsb.Release
  330. } else if common.PathExists(common.HostEtc("alpine-release")) {
  331. platform = "alpine"
  332. contents, err := common.ReadLines(common.HostEtc("alpine-release"))
  333. if err == nil && len(contents) > 0 {
  334. version = contents[0]
  335. }
  336. } else if common.PathExists(common.HostEtc("os-release")) {
  337. p, v, err := getOSRelease()
  338. if err == nil {
  339. platform = p
  340. version = v
  341. }
  342. } else if lsb.ID == "RedHat" {
  343. platform = "redhat"
  344. version = lsb.Release
  345. } else if lsb.ID == "Amazon" {
  346. platform = "amazon"
  347. version = lsb.Release
  348. } else if lsb.ID == "ScientificSL" {
  349. platform = "scientific"
  350. version = lsb.Release
  351. } else if lsb.ID == "XenServer" {
  352. platform = "xenserver"
  353. version = lsb.Release
  354. } else if lsb.ID != "" {
  355. platform = strings.ToLower(lsb.ID)
  356. version = lsb.Release
  357. }
  358. switch platform {
  359. case "debian", "ubuntu", "linuxmint", "raspbian":
  360. family = "debian"
  361. case "fedora":
  362. family = "fedora"
  363. case "oracle", "centos", "redhat", "scientific", "enterpriseenterprise", "amazon", "xenserver", "cloudlinux", "ibm_powerkvm":
  364. family = "rhel"
  365. case "suse", "opensuse":
  366. family = "suse"
  367. case "gentoo":
  368. family = "gentoo"
  369. case "slackware":
  370. family = "slackware"
  371. case "arch":
  372. family = "arch"
  373. case "exherbo":
  374. family = "exherbo"
  375. case "alpine":
  376. family = "alpine"
  377. case "coreos":
  378. family = "coreos"
  379. }
  380. return platform, family, version, nil
  381. }
  382. func KernelVersion() (version string, err error) {
  383. return KernelVersionWithContext(context.Background())
  384. }
  385. func KernelVersionWithContext(ctx context.Context) (version string, err error) {
  386. filename := common.HostProc("sys/kernel/osrelease")
  387. if common.PathExists(filename) {
  388. contents, err := common.ReadLines(filename)
  389. if err != nil {
  390. return "", err
  391. }
  392. if len(contents) > 0 {
  393. version = contents[0]
  394. }
  395. }
  396. return version, nil
  397. }
  398. func getSlackwareVersion(contents []string) string {
  399. c := strings.ToLower(strings.Join(contents, ""))
  400. c = strings.Replace(c, "slackware ", "", 1)
  401. return c
  402. }
  403. func getRedhatishVersion(contents []string) string {
  404. c := strings.ToLower(strings.Join(contents, ""))
  405. if strings.Contains(c, "rawhide") {
  406. return "rawhide"
  407. }
  408. if matches := regexp.MustCompile(`release (\d[\d.]*)`).FindStringSubmatch(c); matches != nil {
  409. return matches[1]
  410. }
  411. return ""
  412. }
  413. func getRedhatishPlatform(contents []string) string {
  414. c := strings.ToLower(strings.Join(contents, ""))
  415. if strings.Contains(c, "red hat") {
  416. return "redhat"
  417. }
  418. f := strings.Split(c, " ")
  419. return f[0]
  420. }
  421. func getSuseVersion(contents []string) string {
  422. version := ""
  423. for _, line := range contents {
  424. if matches := regexp.MustCompile(`VERSION = ([\d.]+)`).FindStringSubmatch(line); matches != nil {
  425. version = matches[1]
  426. } else if matches := regexp.MustCompile(`PATCHLEVEL = ([\d]+)`).FindStringSubmatch(line); matches != nil {
  427. version = version + "." + matches[1]
  428. }
  429. }
  430. return version
  431. }
  432. func getSusePlatform(contents []string) string {
  433. c := strings.ToLower(strings.Join(contents, ""))
  434. if strings.Contains(c, "opensuse") {
  435. return "opensuse"
  436. }
  437. return "suse"
  438. }
  439. func Virtualization() (string, string, error) {
  440. return VirtualizationWithContext(context.Background())
  441. }
  442. func VirtualizationWithContext(ctx context.Context) (string, string, error) {
  443. var system string
  444. var role string
  445. filename := common.HostProc("xen")
  446. if common.PathExists(filename) {
  447. system = "xen"
  448. role = "guest" // assume guest
  449. if common.PathExists(filepath.Join(filename, "capabilities")) {
  450. contents, err := common.ReadLines(filepath.Join(filename, "capabilities"))
  451. if err == nil {
  452. if common.StringsContains(contents, "control_d") {
  453. role = "host"
  454. }
  455. }
  456. }
  457. }
  458. filename = common.HostProc("modules")
  459. if common.PathExists(filename) {
  460. contents, err := common.ReadLines(filename)
  461. if err == nil {
  462. if common.StringsContains(contents, "kvm") {
  463. system = "kvm"
  464. role = "host"
  465. } else if common.StringsContains(contents, "vboxdrv") {
  466. system = "vbox"
  467. role = "host"
  468. } else if common.StringsContains(contents, "vboxguest") {
  469. system = "vbox"
  470. role = "guest"
  471. } else if common.StringsContains(contents, "vmware") {
  472. system = "vmware"
  473. role = "guest"
  474. }
  475. }
  476. }
  477. filename = common.HostProc("cpuinfo")
  478. if common.PathExists(filename) {
  479. contents, err := common.ReadLines(filename)
  480. if err == nil {
  481. if common.StringsContains(contents, "QEMU Virtual CPU") ||
  482. common.StringsContains(contents, "Common KVM processor") ||
  483. common.StringsContains(contents, "Common 32-bit KVM processor") {
  484. system = "kvm"
  485. role = "guest"
  486. }
  487. }
  488. }
  489. filename = common.HostProc()
  490. if common.PathExists(filepath.Join(filename, "bc", "0")) {
  491. system = "openvz"
  492. role = "host"
  493. } else if common.PathExists(filepath.Join(filename, "vz")) {
  494. system = "openvz"
  495. role = "guest"
  496. }
  497. // not use dmidecode because it requires root
  498. if common.PathExists(filepath.Join(filename, "self", "status")) {
  499. contents, err := common.ReadLines(filepath.Join(filename, "self", "status"))
  500. if err == nil {
  501. if common.StringsContains(contents, "s_context:") ||
  502. common.StringsContains(contents, "VxID:") {
  503. system = "linux-vserver"
  504. }
  505. // TODO: guest or host
  506. }
  507. }
  508. if common.PathExists(filepath.Join(filename, "self", "cgroup")) {
  509. contents, err := common.ReadLines(filepath.Join(filename, "self", "cgroup"))
  510. if err == nil {
  511. if common.StringsContains(contents, "lxc") {
  512. system = "lxc"
  513. role = "guest"
  514. } else if common.StringsContains(contents, "docker") {
  515. system = "docker"
  516. role = "guest"
  517. } else if common.StringsContains(contents, "machine-rkt") {
  518. system = "rkt"
  519. role = "guest"
  520. } else if common.PathExists("/usr/bin/lxc-version") {
  521. system = "lxc"
  522. role = "host"
  523. }
  524. }
  525. }
  526. if common.PathExists(common.HostEtc("os-release")) {
  527. p, _, err := getOSRelease()
  528. if err == nil && p == "coreos" {
  529. system = "rkt" // Is it true?
  530. role = "host"
  531. }
  532. }
  533. return system, role, nil
  534. }
  535. func SensorsTemperatures() ([]TemperatureStat, error) {
  536. return SensorsTemperaturesWithContext(context.Background())
  537. }
  538. func SensorsTemperaturesWithContext(ctx context.Context) ([]TemperatureStat, error) {
  539. var temperatures []TemperatureStat
  540. files, err := filepath.Glob(common.HostSys("/class/hwmon/hwmon*/temp*_*"))
  541. if err != nil {
  542. return temperatures, err
  543. }
  544. if len(files) == 0 {
  545. // CentOS has an intermediate /device directory:
  546. // https://github.com/giampaolo/psutil/issues/971
  547. files, err = filepath.Glob(common.HostSys("/class/hwmon/hwmon*/device/temp*_*"))
  548. if err != nil {
  549. return temperatures, err
  550. }
  551. }
  552. // example directory
  553. // device/ temp1_crit_alarm temp2_crit_alarm temp3_crit_alarm temp4_crit_alarm temp5_crit_alarm temp6_crit_alarm temp7_crit_alarm
  554. // name temp1_input temp2_input temp3_input temp4_input temp5_input temp6_input temp7_input
  555. // power/ temp1_label temp2_label temp3_label temp4_label temp5_label temp6_label temp7_label
  556. // subsystem/ temp1_max temp2_max temp3_max temp4_max temp5_max temp6_max temp7_max
  557. // temp1_crit temp2_crit temp3_crit temp4_crit temp5_crit temp6_crit temp7_crit uevent
  558. for _, file := range files {
  559. filename := strings.Split(filepath.Base(file), "_")
  560. if filename[1] == "label" {
  561. // Do not try to read the temperature of the label file
  562. continue
  563. }
  564. // Get the label of the temperature you are reading
  565. var label string
  566. c, _ := ioutil.ReadFile(filepath.Join(filepath.Dir(file), filename[0]+"_label"))
  567. if c != nil {
  568. //format the label from "Core 0" to "core0_"
  569. label = fmt.Sprintf("%s_", strings.Join(strings.Split(strings.TrimSpace(strings.ToLower(string(c))), " "), ""))
  570. }
  571. // Get the name of the tempearture you are reading
  572. name, err := ioutil.ReadFile(filepath.Join(filepath.Dir(file), "name"))
  573. if err != nil {
  574. return temperatures, err
  575. }
  576. // Get the temperature reading
  577. current, err := ioutil.ReadFile(file)
  578. if err != nil {
  579. return temperatures, err
  580. }
  581. temperature, err := strconv.ParseFloat(strings.TrimSpace(string(current)), 64)
  582. if err != nil {
  583. continue
  584. }
  585. tempName := strings.TrimSpace(strings.ToLower(string(strings.Join(filename[1:], ""))))
  586. temperatures = append(temperatures, TemperatureStat{
  587. SensorKey: fmt.Sprintf("%s_%s%s", strings.TrimSpace(string(name)), label, tempName),
  588. Temperature: temperature / 1000.0,
  589. })
  590. }
  591. return temperatures, nil
  592. }