2017-02-01 10:26:00 -08:00
// +build windows
package host
import (
2018-10-19 11:33:23 -07:00
"context"
2017-02-01 10:26:00 -08:00
"fmt"
2018-10-19 11:33:23 -07:00
"math"
2017-02-01 10:26:00 -08:00
"os"
"runtime"
"strings"
2018-10-19 11:33:23 -07:00
"sync/atomic"
2017-02-02 16:11:54 -08:00
"syscall"
2017-02-01 10:26:00 -08:00
"time"
2017-02-02 16:11:54 -08:00
"unsafe"
2017-02-01 10:26:00 -08:00
"github.com/StackExchange/wmi"
"github.com/shirou/gopsutil/internal/common"
process "github.com/shirou/gopsutil/process"
2018-10-19 11:33:23 -07:00
"golang.org/x/sys/windows"
2017-02-01 10:26:00 -08:00
)
var (
procGetSystemTimeAsFileTime = common . Modkernel32 . NewProc ( "GetSystemTimeAsFileTime" )
2018-10-19 11:33:23 -07:00
procGetTickCount32 = common . Modkernel32 . NewProc ( "GetTickCount" )
procGetTickCount64 = common . Modkernel32 . NewProc ( "GetTickCount64" )
procRtlGetVersion = common . ModNt . NewProc ( "RtlGetVersion" )
2017-02-01 10:26:00 -08:00
)
2018-10-19 11:33:23 -07:00
// https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/ns-wdm-_osversioninfoexw
type osVersionInfoExW struct {
dwOSVersionInfoSize uint32
dwMajorVersion uint32
dwMinorVersion uint32
dwBuildNumber uint32
dwPlatformId uint32
szCSDVersion [ 128 ] uint16
wServicePackMajor uint16
wServicePackMinor uint16
wSuiteMask uint16
wProductType uint8
wReserved uint8
}
type msAcpi_ThermalZoneTemperature struct {
Active bool
CriticalTripPoint uint32
CurrentTemperature uint32
InstanceName string
2017-02-01 10:26:00 -08:00
}
func Info ( ) ( * InfoStat , error ) {
2018-10-19 11:33:23 -07:00
return InfoWithContext ( context . Background ( ) )
}
func InfoWithContext ( ctx context . Context ) ( * InfoStat , error ) {
2017-02-01 10:26:00 -08:00
ret := & InfoStat {
OS : runtime . GOOS ,
}
2017-02-02 16:11:54 -08:00
{
hostname , err := os . Hostname ( )
if err == nil {
ret . Hostname = hostname
}
2017-02-01 10:26:00 -08:00
}
2017-02-02 16:11:54 -08:00
{
2018-10-19 11:33:23 -07:00
platform , family , version , err := PlatformInformationWithContext ( ctx )
2017-02-02 16:11:54 -08:00
if err == nil {
ret . Platform = platform
ret . PlatformFamily = family
ret . PlatformVersion = version
} else {
return ret , err
}
}
{
boot , err := BootTime ( )
if err == nil {
ret . BootTime = boot
ret . Uptime , _ = Uptime ( )
}
2017-02-01 10:26:00 -08:00
}
2017-02-02 16:11:54 -08:00
{
hostID , err := getMachineGuid ( )
if err == nil {
2017-02-06 16:27:56 -08:00
ret . HostID = strings . ToLower ( hostID )
2017-02-02 16:11:54 -08:00
}
2017-02-01 10:26:00 -08:00
}
2017-02-02 16:11:54 -08:00
{
procs , err := process . Pids ( )
if err == nil {
ret . Procs = uint64 ( len ( procs ) )
}
2017-02-01 10:26:00 -08:00
}
return ret , nil
}
2017-02-02 16:11:54 -08:00
func getMachineGuid ( ) ( string , error ) {
2018-10-19 11:33:23 -07:00
// 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
// for rationale of using windows.RegOpenKeyEx/RegQueryValueEx instead of registry.OpenKey/GetStringValue
var h windows . Handle
err := windows . RegOpenKeyEx ( windows . HKEY_LOCAL_MACHINE , windows . StringToUTF16Ptr ( ` SOFTWARE\Microsoft\Cryptography ` ) , 0 , windows . KEY_READ | windows . KEY_WOW64_64KEY , & h )
2017-02-02 16:11:54 -08:00
if err != nil {
return "" , err
}
2018-10-19 11:33:23 -07:00
defer windows . RegCloseKey ( h )
2017-02-02 16:11:54 -08:00
const windowsRegBufLen = 74 // len(`{`) + len(`abcdefgh-1234-456789012-123345456671` * 2) + len(`}`) // 2 == bytes/UTF16
const uuidLen = 36
var regBuf [ windowsRegBufLen ] uint16
bufLen := uint32 ( windowsRegBufLen )
var valType uint32
2018-10-19 11:33:23 -07:00
err = windows . RegQueryValueEx ( h , windows . StringToUTF16Ptr ( ` MachineGuid ` ) , nil , & valType , ( * byte ) ( unsafe . Pointer ( & regBuf [ 0 ] ) ) , & bufLen )
2017-02-02 16:11:54 -08:00
if err != nil {
return "" , err
}
2018-10-19 11:33:23 -07:00
hostID := windows . UTF16ToString ( regBuf [ : ] )
2017-02-02 16:11:54 -08:00
hostIDLen := len ( hostID )
if hostIDLen != uuidLen {
return "" , fmt . Errorf ( "HostID incorrect: %q\n" , hostID )
}
return hostID , nil
}
2018-10-19 11:33:23 -07:00
func Uptime ( ) ( uint64 , error ) {
return UptimeWithContext ( context . Background ( ) )
2017-02-01 10:26:00 -08:00
}
2018-10-19 11:33:23 -07:00
func UptimeWithContext ( ctx context . Context ) ( uint64 , error ) {
procGetTickCount := procGetTickCount64
err := procGetTickCount64 . Find ( )
if err != nil {
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
2017-02-01 10:26:00 -08:00
}
2018-10-19 11:33:23 -07:00
r1 , _ , lastErr := syscall . Syscall ( procGetTickCount . Addr ( ) , 0 , 0 , 0 , 0 )
if lastErr != 0 {
return 0 , lastErr
}
return uint64 ( ( time . Duration ( r1 ) * time . Millisecond ) . Seconds ( ) ) , nil
2017-02-01 10:26:00 -08:00
}
2018-10-19 11:33:23 -07:00
func bootTimeFromUptime ( up uint64 ) uint64 {
2017-02-01 10:26:00 -08:00
return uint64 ( time . Now ( ) . Unix ( ) ) - up
}
2018-10-19 11:33:23 -07:00
// cachedBootTime must be accessed via atomic.Load/StoreUint64
var cachedBootTime uint64
2017-02-01 10:26:00 -08:00
func BootTime ( ) ( uint64 , error ) {
2018-10-19 11:33:23 -07:00
return BootTimeWithContext ( context . Background ( ) )
}
func BootTimeWithContext ( ctx context . Context ) ( uint64 , error ) {
t := atomic . LoadUint64 ( & cachedBootTime )
if t != 0 {
return t , nil
2017-02-01 10:26:00 -08:00
}
up , err := Uptime ( )
if err != nil {
return 0 , err
}
2018-10-19 11:33:23 -07:00
t = bootTimeFromUptime ( up )
atomic . StoreUint64 ( & cachedBootTime , t )
return t , nil
2017-02-01 10:26:00 -08:00
}
func PlatformInformation ( ) ( platform string , family string , version string , err error ) {
2018-10-19 11:33:23 -07:00
return PlatformInformationWithContext ( context . Background ( ) )
}
func PlatformInformationWithContext ( ctx context . Context ) ( platform string , family string , version string , err error ) {
// GetVersionEx lies on Windows 8.1 and returns as Windows 8 if we don't declare compatibility in manifest
// RtlGetVersion bypasses this lying layer and returns the true Windows version
// https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/nf-wdm-rtlgetversion
// https://docs.microsoft.com/en-us/windows-hardware/drivers/ddi/content/wdm/ns-wdm-_osversioninfoexw
var osInfo osVersionInfoExW
osInfo . dwOSVersionInfoSize = uint32 ( unsafe . Sizeof ( osInfo ) )
ret , _ , err := procRtlGetVersion . Call ( uintptr ( unsafe . Pointer ( & osInfo ) ) )
if ret != 0 {
return
2017-02-01 10:26:00 -08:00
}
// Platform
2018-10-19 11:33:23 -07:00
var h windows . Handle // like getMachineGuid(), we query the registry using the raw windows.RegOpenKeyEx/RegQueryValueEx
err = windows . RegOpenKeyEx ( windows . HKEY_LOCAL_MACHINE , windows . StringToUTF16Ptr ( ` SOFTWARE\Microsoft\Windows NT\CurrentVersion ` ) , 0 , windows . KEY_READ | windows . KEY_WOW64_64KEY , & h )
if err != nil {
return
}
defer windows . RegCloseKey ( h )
var bufLen uint32
var valType uint32
err = windows . RegQueryValueEx ( h , windows . StringToUTF16Ptr ( ` ProductName ` ) , nil , & valType , nil , & bufLen )
if err != nil {
return
}
regBuf := make ( [ ] uint16 , bufLen / 2 + 1 )
err = windows . RegQueryValueEx ( h , windows . StringToUTF16Ptr ( ` ProductName ` ) , nil , & valType , ( * byte ) ( unsafe . Pointer ( & regBuf [ 0 ] ) ) , & bufLen )
if err != nil {
return
}
platform = windows . UTF16ToString ( regBuf [ : ] )
if ! strings . HasPrefix ( platform , "Microsoft" ) {
platform = "Microsoft " + platform
}
err = windows . RegQueryValueEx ( h , windows . StringToUTF16Ptr ( ` CSDVersion ` ) , nil , & valType , nil , & bufLen ) // append Service Pack number, only on success
if err == nil { // don't return an error if only the Service Pack retrieval fails
regBuf = make ( [ ] uint16 , bufLen / 2 + 1 )
err = windows . RegQueryValueEx ( h , windows . StringToUTF16Ptr ( ` CSDVersion ` ) , nil , & valType , ( * byte ) ( unsafe . Pointer ( & regBuf [ 0 ] ) ) , & bufLen )
if err == nil {
platform += " " + windows . UTF16ToString ( regBuf [ : ] )
}
}
2017-02-01 10:26:00 -08:00
// PlatformFamily
2018-10-19 11:33:23 -07:00
switch osInfo . wProductType {
2017-02-01 10:26:00 -08:00
case 1 :
family = "Standalone Workstation"
case 2 :
family = "Server (Domain Controller)"
case 3 :
family = "Server"
}
// Platform Version
2018-10-19 11:33:23 -07:00
version = fmt . Sprintf ( "%d.%d.%d Build %d" , osInfo . dwMajorVersion , osInfo . dwMinorVersion , osInfo . dwBuildNumber , osInfo . dwBuildNumber )
2017-02-01 10:26:00 -08:00
2018-10-19 11:33:23 -07:00
return platform , family , version , nil
2017-02-01 10:26:00 -08:00
}
func Users ( ) ( [ ] UserStat , error ) {
2018-10-19 11:33:23 -07:00
return UsersWithContext ( context . Background ( ) )
}
func UsersWithContext ( ctx context . Context ) ( [ ] UserStat , error ) {
2017-02-01 10:26:00 -08:00
var ret [ ] UserStat
return ret , nil
}
2018-10-19 11:33:23 -07:00
func SensorsTemperatures ( ) ( [ ] TemperatureStat , error ) {
return SensorsTemperaturesWithContext ( context . Background ( ) )
}
func SensorsTemperaturesWithContext ( ctx context . Context ) ( [ ] TemperatureStat , error ) {
var ret [ ] TemperatureStat
var dst [ ] msAcpi_ThermalZoneTemperature
q := wmi . CreateQuery ( & dst , "" )
if err := common . WMIQueryWithContext ( ctx , q , & dst , nil , "root/wmi" ) ; err != nil {
return ret , err
}
for _ , v := range dst {
ts := TemperatureStat {
SensorKey : v . InstanceName ,
Temperature : kelvinToCelsius ( v . CurrentTemperature , 2 ) ,
}
ret = append ( ret , ts )
}
return ret , nil
}
func kelvinToCelsius ( temp uint32 , n int ) float64 {
// wmi return temperature Kelvin * 10, so need to divide the result by 10,
// and then minus 273.15 to get °Celsius.
t := float64 ( temp / 10 ) - 273.15
n10 := math . Pow10 ( n )
return math . Trunc ( ( t + 0.5 / n10 ) * n10 ) / n10
}
func Virtualization ( ) ( string , string , error ) {
return VirtualizationWithContext ( context . Background ( ) )
}
func VirtualizationWithContext ( ctx context . Context ) ( string , string , error ) {
return "" , "" , common . ErrNotImplementedError
}
func KernelVersion ( ) ( string , error ) {
return KernelVersionWithContext ( context . Background ( ) )
}
func KernelVersionWithContext ( ctx context . Context ) ( string , error ) {
_ , _ , version , err := PlatformInformation ( )
return version , err
}