mirror of
https://github.com/prometheus-community/windows_exporter
synced 2025-01-09 15:59:38 +00:00
a9f8b3b722
Signed-off-by: Jan-Otto Kröpke <mail@jkroepke.de>
304 lines
7.7 KiB
Go
304 lines
7.7 KiB
Go
// Copyright 2024 The Prometheus Authors
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
//go:build windows
|
|
|
|
package iis
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"regexp"
|
|
"slices"
|
|
"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"
|
|
"golang.org/x/sys/windows/registry"
|
|
)
|
|
|
|
const Name = "iis"
|
|
|
|
type Config struct {
|
|
SiteInclude *regexp.Regexp `yaml:"site_include"`
|
|
SiteExclude *regexp.Regexp `yaml:"site_exclude"`
|
|
AppInclude *regexp.Regexp `yaml:"app_include"`
|
|
AppExclude *regexp.Regexp `yaml:"app_exclude"`
|
|
}
|
|
|
|
//nolint:gochecknoglobals
|
|
var ConfigDefaults = Config{
|
|
SiteInclude: types.RegExpAny,
|
|
SiteExclude: types.RegExpEmpty,
|
|
AppInclude: types.RegExpAny,
|
|
AppExclude: types.RegExpEmpty,
|
|
}
|
|
|
|
type Collector struct {
|
|
config Config
|
|
iisVersion simpleVersion
|
|
|
|
info *prometheus.Desc
|
|
collectorWebService
|
|
collectorAppPoolWAS
|
|
collectorW3SVCW3WP
|
|
collectorWebServiceCache
|
|
}
|
|
|
|
func New(config *Config) *Collector {
|
|
if config == nil {
|
|
config = &ConfigDefaults
|
|
}
|
|
|
|
if config.AppExclude == nil {
|
|
config.AppExclude = ConfigDefaults.AppExclude
|
|
}
|
|
|
|
if config.AppInclude == nil {
|
|
config.AppInclude = ConfigDefaults.AppInclude
|
|
}
|
|
|
|
if config.SiteExclude == nil {
|
|
config.SiteExclude = ConfigDefaults.SiteExclude
|
|
}
|
|
|
|
if config.SiteInclude == nil {
|
|
config.SiteInclude = ConfigDefaults.SiteInclude
|
|
}
|
|
|
|
c := &Collector{
|
|
config: *config,
|
|
}
|
|
|
|
return c
|
|
}
|
|
|
|
func NewWithFlags(app *kingpin.Application) *Collector {
|
|
c := &Collector{
|
|
config: ConfigDefaults,
|
|
}
|
|
|
|
var appExclude, appInclude, siteExclude, siteInclude string
|
|
|
|
app.Flag(
|
|
"collector.iis.app-exclude",
|
|
"Regexp of apps to exclude. App name must both match include and not match exclude to be included.",
|
|
).Default("").StringVar(&appExclude)
|
|
|
|
app.Flag(
|
|
"collector.iis.app-include",
|
|
"Regexp of apps to include. App name must both match include and not match exclude to be included.",
|
|
).Default(".+").StringVar(&appInclude)
|
|
|
|
app.Flag(
|
|
"collector.iis.site-exclude",
|
|
"Regexp of sites to exclude. Site name must both match include and not match exclude to be included.",
|
|
).Default("").StringVar(&siteExclude)
|
|
|
|
app.Flag(
|
|
"collector.iis.site-include",
|
|
"Regexp of sites to include. Site name must both match include and not match exclude to be included.",
|
|
).Default(".+").StringVar(&siteInclude)
|
|
|
|
app.Action(func(*kingpin.ParseContext) error {
|
|
var err error
|
|
|
|
c.config.AppExclude, err = regexp.Compile(fmt.Sprintf("^(?:%s)$", appExclude))
|
|
if err != nil {
|
|
return fmt.Errorf("collector.iis.app-exclude: %w", err)
|
|
}
|
|
|
|
c.config.AppInclude, err = regexp.Compile(fmt.Sprintf("^(?:%s)$", appInclude))
|
|
if err != nil {
|
|
return fmt.Errorf("collector.iis.app-include: %w", err)
|
|
}
|
|
|
|
c.config.SiteExclude, err = regexp.Compile(fmt.Sprintf("^(?:%s)$", siteExclude))
|
|
if err != nil {
|
|
return fmt.Errorf("collector.iis.site-exclude: %w", err)
|
|
}
|
|
|
|
c.config.SiteInclude, err = regexp.Compile(fmt.Sprintf("^(?:%s)$", siteInclude))
|
|
if err != nil {
|
|
return fmt.Errorf("collector.iis.site-include: %w", err)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
|
|
return c
|
|
}
|
|
|
|
func (c *Collector) GetName() string {
|
|
return Name
|
|
}
|
|
|
|
func (c *Collector) Close() error {
|
|
c.perfDataCollectorWebService.Close()
|
|
c.perfDataCollectorAppPoolWAS.Close()
|
|
c.w3SVCW3WPPerfDataCollector.Close()
|
|
c.serviceCachePerfDataCollector.Close()
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *Collector) Build(logger *slog.Logger, _ *mi.Session) error {
|
|
logger = logger.With(slog.String("collector", Name))
|
|
|
|
c.iisVersion = c.getIISVersion(logger)
|
|
|
|
c.info = prometheus.NewDesc(
|
|
prometheus.BuildFQName(types.Namespace, Name, "info"),
|
|
"ISS information",
|
|
[]string{},
|
|
prometheus.Labels{"version": fmt.Sprintf("%d.%d", c.iisVersion.major, c.iisVersion.minor)},
|
|
)
|
|
|
|
errs := make([]error, 0, 4)
|
|
|
|
if err := c.buildWebService(); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to build Web Service collector: %w", err))
|
|
}
|
|
|
|
if err := c.buildAppPoolWAS(); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to build APP_POOL_WAS collector: %w", err))
|
|
}
|
|
|
|
if err := c.buildW3SVCW3WP(); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to build W3SVC_W3WP collector: %w", err))
|
|
}
|
|
|
|
if err := c.buildWebServiceCache(); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to build Web Service Cache collector: %w", err))
|
|
}
|
|
|
|
return errors.Join(errs...)
|
|
}
|
|
|
|
type simpleVersion struct {
|
|
major uint64
|
|
minor uint64
|
|
}
|
|
|
|
func (c *Collector) getIISVersion(logger *slog.Logger) simpleVersion {
|
|
k, err := registry.OpenKey(registry.LOCAL_MACHINE, `SOFTWARE\Microsoft\InetStp\`, registry.QUERY_VALUE)
|
|
if err != nil {
|
|
logger.Warn("couldn't open registry to determine IIS version",
|
|
slog.Any("err", err),
|
|
)
|
|
|
|
return simpleVersion{}
|
|
}
|
|
|
|
defer func() {
|
|
err = k.Close()
|
|
if err != nil {
|
|
logger.Warn("Failed to close registry key",
|
|
slog.Any("err", err),
|
|
)
|
|
}
|
|
}()
|
|
|
|
major, _, err := k.GetIntegerValue("MajorVersion")
|
|
if err != nil {
|
|
logger.Warn("Couldn't open registry to determine IIS version",
|
|
slog.Any("err", err),
|
|
)
|
|
|
|
return simpleVersion{}
|
|
}
|
|
|
|
minor, _, err := k.GetIntegerValue("MinorVersion")
|
|
if err != nil {
|
|
logger.Warn("Couldn't open registry to determine IIS version",
|
|
slog.Any("err", err),
|
|
)
|
|
|
|
return simpleVersion{}
|
|
}
|
|
|
|
logger.Debug(fmt.Sprintf("Detected IIS %d.%d\n", major, minor))
|
|
|
|
return simpleVersion{
|
|
major: major,
|
|
minor: minor,
|
|
}
|
|
}
|
|
|
|
// Collect sends the metric values for each metric
|
|
// to the provided prometheus Metric channel.
|
|
func (c *Collector) Collect(ch chan<- prometheus.Metric) error {
|
|
ch <- prometheus.MustNewConstMetric(
|
|
c.info,
|
|
prometheus.GaugeValue,
|
|
1,
|
|
)
|
|
|
|
errs := make([]error, 0, 4)
|
|
|
|
if err := c.collectWebService(ch); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to collect Web Service metrics: %w", err))
|
|
}
|
|
|
|
if err := c.collectAppPoolWAS(ch); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to collect APP_POOL_WAS metrics: %w", err))
|
|
}
|
|
|
|
if err := c.collectW3SVCW3WP(ch); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to collect W3SVC_W3WP metrics: %w", err))
|
|
}
|
|
|
|
if err := c.collectWebServiceCache(ch); err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to collect Web Service Cache metrics: %w", err))
|
|
}
|
|
|
|
return errors.Join(errs...)
|
|
}
|
|
|
|
type collectorName interface {
|
|
GetName() string
|
|
}
|
|
|
|
// deduplicateIISNames deduplicate IIS site names from various IIS perflib objects.
|
|
//
|
|
// E.G. Given the following list of site names, "Site_B" would be
|
|
// discarded, and "Site_B#2" would be kept and presented as "Site_B" in the
|
|
// Collector metrics.
|
|
// [ "Site_A", "Site_B", "Site_C", "Site_B#2" ].
|
|
func deduplicateIISNames[T collectorName](counterValues []T) {
|
|
indexes := make(map[string]int)
|
|
|
|
// Ensure IIS entry with the highest suffix occurs last
|
|
slices.SortFunc(counterValues, func(a, b T) int {
|
|
return strings.Compare(a.GetName(), b.GetName())
|
|
})
|
|
|
|
// Use map to deduplicate IIS entries
|
|
for index, counterValue := range counterValues {
|
|
name := strings.Split(counterValue.GetName(), "#")[0]
|
|
if name == counterValue.GetName() {
|
|
continue
|
|
}
|
|
|
|
if originalIndex, ok := indexes[name]; !ok {
|
|
counterValues[originalIndex] = counterValue
|
|
counterValues = slices.Delete(counterValues, index, 1)
|
|
} else {
|
|
indexes[name] = index
|
|
}
|
|
}
|
|
}
|