// 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 testutils import ( "errors" "io" "log/slog" "os" "sync" "testing" "time" "github.com/alecthomas/kingpin/v2" "github.com/prometheus-community/windows_exporter/internal/collector/update" "github.com/prometheus-community/windows_exporter/internal/mi" "github.com/prometheus-community/windows_exporter/internal/perfdata" "github.com/prometheus-community/windows_exporter/pkg/collector" "github.com/prometheus/client_golang/prometheus" "github.com/stretchr/testify/require" "golang.org/x/sys/windows" ) func FuncBenchmarkCollector[C collector.Collector](b *testing.B, name string, collectFunc collector.BuilderWithFlags[C]) { b.Helper() logger := slog.New(slog.NewTextHandler(io.Discard, nil)) c := collectFunc(kingpin.CommandLine) collectors := collector.New(map[string]collector.Collector{name: c}) require.NoError(b, collectors.Build(logger)) metrics := make(chan prometheus.Metric) go func() { for { <-metrics } }() for i := 0; i < b.N; i++ { require.NoError(b, c.Collect(metrics)) } } func TestCollector[C collector.Collector, V interface{}](t *testing.T, fn func(*V) C, conf *V) { t.Helper() var ( metrics []prometheus.Metric err error ) logger := slog.New(slog.NewTextHandler(io.Discard, nil)) c := fn(conf) ch := make(chan prometheus.Metric, 10000) miApp, err := mi.Application_Initialize() require.NoError(t, err) miSession, err := miApp.NewSession(nil) require.NoError(t, err) t.Cleanup(func() { require.NoError(t, c.Close()) require.NoError(t, miSession.Close()) require.NoError(t, miApp.Close()) }) wg := sync.WaitGroup{} wg.Add(1) go func() { defer wg.Done() for metric := range ch { metrics = append(metrics, metric) } }() err = c.Build(logger, miSession) switch { case err == nil: case errors.Is(err, mi.MI_RESULT_INVALID_NAMESPACE), errors.Is(err, perfdata.NewPdhError(perfdata.PdhCstatusNoCounter)), errors.Is(err, perfdata.NewPdhError(perfdata.PdhCstatusNoObject)), errors.Is(err, update.ErrUpdateServiceDisabled), errors.Is(err, os.ErrNotExist): default: require.NoError(t, err) } time.Sleep(1 * time.Second) err = c.Collect(ch) switch { // container collector case errors.Is(err, windows.Errno(2151088411)), errors.Is(err, perfdata.ErrPerformanceCounterNotInitialized), errors.Is(err, perfdata.ErrNoData), errors.Is(err, mi.MI_RESULT_INVALID_NAMESPACE), errors.Is(err, mi.MI_RESULT_INVALID_QUERY), errors.Is(err, update.ErrNoUpdates): t.Skip("collector not supported on this system") default: require.NoError(t, err) } close(ch) wg.Wait() }