-
Notifications
You must be signed in to change notification settings - Fork 2.5k
/
Copy pathscraper_windows.go
145 lines (119 loc) · 4.59 KB
/
scraper_windows.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
// Copyright The OpenTelemetry Authors
// SPDX-License-Identifier: Apache-2.0
//go:build windows
package sqlserverreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/sqlserverreceiver"
import (
"context"
"time"
"go.opentelemetry.io/collector/component"
"go.opentelemetry.io/collector/pdata/pcommon"
"go.opentelemetry.io/collector/pdata/pmetric"
"go.opentelemetry.io/collector/receiver"
"go.uber.org/multierr"
"go.uber.org/zap"
"github.com/open-telemetry/opentelemetry-collector-contrib/pkg/winperfcounters"
"github.com/open-telemetry/opentelemetry-collector-contrib/receiver/sqlserverreceiver/internal/metadata"
)
// SQL Server Perf Counter (PC) Scraper. This is used to scrape metrics from Windows Perf Counters.
type sqlServerPCScraper struct {
logger *zap.Logger
config *Config
watcherRecorders []watcherRecorder
mb *metadata.MetricsBuilder
}
// watcherRecorder is a struct containing perf counter watcher along with corresponding value recorder.
type watcherRecorder struct {
watcher winperfcounters.PerfCounterWatcher
recorder recordFunc
}
// curriedRecorder is a recorder function that already has value to be recorded,
// it needs metadata.MetricsBuilder and timestamp as arguments.
type curriedRecorder func(*metadata.MetricsBuilder, pcommon.Timestamp)
// newSQLServerPCScraper returns a new sqlServerPCScraper.
func newSQLServerPCScraper(params receiver.Settings, cfg *Config) *sqlServerPCScraper {
return &sqlServerPCScraper{
logger: params.Logger,
config: cfg,
mb: metadata.NewMetricsBuilder(cfg.MetricsBuilderConfig, params),
}
}
// start creates and sets the watchers for the scraper.
func (s *sqlServerPCScraper) start(_ context.Context, _ component.Host) error {
s.watcherRecorders = []watcherRecorder{}
for _, pcr := range perfCounterRecorders {
for perfCounterName, recorder := range pcr.recorders {
perfCounterObj := defaultObjectName + ":" + pcr.object
if s.config.InstanceName != "" {
// The instance name must be preceded by "MSSQL$" to indicate that it is a named instance
perfCounterObj = "\\" + s.config.ComputerName + "\\MSSQL$" + s.config.InstanceName + ":" + pcr.object
}
w, err := winperfcounters.NewWatcher(perfCounterObj, pcr.instance, perfCounterName)
if err != nil {
s.logger.Warn(err.Error())
continue
}
s.watcherRecorders = append(s.watcherRecorders, watcherRecorder{w, recorder})
}
}
return nil
}
// scrape collects windows performance counter data from all watchers and then records/emits it using the metricBuilder
func (s *sqlServerPCScraper) scrape(_ context.Context) (pmetric.Metrics, error) {
recordersByDatabase, errs := recordersPerDatabase(s.watcherRecorders)
for dbName, recorders := range recordersByDatabase {
s.emitMetricGroup(recorders, dbName)
}
return s.mb.Emit(), errs
}
// recordersPerDatabase scrapes perf counter values using provided []watcherRecorder and returns
// a map of database name to curriedRecorder that includes the recorded value in its closure.
func recordersPerDatabase(watcherRecorders []watcherRecorder) (map[string][]curriedRecorder, error) {
var errs error
dbToRecorders := make(map[string][]curriedRecorder)
for _, wr := range watcherRecorders {
counterValues, err := wr.watcher.ScrapeData()
if err != nil {
errs = multierr.Append(errs, err)
continue
}
for _, counterValue := range counterValues {
dbName := counterValue.InstanceName
// it's important to initialize new values for the closure.
val := counterValue.Value
recorder := wr.recorder
if _, ok := dbToRecorders[dbName]; !ok {
dbToRecorders[dbName] = []curriedRecorder{}
}
dbToRecorders[dbName] = append(dbToRecorders[dbName], func(mb *metadata.MetricsBuilder, ts pcommon.Timestamp) {
recorder(mb, ts, val)
})
}
}
return dbToRecorders, errs
}
func (s *sqlServerPCScraper) emitMetricGroup(recorders []curriedRecorder, databaseName string) {
now := pcommon.NewTimestampFromTime(time.Now())
for _, recorder := range recorders {
recorder(s.mb, now)
}
rb := s.mb.NewResourceBuilder()
if databaseName != "" {
rb.SetSqlserverDatabaseName(databaseName)
}
if s.config.InstanceName != "" {
rb.SetSqlserverComputerName(s.config.ComputerName)
rb.SetSqlserverInstanceName(s.config.InstanceName)
}
s.mb.EmitForResource(metadata.WithResource(rb.Emit()))
}
// shutdown stops all of the watchers for the scraper.
func (s sqlServerPCScraper) shutdown(_ context.Context) error {
var errs error
for _, wr := range s.watcherRecorders {
err := wr.watcher.Close()
if err != nil {
errs = multierr.Append(errs, err)
}
}
return errs
}