@@ -181,3 +181,189 @@ func getHTTPBodyAsLines(t *testing.T, url string) []string {
181
181
resp .Body .Close ()
182
182
return lines
183
183
}
184
+
185
+ func TestAllMetricsGenerated (t * testing.T ) {
186
+ integration2 .BeforeTest (t )
187
+
188
+ var (
189
+ addr = "localhost:27989"
190
+ ln net.Listener
191
+ )
192
+
193
+ srv := & http.Server {Handler : promhttp .Handler ()}
194
+ srv .SetKeepAlivesEnabled (false )
195
+
196
+ ln , err := transport .NewUnixListener (addr )
197
+ if err != nil {
198
+ t .Errorf ("Error: %v occurred while listening on addr: %v" , err , addr )
199
+ }
200
+
201
+ donec := make (chan struct {})
202
+ defer func () {
203
+ ln .Close ()
204
+ <- donec
205
+ }()
206
+
207
+ // listen for all Prometheus metrics
208
+ go func () {
209
+ defer close (donec )
210
+
211
+ serr := srv .Serve (ln )
212
+ if serr != nil && ! transport .IsClosedConnError (serr ) {
213
+ t .Errorf ("Err serving http requests: %v" , serr )
214
+ }
215
+ }()
216
+
217
+ url := "unix://" + addr + "/metrics"
218
+
219
+ clus := integration2 .NewCluster (t , & integration2.ClusterConfig {Size : 1 , Metrics : "extensive" })
220
+ defer clus .Terminate (t )
221
+
222
+ clientMetrics := grpcprom .NewClientMetrics ()
223
+ prometheus .Register (clientMetrics )
224
+
225
+ cfg := clientv3.Config {
226
+ Endpoints : []string {clus .Members [0 ].GRPCURL },
227
+ DialOptions : []grpc.DialOption {
228
+ grpc .WithUnaryInterceptor (clientMetrics .UnaryClientInterceptor ()),
229
+ grpc .WithStreamInterceptor (clientMetrics .StreamClientInterceptor ()),
230
+ },
231
+ }
232
+ cli , cerr := integration2 .NewClient (t , cfg )
233
+ if cerr != nil {
234
+ t .Fatal (cerr )
235
+ }
236
+ defer cli .Close ()
237
+
238
+ // Perform some operations to generate metrics
239
+ wc := cli .Watch (context .Background (), "foo" )
240
+ _ , err = cli .Put (context .Background (), "foo" , "bar" )
241
+ if err != nil {
242
+ t .Errorf ("Error putting value in key store" )
243
+ }
244
+
245
+ // consume watch response
246
+ select {
247
+ case <- wc :
248
+ case <- time .After (10 * time .Second ):
249
+ t .Error ("Timeout occurred for getting watch response" )
250
+ }
251
+
252
+ // Define the expected list of metrics
253
+ expectedMetrics := []string {
254
+ "etcd_cluster_version" ,
255
+ "etcd_disk_backend_commit_duration_seconds_bucket" ,
256
+ "etcd_disk_backend_commit_duration_seconds_count" ,
257
+ "etcd_disk_backend_commit_duration_seconds_sum" ,
258
+ "etcd_disk_backend_defrag_duration_seconds_bucket" ,
259
+ "etcd_disk_backend_defrag_duration_seconds_count" ,
260
+ "etcd_disk_backend_defrag_duration_seconds_sum" ,
261
+ "etcd_disk_backend_snapshot_duration_seconds_bucket" ,
262
+ "etcd_disk_backend_snapshot_duration_seconds_count" ,
263
+ "etcd_disk_backend_snapshot_duration_seconds_sum" ,
264
+ "etcd_disk_defrag_inflight" ,
265
+ "etcd_disk_wal_fsync_duration_seconds_bucket" ,
266
+ "etcd_disk_wal_fsync_duration_seconds_count" ,
267
+ "etcd_disk_wal_fsync_duration_seconds_sum" ,
268
+ "etcd_disk_wal_write_bytes_total" ,
269
+ "etcd_disk_wal_write_duration_seconds_bucket" ,
270
+ "etcd_disk_wal_write_duration_seconds_count" ,
271
+ "etcd_disk_wal_write_duration_seconds_sum" ,
272
+ "etcd_mvcc_db_open_read_transactions" ,
273
+ "etcd_mvcc_db_total_size_in_bytes" ,
274
+ "etcd_mvcc_db_total_size_in_use_in_bytes" ,
275
+ "etcd_mvcc_delete_total" ,
276
+ "etcd_mvcc_hash_duration_seconds_bucket" ,
277
+ "etcd_mvcc_hash_duration_seconds_count" ,
278
+ "etcd_mvcc_hash_duration_seconds_sum" ,
279
+ "etcd_mvcc_hash_rev_duration_seconds_bucket" ,
280
+ "etcd_mvcc_hash_rev_duration_seconds_count" ,
281
+ "etcd_mvcc_hash_rev_duration_seconds_sum" ,
282
+ "etcd_mvcc_put_total" ,
283
+ "etcd_mvcc_range_total" ,
284
+ "etcd_mvcc_txn_total" ,
285
+ "etcd_network_client_grpc_received_bytes_total" ,
286
+ "etcd_network_client_grpc_sent_bytes_total" ,
287
+ "etcd_network_known_peers" ,
288
+ "etcd_server_apply_duration_seconds_bucket" ,
289
+ "etcd_server_apply_duration_seconds_count" ,
290
+ "etcd_server_apply_duration_seconds_sum" ,
291
+ "etcd_server_client_requests_total" ,
292
+ "etcd_server_go_version" ,
293
+ "etcd_server_has_leader" ,
294
+ "etcd_server_health_failures" ,
295
+ "etcd_server_health_success" ,
296
+ "etcd_server_heartbeat_send_failures_total" ,
297
+ "etcd_server_id" ,
298
+ "etcd_server_is_leader" ,
299
+ "etcd_server_is_learner" ,
300
+ "etcd_server_leader_changes_seen_total" ,
301
+ "etcd_server_learner_promote_successes" ,
302
+ "etcd_server_proposals_applied_total" ,
303
+ "etcd_server_proposals_committed_total" ,
304
+ "etcd_server_proposals_failed_total" ,
305
+ "etcd_server_proposals_pending" ,
306
+ "etcd_server_quota_backend_bytes" ,
307
+ "etcd_server_read_indexes_failed_total" ,
308
+ "etcd_server_slow_apply_total" ,
309
+ "etcd_server_slow_read_indexes_total" ,
310
+ "etcd_server_snapshot_apply_in_progress_total" ,
311
+ "etcd_server_version" ,
312
+ "etcd_snap_db_fsync_duration_seconds_bucket" ,
313
+ "etcd_snap_db_fsync_duration_seconds_count" ,
314
+ "etcd_snap_db_fsync_duration_seconds_sum" ,
315
+ "etcd_snap_db_save_total_duration_seconds_bucket" ,
316
+ "etcd_snap_db_save_total_duration_seconds_count" ,
317
+ "etcd_snap_db_save_total_duration_seconds_sum" ,
318
+ "etcd_snap_fsync_duration_seconds_bucket" ,
319
+ "etcd_snap_fsync_duration_seconds_count" ,
320
+ "etcd_snap_fsync_duration_seconds_sum" ,
321
+ "grpc_client_handled_total" ,
322
+ "grpc_client_msg_received_total" ,
323
+ "grpc_client_msg_sent_total" ,
324
+ "grpc_client_started_total" ,
325
+ "grpc_server_handled_total" ,
326
+ "grpc_server_handling_seconds_bucket" ,
327
+ "grpc_server_handling_seconds_count" ,
328
+ "grpc_server_handling_seconds_sum" ,
329
+ "grpc_server_msg_received_total" ,
330
+ "grpc_server_msg_sent_total" ,
331
+ "grpc_server_started_total" ,
332
+ }
333
+
334
+ // Get the list of generated metrics
335
+ generatedMetrics := getMetricsList (t , url )
336
+ for _ , metric := range expectedMetrics {
337
+ if ! contains (generatedMetrics , metric ) {
338
+ t .Errorf ("Expected metric %s not found in generated metrics" , metric )
339
+ }
340
+ }
341
+ }
342
+
343
+ func getMetricsList (t * testing.T , url string ) []string {
344
+ lines := getHTTPBodyAsLines (t , url )
345
+ metrics := make (map [string ]struct {})
346
+ for _ , line := range lines {
347
+ if strings .Contains (line , "{" ) {
348
+ metric := line [:strings .Index (line , "{" )]
349
+ metrics [metric ] = struct {}{}
350
+ } else {
351
+ metric := line [:strings .Index (line , " " )]
352
+ metrics [metric ] = struct {}{}
353
+ }
354
+ }
355
+ var metricList []string
356
+ for metric := range metrics {
357
+ metricList = append (metricList , metric )
358
+ }
359
+ return metricList
360
+ }
361
+
362
+ func contains (slice []string , item string ) bool {
363
+ for _ , s := range slice {
364
+ if s == item {
365
+ return true
366
+ }
367
+ }
368
+ return false
369
+ }
0 commit comments