package prometheus import ( "fmt" "io" "sync" "time" "github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus/collectors" dto "github.com/prometheus/client_model/go" "github.com/prometheus/common/expfmt" "go.unistack.org/micro/v3/meter" ) type prometheusMeter struct { opts meter.Options set prometheus.Registerer counter map[string]prometheusCounter floatCounter map[string]prometheusFloatCounter gauge map[string]prometheusGauge histogram map[string]prometheusHistogram summary map[string]prometheusSummary sync.Mutex } func NewMeter(opts ...meter.Option) meter.Meter { return &prometheusMeter{ set: prometheus.DefaultRegisterer, opts: meter.NewOptions(opts...), counter: make(map[string]prometheusCounter), floatCounter: make(map[string]prometheusFloatCounter), gauge: make(map[string]prometheusGauge), histogram: make(map[string]prometheusHistogram), summary: make(map[string]prometheusSummary), } } func (m *prometheusMeter) buildMetric(name string, labels ...string) string { if len(m.opts.MetricPrefix) > 0 { name = m.opts.MetricPrefix + name } nl := len(m.opts.Labels) + len(labels) if nl == 0 { return name } nlabels := make([]string, 0, nl) nlabels = append(nlabels, m.opts.Labels...) nlabels = append(nlabels, labels...) if len(m.opts.LabelPrefix) == 0 { return meter.BuildName(name, nlabels...) } for idx := 0; idx < nl; idx++ { nlabels[idx] = m.opts.LabelPrefix + nlabels[idx] idx++ } return meter.BuildName(name, nlabels...) } func (m *prometheusMeter) buildName(name string) string { if len(m.opts.MetricPrefix) > 0 { name = m.opts.MetricPrefix + name } return name } func (m *prometheusMeter) buildLabels(labels ...string) []string { nl := len(m.opts.Labels) + len(labels) if nl == 0 { return nil } nlabels := make([]string, 0, nl) nlabels = append(nlabels, m.opts.Labels...) nlabels = append(nlabels, labels...) for idx := 0; idx < nl; idx++ { nlabels[idx] = m.opts.LabelPrefix + nlabels[idx] idx++ } return nlabels } func (m *prometheusMeter) Name() string { return m.opts.Name } func (m *prometheusMeter) mapLabels(src ...string) map[string]string { src = m.buildLabels(src...) mp := make(map[string]string, len(src)/2) for idx := 0; idx < len(src); idx++ { mp[src[idx]] = src[idx+1] idx++ } return mp } func (m *prometheusMeter) metricEqual(src []string, dst []string) bool { if len(src) != len(dst)/2 { return false } dst = m.buildLabels(dst...) mp := make(map[string]struct{}, len(src)) for idx := range src { mp[src[idx]] = struct{}{} } for idx := 0; idx < len(dst); idx++ { if _, ok := mp[dst[idx]]; !ok { return false } idx++ } return true } func (m *prometheusMeter) labelNames(src map[string]string, dst []string) []string { dst = m.buildLabels(dst...) nlabels := make([]string, 0, len(src)) for idx := 0; idx < len(dst); idx++ { if src == nil { nlabels = append(nlabels, dst[idx]) } else if _, ok := src[dst[idx]]; ok { nlabels = append(nlabels, dst[idx]) } else { nlabels = append(nlabels, dst[idx]) } idx++ } return nlabels } func (m *prometheusMeter) Counter(name string, labels ...string) meter.Counter { m.Lock() defer m.Unlock() nm := m.buildName(name) c, ok := m.counter[nm] var lnames []string if !ok { lnames = m.labelNames(nil, labels) fmt.Printf("!ok lnames: %v\n", lnames) nc := prometheus.NewGaugeVec(prometheus.GaugeOpts{Name: nm}, lnames) c = prometheusCounter{c: nc, lnames: lnames} m.counter[nm] = c } else if !m.metricEqual(c.lnames, labels) { fmt.Printf("ok && !m.metricEqual lnames: %v labels: %v\n", c.lnames, labels) lnames = m.labelNames(c.labels, labels) m.set.Unregister(c.c) nc := prometheus.NewGaugeVec(prometheus.GaugeOpts{Name: nm}, lnames) c = prometheusCounter{c: nc, lnames: lnames} m.counter[nm] = c m.set.MustRegister(c.c) } else { lnames = c.lnames } fmt.Printf("lnames %v\n", lnames) return prometheusCounter{c: c.c, lnames: lnames, labels: m.mapLabels(labels...)} } func (m *prometheusMeter) FloatCounter(name string, labels ...string) meter.FloatCounter { m.Lock() defer m.Unlock() nm := m.buildName(name) c, ok := m.floatCounter[nm] if !ok { nc := prometheus.NewGaugeVec(prometheus.GaugeOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusFloatCounter{c: nc} m.floatCounter[nm] = c } else if !m.metricEqual(c.lnames, labels) { m.set.Unregister(c.c) nc := prometheus.NewGaugeVec(prometheus.GaugeOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusFloatCounter{c: nc} m.floatCounter[nm] = c m.set.MustRegister(c.c) } return prometheusFloatCounter{c: c.c, labels: m.mapLabels(labels...)} } func (m *prometheusMeter) Gauge(name string, fn func() float64, labels ...string) meter.Gauge { m.Lock() defer m.Unlock() nm := m.buildName(name) c, ok := m.gauge[nm] if !ok { nc := prometheus.NewGaugeVec(prometheus.GaugeOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusGauge{c: nc} m.gauge[nm] = c } else if !m.metricEqual(c.lnames, labels) { m.set.Unregister(c.c) nc := prometheus.NewGaugeVec(prometheus.GaugeOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusGauge{c: nc} m.gauge[nm] = c m.set.MustRegister(c.c) } return prometheusGauge{c: c.c, labels: m.mapLabels(labels...)} } func (m *prometheusMeter) Histogram(name string, labels ...string) meter.Histogram { m.Lock() defer m.Unlock() nm := m.buildName(name) c, ok := m.histogram[nm] if !ok { nc := prometheus.NewHistogramVec(prometheus.HistogramOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusHistogram{c: nc} m.histogram[nm] = c } else if !m.metricEqual(c.lnames, labels) { m.set.Unregister(c.c) nc := prometheus.NewHistogramVec(prometheus.HistogramOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusHistogram{c: nc} m.histogram[nm] = c m.set.MustRegister(c.c) } return prometheusHistogram{c: c.c, labels: m.mapLabels(labels...)} } func (m *prometheusMeter) Summary(name string, labels ...string) meter.Summary { m.Lock() defer m.Unlock() nm := m.buildName(name) c, ok := m.summary[nm] if !ok { nc := prometheus.NewSummaryVec(prometheus.SummaryOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusSummary{c: nc} m.summary[nm] = c } else if !m.metricEqual(c.lnames, labels) { m.set.Unregister(c.c) nc := prometheus.NewSummaryVec(prometheus.SummaryOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusSummary{c: nc} m.summary[nm] = c m.set.MustRegister(c.c) } return prometheusSummary{c: c.c, labels: m.mapLabels(labels...)} } func (m *prometheusMeter) SummaryExt(name string, window time.Duration, quantiles []float64, labels ...string) meter.Summary { m.Lock() defer m.Unlock() nm := m.buildName(name) c, ok := m.summary[nm] if !ok { nc := prometheus.NewSummaryVec(prometheus.SummaryOpts{ Name: nm, MaxAge: window, Objectives: map[float64]float64{0.5: 0.05, 0.9: 0.01, 0.99: 0.001}, }, m.labelNames(c.labels, labels)) c = prometheusSummary{c: nc} m.summary[nm] = c } else if !m.metricEqual(c.lnames, labels) { m.set.Unregister(c.c) nc := prometheus.NewSummaryVec(prometheus.SummaryOpts{Name: nm}, m.labelNames(c.labels, labels)) c = prometheusSummary{c: nc} m.summary[nm] = c m.set.MustRegister(c.c) } return prometheusSummary{c: c.c, labels: m.mapLabels(labels...)} } func (m *prometheusMeter) Init(opts ...meter.Option) error { for _, o := range opts { o(&m.opts) } return nil } func (m *prometheusMeter) Write(w io.Writer, opts ...meter.Option) error { options := m.opts for _, o := range opts { o(&options) } if options.WriteProcessMetrics || options.WriteFDMetrics { c := collectors.NewProcessCollector(collectors.ProcessCollectorOpts{}) _ = m.set.Register(c) } g, ok := m.set.(prometheus.Gatherer) if !ok { return fmt.Errorf("set type %T not prometheus.Gatherer", m.set) } mfs, err := g.Gather() if err != nil { return err } enc := expfmt.NewEncoder(w, expfmt.FmtText) for _, mf := range mfs { _ = enc.Encode(mf) } if closer, ok := enc.(io.Closer); ok { _ = closer.Close() } return nil } func (m *prometheusMeter) Clone(opts ...meter.Option) meter.Meter { options := m.opts for _, o := range opts { o(&options) } return &prometheusMeter{ set: m.set, opts: options, counter: m.counter, gauge: m.gauge, histogram: m.histogram, summary: m.summary, } } func (m *prometheusMeter) Options() meter.Options { return m.opts } func (m *prometheusMeter) String() string { return "prometheus" } func (m *prometheusMeter) Set(opts ...meter.Option) meter.Meter { nm := &prometheusMeter{opts: m.opts} for _, o := range opts { o(&nm.opts) } nm.set = prometheus.NewRegistry() return nm } type prometheusCounter struct { c *prometheus.GaugeVec lnames []string labels prometheus.Labels } func (c prometheusCounter) Add(n int) { nc, _ := c.c.GetMetricWith(c.labels) nc.Add(float64(n)) } func (c prometheusCounter) Dec() { c.c.With(c.labels).Dec() } func (c prometheusCounter) Inc() { c.c.With(c.labels).Inc() } func (c prometheusCounter) Get() uint64 { m := &dto.Metric{} if err := c.c.With(c.labels).Write(m); err != nil { return 0 } return uint64(m.GetGauge().GetValue()) } func (c prometheusCounter) Set(n uint64) { c.c.With(c.labels).Set(float64(n)) } type prometheusFloatCounter struct { c *prometheus.GaugeVec lnames []string labels prometheus.Labels } func (c prometheusFloatCounter) Add(n float64) { c.c.With(c.labels).Add(n) } func (c prometheusFloatCounter) Get() float64 { m := &dto.Metric{} if err := c.c.With(c.labels).Write(m); err != nil { return 0 } return m.GetGauge().GetValue() } func (c prometheusFloatCounter) Set(n float64) { c.c.With(c.labels).Set(n) } func (c prometheusFloatCounter) Sub(n float64) { c.c.With(c.labels).Add(-n) } type prometheusGauge struct { c *prometheus.GaugeVec lnames []string labels prometheus.Labels } func (c prometheusGauge) Get() float64 { m := &dto.Metric{} if err := c.c.With(c.labels).Write(m); err != nil { return 0 } return float64(m.GetGauge().GetValue()) } type prometheusHistogram struct { c *prometheus.HistogramVec lnames []string labels prometheus.Labels } func (c prometheusHistogram) Reset() { } func (c prometheusHistogram) Update(n float64) { c.c.With(c.labels).Observe(n) } func (c prometheusHistogram) UpdateDuration(n time.Time) { c.c.With(c.labels).Observe(time.Since(n).Seconds()) } type prometheusSummary struct { c *prometheus.SummaryVec lnames []string labels prometheus.Labels } func (c prometheusSummary) Update(n float64) { c.c.With(c.labels).Observe(n) } func (c prometheusSummary) UpdateDuration(n time.Time) { c.c.With(c.labels).Observe(time.Since(n).Seconds()) }