// 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 } } }