2023-01-28 20:56:57 +01:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"runtime"
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
2023-12-12 18:42:50 +01:00
|
|
|
|
2023-01-28 20:56:57 +01:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/collectors"
|
2023-12-12 18:42:50 +01:00
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
2023-01-28 20:56:57 +01:00
|
|
|
"github.com/prometheus/common/expfmt"
|
|
|
|
|
2023-12-12 18:42:50 +01:00
|
|
|
"github.com/photoprism/photoprism/internal/acl"
|
2023-01-28 20:56:57 +01:00
|
|
|
"github.com/photoprism/photoprism/internal/config"
|
|
|
|
"github.com/photoprism/photoprism/internal/get"
|
|
|
|
)
|
|
|
|
|
2023-12-12 18:42:50 +01:00
|
|
|
// GetMetrics provides a prometheus-compatible metrics endpoint for monitoring.
|
|
|
|
//
|
2023-01-28 20:56:57 +01:00
|
|
|
// GET /api/v1/metrics
|
|
|
|
func GetMetrics(router *gin.RouterGroup) {
|
|
|
|
router.GET("/metrics", func(c *gin.Context) {
|
2023-12-12 18:42:50 +01:00
|
|
|
s := Auth(c, acl.ResourceMetrics, acl.AccessAll)
|
|
|
|
|
|
|
|
// Abort if permission was not granted.
|
|
|
|
if s.Abort(c) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-01-28 20:56:57 +01:00
|
|
|
conf := get.Config()
|
2024-01-29 14:18:17 +01:00
|
|
|
counts := conf.ClientUser(false).Count
|
2023-01-28 20:56:57 +01:00
|
|
|
|
|
|
|
c.Stream(func(w io.Writer) bool {
|
|
|
|
reg := prometheus.NewRegistry()
|
|
|
|
reg.MustRegister(collectors.NewGoCollector())
|
|
|
|
|
|
|
|
factory := promauto.With(reg)
|
|
|
|
|
|
|
|
registerCountMetrics(factory, counts)
|
|
|
|
registerBuildInfoMetric(factory, conf.ClientPublic())
|
|
|
|
|
|
|
|
metrics, err := reg.Gather()
|
|
|
|
if err != nil {
|
2024-01-18 11:23:59 +01:00
|
|
|
logErr("metrics", err)
|
2023-01-28 20:56:57 +01:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, metric := range metrics {
|
|
|
|
if _, err := expfmt.MetricFamilyToText(w, metric); err != nil {
|
2024-01-18 11:23:59 +01:00
|
|
|
logErr("metrics", err)
|
2023-01-28 20:56:57 +01:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-12-12 18:42:50 +01:00
|
|
|
// registerCountMetrics registers metrics that can be monitored with the /api/v1/metrics endpoint.=
|
2023-01-28 20:56:57 +01:00
|
|
|
func registerCountMetrics(factory promauto.Factory, counts config.ClientCounts) {
|
|
|
|
metric := factory.NewGaugeVec(
|
|
|
|
prometheus.GaugeOpts{
|
|
|
|
Namespace: "photoprism",
|
|
|
|
Subsystem: "statistics",
|
|
|
|
Name: "media_count",
|
|
|
|
Help: "media statistics for this photoprism instance",
|
|
|
|
}, []string{"stat"},
|
|
|
|
)
|
|
|
|
|
|
|
|
metric.With(prometheus.Labels{"stat": "all"}).Set(float64(counts.All))
|
|
|
|
metric.With(prometheus.Labels{"stat": "photos"}).Set(float64(counts.Photos))
|
|
|
|
metric.With(prometheus.Labels{"stat": "videos"}).Set(float64(counts.Videos))
|
|
|
|
metric.With(prometheus.Labels{"stat": "albums"}).Set(float64(counts.Albums))
|
|
|
|
metric.With(prometheus.Labels{"stat": "folders"}).Set(float64(counts.Folders))
|
|
|
|
metric.With(prometheus.Labels{"stat": "files"}).Set(float64(counts.Files))
|
|
|
|
}
|
|
|
|
|
2023-12-12 18:42:50 +01:00
|
|
|
// registerBuildInfoMetric registers a metric that provides build information.
|
2023-01-28 20:56:57 +01:00
|
|
|
func registerBuildInfoMetric(factory promauto.Factory, conf config.ClientConfig) {
|
|
|
|
factory.NewGaugeVec(
|
|
|
|
prometheus.GaugeOpts{
|
|
|
|
Namespace: "photoprism",
|
|
|
|
Name: "build_info",
|
|
|
|
Help: "information about the photoprism instance",
|
|
|
|
}, []string{"edition", "goversion", "version"},
|
|
|
|
).With(prometheus.Labels{
|
|
|
|
"edition": conf.Edition,
|
|
|
|
"goversion": runtime.Version(),
|
|
|
|
"version": conf.Version,
|
|
|
|
}).Set(1.0)
|
|
|
|
}
|