host_windows.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. // +build windows
  2. package host
  3. import (
  4. "context"
  5. "fmt"
  6. "math"
  7. "os"
  8. "runtime"
  9. "strings"
  10. "sync/atomic"
  11. "syscall"
  12. "time"
  13. "unsafe"
  14. "github.com/StackExchange/wmi"
  15. "github.com/shirou/gopsutil/internal/common"
  16. process "github.com/shirou/gopsutil/process"
  17. "golang.org/x/sys/windows"
  18. )
  19. var (
  20. procGetSystemTimeAsFileTime = common.Modkernel32.NewProc("GetSystemTimeAsFileTime")
  21. procGetTickCount32 = common.Modkernel32.NewProc("GetTickCount")
  22. procGetTickCount64 = common.Modkernel32.NewProc("GetTickCount64")
  23. procRtlGetVersion = common.ModNt.NewProc("RtlGetVersion")
  24. )
  25. // https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/ns-wdm-_osversioninfoexw
  26. type osVersionInfoExW struct {
  27. dwOSVersionInfoSize uint32
  28. dwMajorVersion uint32
  29. dwMinorVersion uint32
  30. dwBuildNumber uint32
  31. dwPlatformId uint32
  32. szCSDVersion [128]uint16
  33. wServicePackMajor uint16
  34. wServicePackMinor uint16
  35. wSuiteMask uint16
  36. wProductType uint8
  37. wReserved uint8
  38. }
  39. type msAcpi_ThermalZoneTemperature struct {
  40. Active bool
  41. CriticalTripPoint uint32
  42. CurrentTemperature uint32
  43. InstanceName string
  44. }
  45. func Info() (*InfoStat, error) {
  46. return InfoWithContext(context.Background())
  47. }
  48. func InfoWithContext(ctx context.Context) (*InfoStat, error) {
  49. ret := &InfoStat{
  50. OS: runtime.GOOS,
  51. }
  52. {
  53. hostname, err := os.Hostname()
  54. if err == nil {
  55. ret.Hostname = hostname
  56. }
  57. }
  58. {
  59. platform, family, version, err := PlatformInformationWithContext(ctx)
  60. if err == nil {
  61. ret.Platform = platform
  62. ret.PlatformFamily = family
  63. ret.PlatformVersion = version
  64. } else {
  65. return ret, err
  66. }
  67. }
  68. {
  69. boot, err := BootTime()
  70. if err == nil {
  71. ret.BootTime = boot
  72. ret.Uptime, _ = Uptime()
  73. }
  74. }
  75. {
  76. hostID, err := getMachineGuid()
  77. if err == nil {
  78. ret.HostID = strings.ToLower(hostID)
  79. }
  80. }
  81. {
  82. procs, err := process.Pids()
  83. if err == nil {
  84. ret.Procs = uint64(len(procs))
  85. }
  86. }
  87. return ret, nil
  88. }
  89. func getMachineGuid() (string, error) {
  90. // there has been reports of issues on 32bit using golang.org/x/sys/windows/registry, see https://github.com/shirou/gopsutil/pull/312#issuecomment-277422612
  91. // for rationale of using windows.RegOpenKeyEx/RegQueryValueEx instead of registry.OpenKey/GetStringValue
  92. var h windows.Handle
  93. err := windows.RegOpenKeyEx(windows.HKEY_LOCAL_MACHINE, windows.StringToUTF16Ptr(`SOFTWARE\Microsoft\Cryptography`), 0, windows.KEY_READ|windows.KEY_WOW64_64KEY, &h)
  94. if err != nil {
  95. return "", err
  96. }
  97. defer windows.RegCloseKey(h)
  98. const windowsRegBufLen = 74 // len(`{`) + len(`abcdefgh-1234-456789012-123345456671` * 2) + len(`}`) // 2 == bytes/UTF16
  99. const uuidLen = 36
  100. var regBuf [windowsRegBufLen]uint16
  101. bufLen := uint32(windowsRegBufLen)
  102. var valType uint32
  103. err = windows.RegQueryValueEx(h, windows.StringToUTF16Ptr(`MachineGuid`), nil, &valType, (*byte)(unsafe.Pointer(&regBuf[0])), &bufLen)
  104. if err != nil {
  105. return "", err
  106. }
  107. hostID := windows.UTF16ToString(regBuf[:])
  108. hostIDLen := len(hostID)
  109. if hostIDLen != uuidLen {
  110. return "", fmt.Errorf("HostID incorrect: %q\n", hostID)
  111. }
  112. return hostID, nil
  113. }
  114. func Uptime() (uint64, error) {
  115. return UptimeWithContext(context.Background())
  116. }
  117. func UptimeWithContext(ctx context.Context) (uint64, error) {
  118. procGetTickCount := procGetTickCount64
  119. err := procGetTickCount64.Find()
  120. if err != nil {
  121. procGetTickCount = procGetTickCount32 // handle WinXP, but keep in mind that "the time will wrap around to zero if the system is run continuously for 49.7 days." from MSDN
  122. }
  123. r1, _, lastErr := syscall.Syscall(procGetTickCount.Addr(), 0, 0, 0, 0)
  124. if lastErr != 0 {
  125. return 0, lastErr
  126. }
  127. return uint64((time.Duration(r1) * time.Millisecond).Seconds()), nil
  128. }
  129. func bootTimeFromUptime(up uint64) uint64 {
  130. return uint64(time.Now().Unix()) - up
  131. }
  132. // cachedBootTime must be accessed via atomic.Load/StoreUint64
  133. var cachedBootTime uint64
  134. func BootTime() (uint64, error) {
  135. return BootTimeWithContext(context.Background())
  136. }
  137. func BootTimeWithContext(ctx context.Context) (uint64, error) {
  138. t := atomic.LoadUint64(&cachedBootTime)
  139. if t != 0 {
  140. return t, nil
  141. }
  142. up, err := Uptime()
  143. if err != nil {
  144. return 0, err
  145. }
  146. t = bootTimeFromUptime(up)
  147. atomic.StoreUint64(&cachedBootTime, t)
  148. return t, nil
  149. }
  150. func PlatformInformation() (platform string, family string, version string, err error) {
  151. return PlatformInformationWithContext(context.Background())
  152. }
  153. func PlatformInformationWithContext(ctx context.Context) (platform string, family string, version string, err error) {
  154. // GetVersionEx lies on Windows 8.1 and returns as Windows 8 if we don't declare compatibility in manifest
  155. // RtlGetVersion bypasses this lying layer and returns the true Windows version
  156. // https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/nf-wdm-rtlgetversion
  157. // https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/ns-wdm-_osversioninfoexw
  158. var osInfo osVersionInfoExW
  159. osInfo.dwOSVersionInfoSize = uint32(unsafe.Sizeof(osInfo))
  160. ret, _, err := procRtlGetVersion.Call(uintptr(unsafe.Pointer(&osInfo)))
  161. if ret != 0 {
  162. return
  163. }
  164. // Platform
  165. var h windows.Handle // like getMachineGuid(), we query the registry using the raw windows.RegOpenKeyEx/RegQueryValueEx
  166. err = windows.RegOpenKeyEx(windows.HKEY_LOCAL_MACHINE, windows.StringToUTF16Ptr(`SOFTWARE\Microsoft\Windows NT\CurrentVersion`), 0, windows.KEY_READ|windows.KEY_WOW64_64KEY, &h)
  167. if err != nil {
  168. return
  169. }
  170. defer windows.RegCloseKey(h)
  171. var bufLen uint32
  172. var valType uint32
  173. err = windows.RegQueryValueEx(h, windows.StringToUTF16Ptr(`ProductName`), nil, &valType, nil, &bufLen)
  174. if err != nil {
  175. return
  176. }
  177. regBuf := make([]uint16, bufLen/2+1)
  178. err = windows.RegQueryValueEx(h, windows.StringToUTF16Ptr(`ProductName`), nil, &valType, (*byte)(unsafe.Pointer(&regBuf[0])), &bufLen)
  179. if err != nil {
  180. return
  181. }
  182. platform = windows.UTF16ToString(regBuf[:])
  183. if !strings.HasPrefix(platform, "Microsoft") {
  184. platform = "Microsoft " + platform
  185. }
  186. err = windows.RegQueryValueEx(h, windows.StringToUTF16Ptr(`CSDVersion`), nil, &valType, nil, &bufLen) // append Service Pack number, only on success
  187. if err == nil { // don't return an error if only the Service Pack retrieval fails
  188. regBuf = make([]uint16, bufLen/2+1)
  189. err = windows.RegQueryValueEx(h, windows.StringToUTF16Ptr(`CSDVersion`), nil, &valType, (*byte)(unsafe.Pointer(&regBuf[0])), &bufLen)
  190. if err == nil {
  191. platform += " " + windows.UTF16ToString(regBuf[:])
  192. }
  193. }
  194. // PlatformFamily
  195. switch osInfo.wProductType {
  196. case 1:
  197. family = "Standalone Workstation"
  198. case 2:
  199. family = "Server (Domain Controller)"
  200. case 3:
  201. family = "Server"
  202. }
  203. // Platform Version
  204. version = fmt.Sprintf("%d.%d.%d Build %d", osInfo.dwMajorVersion, osInfo.dwMinorVersion, osInfo.dwBuildNumber, osInfo.dwBuildNumber)
  205. return platform, family, version, nil
  206. }
  207. func Users() ([]UserStat, error) {
  208. return UsersWithContext(context.Background())
  209. }
  210. func UsersWithContext(ctx context.Context) ([]UserStat, error) {
  211. var ret []UserStat
  212. return ret, nil
  213. }
  214. func SensorsTemperatures() ([]TemperatureStat, error) {
  215. return SensorsTemperaturesWithContext(context.Background())
  216. }
  217. func SensorsTemperaturesWithContext(ctx context.Context) ([]TemperatureStat, error) {
  218. var ret []TemperatureStat
  219. var dst []msAcpi_ThermalZoneTemperature
  220. q := wmi.CreateQuery(&dst, "")
  221. if err := common.WMIQueryWithContext(ctx, q, &dst, nil, "root/wmi"); err != nil {
  222. return ret, err
  223. }
  224. for _, v := range dst {
  225. ts := TemperatureStat{
  226. SensorKey: v.InstanceName,
  227. Temperature: kelvinToCelsius(v.CurrentTemperature, 2),
  228. }
  229. ret = append(ret, ts)
  230. }
  231. return ret, nil
  232. }
  233. func kelvinToCelsius(temp uint32, n int) float64 {
  234. // wmi return temperature Kelvin * 10, so need to divide the result by 10,
  235. // and then minus 273.15 to get °Celsius.
  236. t := float64(temp/10) - 273.15
  237. n10 := math.Pow10(n)
  238. return math.Trunc((t+0.5/n10)*n10) / n10
  239. }
  240. func Virtualization() (string, string, error) {
  241. return VirtualizationWithContext(context.Background())
  242. }
  243. func VirtualizationWithContext(ctx context.Context) (string, string, error) {
  244. return "", "", common.ErrNotImplementedError
  245. }
  246. func KernelVersion() (string, error) {
  247. return KernelVersionWithContext(context.Background())
  248. }
  249. func KernelVersionWithContext(ctx context.Context) (string, error) {
  250. _, _, version, err := PlatformInformation()
  251. return version, err
  252. }