mirror of
https://github.com/prometheus-community/windows_exporter
synced 2025-01-16 04:01:04 +00:00
07fff6afc2
* Added physical_disk collector. Signed-off-by: Rob Scheepens <rob.scheepens@nutanix.com> Signed-off-by: Brantley West <brantley@nutanix.com> * exporter.go: Added physical_disk to defaultCollectors Signed-off-by: Rob Scheepens <rob.scheepens@nutanix.com> Signed-off-by: Brantley West <brantley@nutanix.com> * Fix test cases for physicaldisk metrics Signed-off-by: Rob Scheepens <rob.scheepens@nutanix.com> Signed-off-by: Brantley West <brantley@nutanix.com> * physical_disk.go: cleanup gofmt error Signed-off-by: Brantley West <brantley@nutanix.com> * physical_disk.go: populate physical disk 'number' label Signed-off-by: Brantley West <brantley@nutanix.com> * Added docs/collector.physical_disk.md. Signed-off-by: Rob Scheepens <rob.scheepens@nutanix.com> * physical_disk.go: change 'number' label to 'disk' to match node_exporter label Signed-off-by: Brantley West <brantley@nutanix.com> * physical_disk.go: adopt github.com/go-kit/log Signed-off-by: Brantley West <brantley@nutanix.com> * physical_disk.go: adopt include/exclude disk list Signed-off-by: Brantley West <brantley@nutanix.com> * fix: Add init config for physical_disk collector Signed-off-by: Ben Reedy <breed808@breed808.com> * chore: gofmt physical_disk collector Signed-off-by: Ben Reedy <breed808@breed808.com> --------- Signed-off-by: Rob Scheepens <rob.scheepens@nutanix.com> Signed-off-by: Brantley West <brantley@nutanix.com> Signed-off-by: Ben Reedy <breed808@breed808.com> Co-authored-by: Brantley West <brantley@nutanix.com> Co-authored-by: Ben Reedy <breed808@breed808.com>
299 lines
8.5 KiB
Go
299 lines
8.5 KiB
Go
//go:build windows
|
|
// +build windows
|
|
|
|
package collector
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/alecthomas/kingpin/v2"
|
|
"github.com/go-kit/log"
|
|
"github.com/go-kit/log/level"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
)
|
|
|
|
const (
|
|
FlagPhysicalDiskExclude = "collector.physical_disk.disk-exclude"
|
|
FlagPhysicalDiskInclude = "collector.physical_disk.disk-include"
|
|
)
|
|
|
|
var (
|
|
diskInclude *string
|
|
diskExclude *string
|
|
|
|
diskIncludeSet bool
|
|
diskExcludeSet bool
|
|
)
|
|
|
|
// A PhysicalDiskCollector is a Prometheus collector for perflib PhysicalDisk metrics
|
|
type PhysicalDiskCollector struct {
|
|
logger log.Logger
|
|
|
|
RequestsQueued *prometheus.Desc
|
|
ReadBytesTotal *prometheus.Desc
|
|
ReadsTotal *prometheus.Desc
|
|
WriteBytesTotal *prometheus.Desc
|
|
WritesTotal *prometheus.Desc
|
|
ReadTime *prometheus.Desc
|
|
WriteTime *prometheus.Desc
|
|
IdleTime *prometheus.Desc
|
|
SplitIOs *prometheus.Desc
|
|
ReadLatency *prometheus.Desc
|
|
WriteLatency *prometheus.Desc
|
|
ReadWriteLatency *prometheus.Desc
|
|
|
|
diskIncludePattern *regexp.Regexp
|
|
diskExcludePattern *regexp.Regexp
|
|
}
|
|
|
|
// newPhysicalDiskCollectorFlags ...
|
|
func newPhysicalDiskCollectorFlags(app *kingpin.Application) {
|
|
diskInclude = app.Flag(
|
|
FlagPhysicalDiskInclude,
|
|
"Regexp of disks to include. Disk number must both match include and not match exclude to be included.",
|
|
).Default(".+").PreAction(func(c *kingpin.ParseContext) error {
|
|
diskIncludeSet = true
|
|
return nil
|
|
}).String()
|
|
|
|
diskExclude = app.Flag(
|
|
FlagPhysicalDiskExclude,
|
|
"Regexp of disks to exclude. Disk number must both match include and not match exclude to be included.",
|
|
).Default("").PreAction(func(c *kingpin.ParseContext) error {
|
|
diskExcludeSet = true
|
|
return nil
|
|
}).String()
|
|
}
|
|
|
|
// NewPhysicalDiskCollector ...
|
|
func NewPhysicalDiskCollector(logger log.Logger) (Collector, error) {
|
|
const subsystem = "physical_disk"
|
|
logger = log.With(logger, "collector", subsystem)
|
|
|
|
return &PhysicalDiskCollector{
|
|
logger: logger,
|
|
|
|
RequestsQueued: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "requests_queued"),
|
|
"The number of requests queued to the disk (PhysicalDisk.CurrentDiskQueueLength)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
ReadBytesTotal: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "read_bytes_total"),
|
|
"The number of bytes transferred from the disk during read operations (PhysicalDisk.DiskReadBytesPerSec)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
ReadsTotal: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "reads_total"),
|
|
"The number of read operations on the disk (PhysicalDisk.DiskReadsPerSec)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
WriteBytesTotal: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "write_bytes_total"),
|
|
"The number of bytes transferred to the disk during write operations (PhysicalDisk.DiskWriteBytesPerSec)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
WritesTotal: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "writes_total"),
|
|
"The number of write operations on the disk (PhysicalDisk.DiskWritesPerSec)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
ReadTime: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "read_seconds_total"),
|
|
"Seconds that the disk was busy servicing read requests (PhysicalDisk.PercentDiskReadTime)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
WriteTime: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "write_seconds_total"),
|
|
"Seconds that the disk was busy servicing write requests (PhysicalDisk.PercentDiskWriteTime)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
IdleTime: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "idle_seconds_total"),
|
|
"Seconds that the disk was idle (PhysicalDisk.PercentIdleTime)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
SplitIOs: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "split_ios_total"),
|
|
"The number of I/Os to the disk were split into multiple I/Os (PhysicalDisk.SplitIOPerSec)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
ReadLatency: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "read_latency_seconds_total"),
|
|
"Shows the average time, in seconds, of a read operation from the disk (PhysicalDisk.AvgDiskSecPerRead)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
WriteLatency: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "write_latency_seconds_total"),
|
|
"Shows the average time, in seconds, of a write operation to the disk (PhysicalDisk.AvgDiskSecPerWrite)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
ReadWriteLatency: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "read_write_latency_seconds_total"),
|
|
"Shows the time, in seconds, of the average disk transfer (PhysicalDisk.AvgDiskSecPerTransfer)",
|
|
[]string{"disk"},
|
|
nil,
|
|
),
|
|
|
|
diskIncludePattern: regexp.MustCompile(fmt.Sprintf("^(?:%s)$", *diskInclude)),
|
|
diskExcludePattern: regexp.MustCompile(fmt.Sprintf("^(?:%s)$", *diskExclude)),
|
|
}, nil
|
|
}
|
|
|
|
// Collect sends the metric values for each metric
|
|
// to the provided prometheus Metric channel.
|
|
func (c *PhysicalDiskCollector) Collect(ctx *ScrapeContext, ch chan<- prometheus.Metric) error {
|
|
if desc, err := c.collect(ctx, ch); err != nil {
|
|
_ = level.Error(c.logger).Log("failed collecting physical_disk metrics", "desc", desc, "err", err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Win32_PerfRawData_PerfDisk_PhysicalDisk docs:
|
|
// - https://docs.microsoft.com/en-us/previous-versions/aa394308(v=vs.85) - Win32_PerfRawData_PerfDisk_PhysicalDisk class
|
|
type PhysicalDisk struct {
|
|
Name string
|
|
CurrentDiskQueueLength float64 `perflib:"Current Disk Queue Length"`
|
|
DiskReadBytesPerSec float64 `perflib:"Disk Read Bytes/sec"`
|
|
DiskReadsPerSec float64 `perflib:"Disk Reads/sec"`
|
|
DiskWriteBytesPerSec float64 `perflib:"Disk Write Bytes/sec"`
|
|
DiskWritesPerSec float64 `perflib:"Disk Writes/sec"`
|
|
PercentDiskReadTime float64 `perflib:"% Disk Read Time"`
|
|
PercentDiskWriteTime float64 `perflib:"% Disk Write Time"`
|
|
PercentIdleTime float64 `perflib:"% Idle Time"`
|
|
SplitIOPerSec float64 `perflib:"Split IO/Sec"`
|
|
AvgDiskSecPerRead float64 `perflib:"Avg. Disk sec/Read"`
|
|
AvgDiskSecPerWrite float64 `perflib:"Avg. Disk sec/Write"`
|
|
AvgDiskSecPerTransfer float64 `perflib:"Avg. Disk sec/Transfer"`
|
|
}
|
|
|
|
func (c *PhysicalDiskCollector) collect(ctx *ScrapeContext, ch chan<- prometheus.Metric) (*prometheus.Desc, error) {
|
|
var dst []PhysicalDisk
|
|
if err := unmarshalObject(ctx.perfObjects["PhysicalDisk"], &dst, c.logger); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, disk := range dst {
|
|
if disk.Name == "_Total" ||
|
|
c.diskExcludePattern.MatchString(disk.Name) ||
|
|
!c.diskIncludePattern.MatchString(disk.Name) {
|
|
continue
|
|
}
|
|
|
|
// Parse physical disk number from disk.Name. Mountpoint information is
|
|
// sometimes included, e.g. "1 C:".
|
|
disk_number, _, _ := strings.Cut(disk.Name, " ")
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.RequestsQueued,
|
|
prometheus.GaugeValue,
|
|
disk.CurrentDiskQueueLength,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.ReadBytesTotal,
|
|
prometheus.CounterValue,
|
|
disk.DiskReadBytesPerSec,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.ReadsTotal,
|
|
prometheus.CounterValue,
|
|
disk.DiskReadsPerSec,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.WriteBytesTotal,
|
|
prometheus.CounterValue,
|
|
disk.DiskWriteBytesPerSec,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.WritesTotal,
|
|
prometheus.CounterValue,
|
|
disk.DiskWritesPerSec,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.ReadTime,
|
|
prometheus.CounterValue,
|
|
disk.PercentDiskReadTime,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.WriteTime,
|
|
prometheus.CounterValue,
|
|
disk.PercentDiskWriteTime,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.IdleTime,
|
|
prometheus.CounterValue,
|
|
disk.PercentIdleTime,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.SplitIOs,
|
|
prometheus.CounterValue,
|
|
disk.SplitIOPerSec,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.ReadLatency,
|
|
prometheus.CounterValue,
|
|
disk.AvgDiskSecPerRead*ticksToSecondsScaleFactor,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.WriteLatency,
|
|
prometheus.CounterValue,
|
|
disk.AvgDiskSecPerWrite*ticksToSecondsScaleFactor,
|
|
disk_number,
|
|
)
|
|
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.ReadWriteLatency,
|
|
prometheus.CounterValue,
|
|
disk.AvgDiskSecPerTransfer*ticksToSecondsScaleFactor,
|
|
disk_number,
|
|
)
|
|
}
|
|
|
|
return nil, nil
|
|
}
|