File: metrics.go

package info (click to toggle)
golang-github-containerd-stargz-snapshotter 0.14.3-4
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 3,348 kB
  • sloc: sh: 3,634; python: 534; makefile: 91; ansic: 4
file content (113 lines) | stat: -rw-r--r-- 2,593 bytes parent folder | download | duplicates (4)
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
/*
   Copyright The containerd 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.
*/

package layermetrics

import (
	"sync"

	"github.com/containerd/stargz-snapshotter/fs/layer"
	metrics "github.com/docker/go-metrics"
	"github.com/prometheus/client_golang/prometheus"
)

func NewLayerMetrics(ns *metrics.Namespace) *Controller {
	if ns == nil {
		return &Controller{}
	}
	c := &Controller{
		ns:    ns,
		layer: make(map[string]layer.Layer),
	}
	c.metrics = append(c.metrics, layerMetrics...)
	ns.Add(c)
	return c
}

type Controller struct {
	ns      *metrics.Namespace
	metrics []*metric

	layer   map[string]layer.Layer
	layerMu sync.RWMutex
}

func (c *Controller) Describe(ch chan<- *prometheus.Desc) {
	for _, e := range c.metrics {
		ch <- e.desc(c.ns)
	}
}

func (c *Controller) Collect(ch chan<- prometheus.Metric) {
	c.layerMu.RLock()
	wg := &sync.WaitGroup{}
	for mp, l := range c.layer {
		mp, l := mp, l
		wg.Add(1)
		go func() {
			defer wg.Done()
			for _, e := range c.metrics {
				e.collect(mp, l, c.ns, ch)
			}
		}()
	}
	c.layerMu.RUnlock()
	wg.Wait()
}

func (c *Controller) Add(key string, l layer.Layer) {
	if c.ns == nil {
		return
	}
	c.layerMu.Lock()
	c.layer[key] = l
	c.layerMu.Unlock()
}

func (c *Controller) Remove(key string) {
	if c.ns == nil {
		return
	}
	c.layerMu.Lock()
	delete(c.layer, key)
	c.layerMu.Unlock()
}

type value struct {
	v float64
	l []string
}

type metric struct {
	name   string
	help   string
	unit   metrics.Unit
	vt     prometheus.ValueType
	labels []string
	// getValues returns the value and labels for the data
	getValues func(l layer.Layer) []value
}

func (m *metric) desc(ns *metrics.Namespace) *prometheus.Desc {
	return ns.NewDesc(m.name, m.help, m.unit, append([]string{"digest", "mountpoint"}, m.labels...)...)
}

func (m *metric) collect(mountpoint string, l layer.Layer, ns *metrics.Namespace, ch chan<- prometheus.Metric) {
	values := m.getValues(l)
	for _, v := range values {
		ch <- prometheus.MustNewConstMetric(m.desc(ns), m.vt, v.v, append([]string{l.Info().Digest.String(), mountpoint}, v.l...)...)
	}
}