352 lines
11 KiB
Go
352 lines
11 KiB
Go
// +build windows
|
|
|
|
package collector
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"github.com/prometheus-community/windows_exporter/log"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
)
|
|
|
|
func init() {
|
|
registerCollector("remote_fx", NewRemoteFx, "RemoteFX Network", "RemoteFX Graphics")
|
|
}
|
|
|
|
// A RemoteFxNetworkCollector is a Prometheus collector for
|
|
// WMI Win32_PerfRawData_Counters_RemoteFXNetwork & Win32_PerfRawData_Counters_RemoteFXGraphics metrics
|
|
// https://wutils.com/wmi/root/cimv2/win32_perfrawdata_counters_remotefxnetwork/
|
|
// https://wutils.com/wmi/root/cimv2/win32_perfrawdata_counters_remotefxgraphics/
|
|
|
|
type RemoteFxCollector struct {
|
|
// net
|
|
BaseTCPRTT *prometheus.Desc
|
|
BaseUDPRTT *prometheus.Desc
|
|
CurrentTCPBandwidth *prometheus.Desc
|
|
CurrentTCPRTT *prometheus.Desc
|
|
CurrentUDPBandwidth *prometheus.Desc
|
|
CurrentUDPRTT *prometheus.Desc
|
|
TotalReceivedBytes *prometheus.Desc
|
|
TotalSentBytes *prometheus.Desc
|
|
UDPPacketsReceivedPersec *prometheus.Desc
|
|
UDPPacketsSentPersec *prometheus.Desc
|
|
|
|
//gfx
|
|
AverageEncodingTime *prometheus.Desc
|
|
FrameQuality *prometheus.Desc
|
|
FramesSkippedPerSecondInsufficientResources *prometheus.Desc
|
|
GraphicsCompressionratio *prometheus.Desc
|
|
InputFramesPerSecond *prometheus.Desc
|
|
OutputFramesPerSecond *prometheus.Desc
|
|
SourceFramesPerSecond *prometheus.Desc
|
|
}
|
|
|
|
// NewRemoteFx ...
|
|
func NewRemoteFx() (Collector, error) {
|
|
const subsystem = "remote_fx"
|
|
return &RemoteFxCollector{
|
|
// net
|
|
BaseTCPRTT: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_base_tcp_rtt_seconds"),
|
|
"Base TCP round-trip time (RTT) detected in seconds",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
BaseUDPRTT: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_base_udp_rtt_seconds"),
|
|
"Base UDP round-trip time (RTT) detected in seconds.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
CurrentTCPBandwidth: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_current_tcp_bandwidth"),
|
|
"TCP Bandwidth detected in bytes per seccond.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
CurrentTCPRTT: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_current_tcp_rtt_seconds"),
|
|
"Average TCP round-trip time (RTT) detected in seconds.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
CurrentUDPBandwidth: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_current_udp_bandwidth"),
|
|
"UDP Bandwidth detected in bytes per second.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
CurrentUDPRTT: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_current_udp_rtt_seconds"),
|
|
"Average UDP round-trip time (RTT) detected in seconds.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
TotalReceivedBytes: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_received_bytes_total"),
|
|
"(TotalReceivedBytes)",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
TotalSentBytes: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_sent_bytes_total"),
|
|
"(TotalSentBytes)",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
UDPPacketsReceivedPersec: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_udp_packets_received_total"),
|
|
"Rate in packets per second at which packets are received over UDP.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
UDPPacketsSentPersec: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "net_udp_packets_sent_total"),
|
|
"Rate in packets per second at which packets are sent over UDP.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
|
|
//gfx
|
|
AverageEncodingTime: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_average_encoding_time_seconds"),
|
|
"Average frame encoding time in seconds",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
FrameQuality: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_frame_quality"),
|
|
"Quality of the output frame expressed as a percentage of the quality of the source frame.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
FramesSkippedPerSecondInsufficientResources: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_frames_skipped_insufficient_resource_total"),
|
|
"Number of frames skipped per second due to insufficient client resources.",
|
|
[]string{"session_name", "resource"},
|
|
nil,
|
|
),
|
|
GraphicsCompressionratio: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_graphics_compression_ratio"),
|
|
"Ratio of the number of bytes encoded to the number of bytes input.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
InputFramesPerSecond: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_input_frames_total"),
|
|
"Number of sources frames provided as input to RemoteFX graphics per second.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
OutputFramesPerSecond: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_output_frames_total"),
|
|
"Number of frames sent to the client per second.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
SourceFramesPerSecond: prometheus.NewDesc(
|
|
prometheus.BuildFQName(Namespace, subsystem, "gfx_source_frames_total"),
|
|
"Number of frames composed by the source (DWM) per second.",
|
|
[]string{"session_name"},
|
|
nil,
|
|
),
|
|
}, nil
|
|
}
|
|
|
|
// Collect sends the metric values for each metric
|
|
// to the provided prometheus Metric channel.
|
|
func (c *RemoteFxCollector) Collect(ctx *ScrapeContext, ch chan<- prometheus.Metric) error {
|
|
if desc, err := c.collectRemoteFXNetworkCount(ctx, ch); err != nil {
|
|
log.Error("failed collecting terminal services session count metrics:", desc, err)
|
|
return err
|
|
}
|
|
if desc, err := c.collectRemoteFXGraphicsCounters(ctx, ch); err != nil {
|
|
log.Error("failed collecting terminal services session count metrics:", desc, err)
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type perflibRemoteFxNetwork struct {
|
|
Name string
|
|
BaseTCPRTT float64 `perflib:"Base TCP RTT"`
|
|
BaseUDPRTT float64 `perflib:"Base UDP RTT"`
|
|
CurrentTCPBandwidth float64 `perflib:"Current TCP Bandwidth"`
|
|
CurrentTCPRTT float64 `perflib:"Current TCP RTT"`
|
|
CurrentUDPBandwidth float64 `perflib:"Current UDP Bandwidth"`
|
|
CurrentUDPRTT float64 `perflib:"Current UDP RTT"`
|
|
TotalReceivedBytes float64 `perflib:"Total Received Bytes"`
|
|
TotalSentBytes float64 `perflib:"Total Sent Bytes"`
|
|
UDPPacketsReceivedPersec float64 `perflib:"UDP Packets Received/sec"`
|
|
UDPPacketsSentPersec float64 `perflib:"UDP Packets Sent/sec"`
|
|
}
|
|
|
|
func (c *RemoteFxCollector) collectRemoteFXNetworkCount(ctx *ScrapeContext, ch chan<- prometheus.Metric) (*prometheus.Desc, error) {
|
|
dst := make([]perflibRemoteFxNetwork, 0)
|
|
err := unmarshalObject(ctx.perfObjects["RemoteFX Network"], &dst)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, d := range dst {
|
|
// only connect metrics for remote named sessions
|
|
n := strings.ToLower(d.Name)
|
|
if n == "" || n == "services" || n == "console" {
|
|
continue
|
|
}
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.BaseTCPRTT,
|
|
prometheus.GaugeValue,
|
|
milliSecToSec(d.BaseTCPRTT),
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.BaseUDPRTT,
|
|
prometheus.GaugeValue,
|
|
milliSecToSec(d.BaseUDPRTT),
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.CurrentTCPBandwidth,
|
|
prometheus.GaugeValue,
|
|
(d.CurrentTCPBandwidth*1000)/8,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.CurrentTCPRTT,
|
|
prometheus.GaugeValue,
|
|
milliSecToSec(d.CurrentTCPRTT),
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.CurrentUDPBandwidth,
|
|
prometheus.GaugeValue,
|
|
(d.CurrentUDPBandwidth*1000)/8,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.CurrentUDPRTT,
|
|
prometheus.GaugeValue,
|
|
milliSecToSec(d.CurrentUDPRTT),
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.TotalReceivedBytes,
|
|
prometheus.CounterValue,
|
|
d.TotalReceivedBytes,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.TotalSentBytes,
|
|
prometheus.CounterValue,
|
|
d.TotalSentBytes,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.UDPPacketsReceivedPersec,
|
|
prometheus.CounterValue,
|
|
d.UDPPacketsReceivedPersec,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.UDPPacketsSentPersec,
|
|
prometheus.CounterValue,
|
|
d.UDPPacketsSentPersec,
|
|
d.Name,
|
|
)
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
type perflibRemoteFxGraphics struct {
|
|
Name string
|
|
AverageEncodingTime float64 `perflib:"Average Encoding Time"`
|
|
FrameQuality float64 `perflib:"Frame Quality"`
|
|
FramesSkippedPerSecondInsufficientClientResources float64 `perflib:"Frames Skipped/Second - Insufficient Server Resources"`
|
|
FramesSkippedPerSecondInsufficientNetworkResources float64 `perflib:"Frames Skipped/Second - Insufficient Network Resources"`
|
|
FramesSkippedPerSecondInsufficientServerResources float64 `perflib:"Frames Skipped/Second - Insufficient Client Resources"`
|
|
GraphicsCompressionratio float64 `perflib:"Graphics Compression ratio"`
|
|
InputFramesPerSecond float64 `perflib:"Input Frames/Second"`
|
|
OutputFramesPerSecond float64 `perflib:"Output Frames/Second"`
|
|
SourceFramesPerSecond float64 `perflib:"Source Frames/Second"`
|
|
}
|
|
|
|
func (c *RemoteFxCollector) collectRemoteFXGraphicsCounters(ctx *ScrapeContext, ch chan<- prometheus.Metric) (*prometheus.Desc, error) {
|
|
dst := make([]perflibRemoteFxGraphics, 0)
|
|
err := unmarshalObject(ctx.perfObjects["RemoteFX Graphics"], &dst)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, d := range dst {
|
|
// only connect metrics for remote named sessions
|
|
n := strings.ToLower(d.Name)
|
|
if n == "" || n == "services" || n == "console" {
|
|
continue
|
|
}
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.AverageEncodingTime,
|
|
prometheus.GaugeValue,
|
|
milliSecToSec(d.AverageEncodingTime),
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.FrameQuality,
|
|
prometheus.GaugeValue,
|
|
d.FrameQuality,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.FramesSkippedPerSecondInsufficientResources,
|
|
prometheus.CounterValue,
|
|
d.FramesSkippedPerSecondInsufficientClientResources,
|
|
d.Name,
|
|
"client",
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.FramesSkippedPerSecondInsufficientResources,
|
|
prometheus.CounterValue,
|
|
d.FramesSkippedPerSecondInsufficientNetworkResources,
|
|
d.Name,
|
|
"network",
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.FramesSkippedPerSecondInsufficientResources,
|
|
prometheus.CounterValue,
|
|
d.FramesSkippedPerSecondInsufficientServerResources,
|
|
d.Name,
|
|
"server",
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.GraphicsCompressionratio,
|
|
prometheus.GaugeValue,
|
|
d.GraphicsCompressionratio,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.InputFramesPerSecond,
|
|
prometheus.CounterValue,
|
|
d.InputFramesPerSecond,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.OutputFramesPerSecond,
|
|
prometheus.CounterValue,
|
|
d.OutputFramesPerSecond,
|
|
d.Name,
|
|
)
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.SourceFramesPerSecond,
|
|
prometheus.CounterValue,
|
|
d.SourceFramesPerSecond,
|
|
d.Name,
|
|
)
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
func milliSecToSec(t float64) float64 {
|
|
return t / 1000
|
|
}
|