//go:build windows package diskdrive import ( "errors" "fmt" "log/slog" "strings" "github.com/alecthomas/kingpin/v2" "github.com/prometheus-community/windows_exporter/internal/mi" "github.com/prometheus-community/windows_exporter/internal/types" "github.com/prometheus/client_golang/prometheus" ) const Name = "diskdrive" type Config struct{} var ConfigDefaults = Config{} // A Collector is a Prometheus Collector for a few WMI metrics in Win32_DiskDrive. type Collector struct { config Config miSession *mi.Session miQuery mi.Query availability *prometheus.Desc diskInfo *prometheus.Desc partitions *prometheus.Desc size *prometheus.Desc status *prometheus.Desc } func New(config *Config) *Collector { if config == nil { config = &ConfigDefaults } c := &Collector{ config: *config, } return c } func NewWithFlags(_ *kingpin.Application) *Collector { return &Collector{} } func (c *Collector) GetName() string { return Name } func (c *Collector) GetPerfCounter(_ *slog.Logger) ([]string, error) { return []string{}, nil } func (c *Collector) Close(_ *slog.Logger) error { return nil } func (c *Collector) Build(_ *slog.Logger, miSession *mi.Session) error { if miSession == nil { return errors.New("miSession is nil") } miQuery, err := mi.NewQuery("SELECT DeviceID, Model, Caption, Name, Partitions, Size, Status, Availability FROM WIN32_DiskDrive") if err != nil { return fmt.Errorf("failed to create WMI query: %w", err) } c.miQuery = miQuery c.miSession = miSession c.diskInfo = prometheus.NewDesc( prometheus.BuildFQName(types.Namespace, Name, "info"), "General drive information", []string{ "device_id", "model", "caption", "name", }, nil, ) c.status = prometheus.NewDesc( prometheus.BuildFQName(types.Namespace, Name, "status"), "Status of the drive", []string{"name", "status"}, nil, ) c.size = prometheus.NewDesc( prometheus.BuildFQName(types.Namespace, Name, "size"), "Size of the disk drive. It is calculated by multiplying the total number of cylinders, tracks in each cylinder, sectors in each track, and bytes in each sector.", []string{"name"}, nil, ) c.partitions = prometheus.NewDesc( prometheus.BuildFQName(types.Namespace, Name, "partitions"), "Number of partitions", []string{"name"}, nil, ) c.availability = prometheus.NewDesc( prometheus.BuildFQName(types.Namespace, Name, "availability"), "Availability Status", []string{"name", "availability"}, nil, ) return nil } type win32_DiskDrive struct { DeviceID string `mi:"DeviceID"` Model string `mi:"Model"` Size uint64 `mi:"Size"` Name string `mi:"Name"` Caption string `mi:"Caption"` Partitions uint32 `mi:"Partitions"` Status string `mi:"Status"` Availability uint16 `mi:"Availability"` } var ( allDiskStatus = []string{ "OK", "Error", "Degraded", "Unknown", "Pred fail", "Starting", "Stopping", "Service", "Stressed", "Nonrecover", "No Contact", "Lost Comm", } availMap = map[int]string{ 1: "Other", 2: "Unknown", 3: "Running / Full Power", 4: "Warning", 5: "In Test", 6: "Not Applicable", 7: "Power Off", 8: "Off line", 9: "Off Duty", 10: "Degraded", 11: "Not Installed", 12: "Install Error", 13: "Power Save - Unknown", 14: "Power Save - Low Power Mode", 15: "Power Save - Standby", 16: "Power Cycle", 17: "Power Save - Warning", 18: "Paused", 19: "Not Ready", 20: "Not Configured", 21: "Quiesced", } ) // Collect sends the metric values for each metric to the provided prometheus Metric channel. func (c *Collector) Collect(_ *types.ScrapeContext, logger *slog.Logger, ch chan<- prometheus.Metric) error { logger = logger.With(slog.String("collector", Name)) if err := c.collect(ch); err != nil { logger.Error("failed collecting disk_drive_info metrics", slog.Any("err", err), ) return err } return nil } func (c *Collector) collect(ch chan<- prometheus.Metric) error { var dst []win32_DiskDrive if err := c.miSession.Query(&dst, mi.NamespaceRootCIMv2, c.miQuery); err != nil { return fmt.Errorf("WMI query failed: %w", err) } if len(dst) == 0 { return errors.New("WMI query returned empty result set") } for _, disk := range dst { ch <- prometheus.MustNewConstMetric( c.diskInfo, prometheus.GaugeValue, 1.0, strings.Trim(disk.DeviceID, "\\.\\"), //nolint:staticcheck strings.TrimRight(disk.Model, " "), strings.TrimRight(disk.Caption, " "), strings.TrimRight(disk.Name, "\\.\\"), //nolint:staticcheck ) for _, status := range allDiskStatus { isCurrentState := 0.0 if status == disk.Status { isCurrentState = 1.0 } ch <- prometheus.MustNewConstMetric( c.status, prometheus.GaugeValue, isCurrentState, strings.Trim(disk.Name, "\\.\\"), //nolint:staticcheck status, ) } ch <- prometheus.MustNewConstMetric( c.size, prometheus.GaugeValue, float64(disk.Size), strings.Trim(disk.Name, "\\.\\"), //nolint:staticcheck ) ch <- prometheus.MustNewConstMetric( c.partitions, prometheus.GaugeValue, float64(disk.Partitions), strings.Trim(disk.Name, "\\.\\"), //nolint:staticcheck ) for availNum, val := range availMap { isCurrentState := 0.0 if availNum == int(disk.Availability) { isCurrentState = 1.0 } ch <- prometheus.MustNewConstMetric( c.availability, prometheus.GaugeValue, isCurrentState, strings.Trim(disk.Name, "\\.\\"), //nolint:staticcheck val, ) } } return nil }