mirror of
https://github.com/status-im/consul.git
synced 2025-01-14 07:44:50 +00:00
8c684db488
* agent/debug: add package for debugging, host info * api: add v1/agent/host endpoint * agent: add v1/agent/host endpoint * command/debug: implementation of static capture * command/debug: tests and only configured targets * agent/debug: add basic test for host metrics * command/debug: add methods for dynamic data capture * api: add debug/pprof endpoints * command/debug: add pprof * command/debug: timing, wg, logs to disk * vendor: add gopsutil/disk * command/debug: add a usage section * website: add docs for consul debug * agent/host: require operator:read * api/host: improve docs and no retry timing * command/debug: fail on extra arguments * command/debug: fixup file permissions to 0644 * command/debug: remove server flags * command/debug: improve clarity of usage section * api/debug: add Trace for profiling, fix profile * command/debug: capture profile and trace at the same time * command/debug: add index document * command/debug: use "clusters" in place of members * command/debug: remove address in output * command/debug: improve comment on metrics sleep * command/debug: clarify usage * agent: always register pprof handlers and protect This will allow us to avoid a restart of a target agent for profiling by always registering the pprof handlers. Given this is a potentially sensitive path, it is protected with an operator:read ACL and enable debug being set to true on the target agent. enable_debug still requires a restart. If ACLs are disabled, enable_debug is sufficient. * command/debug: use trace.out instead of .prof More in line with golang docs. * agent: fix comment wording * agent: wrap table driven tests in t.run()
156 lines
4.4 KiB
Go
156 lines
4.4 KiB
Go
// +build windows
|
|
|
|
package disk
|
|
|
|
import (
|
|
"bytes"
|
|
"syscall"
|
|
"unsafe"
|
|
|
|
"github.com/StackExchange/wmi"
|
|
|
|
"github.com/shirou/gopsutil/internal/common"
|
|
)
|
|
|
|
var (
|
|
procGetDiskFreeSpaceExW = common.Modkernel32.NewProc("GetDiskFreeSpaceExW")
|
|
procGetLogicalDriveStringsW = common.Modkernel32.NewProc("GetLogicalDriveStringsW")
|
|
procGetDriveType = common.Modkernel32.NewProc("GetDriveTypeW")
|
|
provGetVolumeInformation = common.Modkernel32.NewProc("GetVolumeInformationW")
|
|
)
|
|
|
|
var (
|
|
FileFileCompression = int64(16) // 0x00000010
|
|
FileReadOnlyVolume = int64(524288) // 0x00080000
|
|
)
|
|
|
|
type Win32_PerfFormattedData struct {
|
|
Name string
|
|
AvgDiskBytesPerRead uint64
|
|
AvgDiskBytesPerWrite uint64
|
|
AvgDiskReadQueueLength uint64
|
|
AvgDiskWriteQueueLength uint64
|
|
AvgDisksecPerRead uint64
|
|
AvgDisksecPerWrite uint64
|
|
}
|
|
|
|
const WaitMSec = 500
|
|
|
|
func Usage(path string) (*UsageStat, error) {
|
|
ret := &UsageStat{}
|
|
|
|
lpFreeBytesAvailable := int64(0)
|
|
lpTotalNumberOfBytes := int64(0)
|
|
lpTotalNumberOfFreeBytes := int64(0)
|
|
diskret, _, err := procGetDiskFreeSpaceExW.Call(
|
|
uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(path))),
|
|
uintptr(unsafe.Pointer(&lpFreeBytesAvailable)),
|
|
uintptr(unsafe.Pointer(&lpTotalNumberOfBytes)),
|
|
uintptr(unsafe.Pointer(&lpTotalNumberOfFreeBytes)))
|
|
if diskret == 0 {
|
|
return nil, err
|
|
}
|
|
ret = &UsageStat{
|
|
Path: path,
|
|
Total: uint64(lpTotalNumberOfBytes),
|
|
Free: uint64(lpTotalNumberOfFreeBytes),
|
|
Used: uint64(lpTotalNumberOfBytes) - uint64(lpTotalNumberOfFreeBytes),
|
|
UsedPercent: (float64(lpTotalNumberOfBytes) - float64(lpTotalNumberOfFreeBytes)) / float64(lpTotalNumberOfBytes) * 100,
|
|
// InodesTotal: 0,
|
|
// InodesFree: 0,
|
|
// InodesUsed: 0,
|
|
// InodesUsedPercent: 0,
|
|
}
|
|
return ret, nil
|
|
}
|
|
|
|
func Partitions(all bool) ([]PartitionStat, error) {
|
|
var ret []PartitionStat
|
|
lpBuffer := make([]byte, 254)
|
|
diskret, _, err := procGetLogicalDriveStringsW.Call(
|
|
uintptr(len(lpBuffer)),
|
|
uintptr(unsafe.Pointer(&lpBuffer[0])))
|
|
if diskret == 0 {
|
|
return ret, err
|
|
}
|
|
for _, v := range lpBuffer {
|
|
if v >= 65 && v <= 90 {
|
|
path := string(v) + ":"
|
|
if path == "A:" || path == "B:" { // skip floppy drives
|
|
continue
|
|
}
|
|
typepath, _ := syscall.UTF16PtrFromString(path)
|
|
typeret, _, _ := procGetDriveType.Call(uintptr(unsafe.Pointer(typepath)))
|
|
if typeret == 0 {
|
|
return ret, syscall.GetLastError()
|
|
}
|
|
// 2: DRIVE_REMOVABLE 3: DRIVE_FIXED 5: DRIVE_CDROM
|
|
|
|
if typeret == 2 || typeret == 3 || typeret == 5 {
|
|
lpVolumeNameBuffer := make([]byte, 256)
|
|
lpVolumeSerialNumber := int64(0)
|
|
lpMaximumComponentLength := int64(0)
|
|
lpFileSystemFlags := int64(0)
|
|
lpFileSystemNameBuffer := make([]byte, 256)
|
|
volpath, _ := syscall.UTF16PtrFromString(string(v) + ":/")
|
|
driveret, _, err := provGetVolumeInformation.Call(
|
|
uintptr(unsafe.Pointer(volpath)),
|
|
uintptr(unsafe.Pointer(&lpVolumeNameBuffer[0])),
|
|
uintptr(len(lpVolumeNameBuffer)),
|
|
uintptr(unsafe.Pointer(&lpVolumeSerialNumber)),
|
|
uintptr(unsafe.Pointer(&lpMaximumComponentLength)),
|
|
uintptr(unsafe.Pointer(&lpFileSystemFlags)),
|
|
uintptr(unsafe.Pointer(&lpFileSystemNameBuffer[0])),
|
|
uintptr(len(lpFileSystemNameBuffer)))
|
|
if driveret == 0 {
|
|
if typeret == 5 {
|
|
continue //device is not ready will happen if there is no disk in the drive
|
|
}
|
|
return ret, err
|
|
}
|
|
opts := "rw"
|
|
if lpFileSystemFlags&FileReadOnlyVolume != 0 {
|
|
opts = "ro"
|
|
}
|
|
if lpFileSystemFlags&FileFileCompression != 0 {
|
|
opts += ".compress"
|
|
}
|
|
|
|
d := PartitionStat{
|
|
Mountpoint: path,
|
|
Device: path,
|
|
Fstype: string(bytes.Replace(lpFileSystemNameBuffer, []byte("\x00"), []byte(""), -1)),
|
|
Opts: opts,
|
|
}
|
|
ret = append(ret, d)
|
|
}
|
|
}
|
|
}
|
|
return ret, nil
|
|
}
|
|
|
|
func IOCounters() (map[string]IOCountersStat, error) {
|
|
ret := make(map[string]IOCountersStat, 0)
|
|
var dst []Win32_PerfFormattedData
|
|
|
|
err := wmi.Query("SELECT * FROM Win32_PerfFormattedData_PerfDisk_LogicalDisk ", &dst)
|
|
if err != nil {
|
|
return ret, err
|
|
}
|
|
for _, d := range dst {
|
|
if len(d.Name) > 3 { // not get _Total or Harddrive
|
|
continue
|
|
}
|
|
ret[d.Name] = IOCountersStat{
|
|
Name: d.Name,
|
|
ReadCount: uint64(d.AvgDiskReadQueueLength),
|
|
WriteCount: d.AvgDiskWriteQueueLength,
|
|
ReadBytes: uint64(d.AvgDiskBytesPerRead),
|
|
WriteBytes: uint64(d.AvgDiskBytesPerWrite),
|
|
ReadTime: d.AvgDisksecPerRead,
|
|
WriteTime: d.AvgDisksecPerWrite,
|
|
}
|
|
}
|
|
return ret, nil
|
|
}
|