2021-02-12 20:37:30 +03:00
|
|
|
|
package api
|
|
|
|
|
|
|
|
|
|
import (
|
2021-02-13 01:06:48 +03:00
|
|
|
|
"errors"
|
2021-02-12 20:37:30 +03:00
|
|
|
|
"github.com/emvi/logbuch"
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
|
conf "github.com/muety/wakapi/config"
|
|
|
|
|
"github.com/muety/wakapi/middlewares"
|
|
|
|
|
"github.com/muety/wakapi/models"
|
|
|
|
|
v1 "github.com/muety/wakapi/models/compat/wakatime/v1"
|
|
|
|
|
mm "github.com/muety/wakapi/models/metrics"
|
|
|
|
|
"github.com/muety/wakapi/services"
|
|
|
|
|
"github.com/muety/wakapi/utils"
|
|
|
|
|
"net/http"
|
2021-10-14 11:22:59 +03:00
|
|
|
|
"runtime"
|
2021-02-12 20:37:30 +03:00
|
|
|
|
"sort"
|
|
|
|
|
"time"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
MetricsPrefix = "wakatime"
|
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
DescHeartbeats = "Total number of tracked heartbeats."
|
2021-02-12 20:37:30 +03:00
|
|
|
|
DescAllTime = "Total seconds (all time)."
|
|
|
|
|
DescTotal = "Total seconds."
|
|
|
|
|
DescEditors = "Total seconds for each editor."
|
|
|
|
|
DescProjects = "Total seconds for each project."
|
|
|
|
|
DescLanguages = "Total seconds for each language."
|
|
|
|
|
DescOperatingSystems = "Total seconds for each operating system."
|
|
|
|
|
DescMachines = "Total seconds for each machine."
|
2021-06-11 21:59:34 +03:00
|
|
|
|
DescLabels = "Total seconds for each project label."
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
DescAdminTotalTime = "Total seconds (all users, all time)."
|
2021-02-13 01:16:20 +03:00
|
|
|
|
DescAdminTotalHeartbeats = "Total number of tracked heartbeats (all users, all time)"
|
|
|
|
|
DescAdminUserHeartbeats = "Total number of tracked heartbeats by user (all time)."
|
2021-02-13 01:06:48 +03:00
|
|
|
|
DescAdminTotalUsers = "Total number of registered users."
|
|
|
|
|
DescAdminActiveUsers = "Number of active users."
|
2021-10-14 11:22:59 +03:00
|
|
|
|
|
2021-10-14 11:35:01 +03:00
|
|
|
|
DescMemAllocTotal = "Total number of bytes allocated for heap"
|
|
|
|
|
DescMemSysTotal = "Total number of bytes obtained from the OS"
|
|
|
|
|
DescGoroutines = "Total number of running goroutines"
|
2021-02-12 20:37:30 +03:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type MetricsHandler struct {
|
|
|
|
|
config *conf.Config
|
|
|
|
|
userSrvc services.IUserService
|
|
|
|
|
summarySrvc services.ISummaryService
|
|
|
|
|
heartbeatSrvc services.IHeartbeatService
|
|
|
|
|
keyValueSrvc services.IKeyValueService
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func NewMetricsHandler(userService services.IUserService, summaryService services.ISummaryService, heartbeatService services.IHeartbeatService, keyValueService services.IKeyValueService) *MetricsHandler {
|
|
|
|
|
return &MetricsHandler{
|
|
|
|
|
userSrvc: userService,
|
|
|
|
|
summarySrvc: summaryService,
|
|
|
|
|
heartbeatSrvc: heartbeatService,
|
|
|
|
|
keyValueSrvc: keyValueService,
|
|
|
|
|
config: conf.Get(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *MetricsHandler) RegisterRoutes(router *mux.Router) {
|
|
|
|
|
if !h.config.Security.ExposeMetrics {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logbuch.Info("exposing prometheus metrics under /api/metrics")
|
|
|
|
|
|
|
|
|
|
r := router.PathPrefix("/metrics").Subrouter()
|
|
|
|
|
r.Use(
|
|
|
|
|
middlewares.NewAuthenticateMiddleware(h.userSrvc).Handler,
|
|
|
|
|
)
|
2021-04-30 19:08:53 +03:00
|
|
|
|
r.Path("").Methods(http.MethodGet).HandlerFunc(h.Get)
|
2021-02-12 20:37:30 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *MetricsHandler) Get(w http.ResponseWriter, r *http.Request) {
|
2021-03-26 15:10:10 +03:00
|
|
|
|
reqUser := middlewares.GetPrincipal(r)
|
2021-02-13 01:06:48 +03:00
|
|
|
|
if reqUser == nil {
|
2021-02-12 20:37:30 +03:00
|
|
|
|
w.WriteHeader(http.StatusUnauthorized)
|
2021-02-13 13:23:58 +03:00
|
|
|
|
w.Write([]byte(conf.ErrUnauthorized))
|
2021-02-12 20:37:30 +03:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
var metrics mm.Metrics
|
|
|
|
|
|
|
|
|
|
if userMetrics, err := h.getUserMetrics(reqUser); err != nil {
|
2021-04-16 16:59:39 +03:00
|
|
|
|
conf.Log().Request(r).Error("%v", err)
|
2021-02-13 01:06:48 +03:00
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
w.Write([]byte(conf.ErrInternalServerError))
|
2021-02-13 13:23:58 +03:00
|
|
|
|
return
|
2021-02-13 01:06:48 +03:00
|
|
|
|
} else {
|
|
|
|
|
for _, m := range *userMetrics {
|
|
|
|
|
metrics = append(metrics, m)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if reqUser.IsAdmin {
|
|
|
|
|
if adminMetrics, err := h.getAdminMetrics(reqUser); err != nil {
|
2021-04-16 16:59:39 +03:00
|
|
|
|
conf.Log().Request(r).Error("%v", err)
|
2021-02-13 01:06:48 +03:00
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
|
w.Write([]byte(conf.ErrInternalServerError))
|
2021-02-13 13:23:58 +03:00
|
|
|
|
return
|
2021-02-13 01:06:48 +03:00
|
|
|
|
} else {
|
|
|
|
|
for _, m := range *adminMetrics {
|
|
|
|
|
metrics = append(metrics, m)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sort.Sort(metrics)
|
|
|
|
|
|
|
|
|
|
w.Header().Set("content-type", "text/plain; charset=utf-8")
|
|
|
|
|
w.Write([]byte(metrics.Print()))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (h *MetricsHandler) getUserMetrics(user *models.User) (*mm.Metrics, error) {
|
|
|
|
|
var metrics mm.Metrics
|
|
|
|
|
|
2021-03-25 01:31:04 +03:00
|
|
|
|
summaryAllTime, err := h.summarySrvc.Aliased(time.Time{}, time.Now(), user, h.summarySrvc.Retrieve, false)
|
2021-02-12 20:37:30 +03:00
|
|
|
|
if err != nil {
|
2021-02-13 01:06:48 +03:00
|
|
|
|
logbuch.Error("failed to retrieve all time summary for user '%s' for metric", user.ID)
|
|
|
|
|
return nil, err
|
2021-02-12 20:37:30 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-25 15:15:18 +03:00
|
|
|
|
from, to := utils.MustResolveIntervalRawTZ("today", user.TZ())
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-03-25 01:31:04 +03:00
|
|
|
|
summaryToday, err := h.summarySrvc.Aliased(from, to, user, h.summarySrvc.Retrieve, false)
|
2021-02-12 20:37:30 +03:00
|
|
|
|
if err != nil {
|
2021-02-13 01:06:48 +03:00
|
|
|
|
logbuch.Error("failed to retrieve today's summary for user '%s' for metric", user.ID)
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
heartbeatCount, err := h.heartbeatSrvc.CountByUser(user)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logbuch.Error("failed to count heartbeats for user '%s' for metric", user.ID)
|
|
|
|
|
return nil, err
|
2021-02-12 20:37:30 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// User Metrics
|
|
|
|
|
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_cumulative_seconds_total",
|
|
|
|
|
Desc: DescAllTime,
|
|
|
|
|
Value: int(v1.NewAllTimeFrom(summaryAllTime, &models.Filters{}).Data.TotalSeconds),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_seconds_total",
|
|
|
|
|
Desc: DescTotal,
|
|
|
|
|
Value: int(summaryToday.TotalTime().Seconds()),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_heartbeats_total",
|
|
|
|
|
Desc: DescHeartbeats,
|
|
|
|
|
Value: int(heartbeatCount),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
2021-02-12 20:37:30 +03:00
|
|
|
|
for _, p := range summaryToday.Projects {
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_project_seconds_total",
|
|
|
|
|
Desc: DescProjects,
|
|
|
|
|
Value: int(summaryToday.TotalTimeByKey(models.SummaryProject, p.Key).Seconds()),
|
|
|
|
|
Labels: []mm.Label{{Key: "name", Value: p.Key}},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, l := range summaryToday.Languages {
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_language_seconds_total",
|
|
|
|
|
Desc: DescLanguages,
|
|
|
|
|
Value: int(summaryToday.TotalTimeByKey(models.SummaryLanguage, l.Key).Seconds()),
|
|
|
|
|
Labels: []mm.Label{{Key: "name", Value: l.Key}},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, e := range summaryToday.Editors {
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_editor_seconds_total",
|
|
|
|
|
Desc: DescEditors,
|
|
|
|
|
Value: int(summaryToday.TotalTimeByKey(models.SummaryEditor, e.Key).Seconds()),
|
|
|
|
|
Labels: []mm.Label{{Key: "name", Value: e.Key}},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, o := range summaryToday.OperatingSystems {
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_operating_system_seconds_total",
|
|
|
|
|
Desc: DescOperatingSystems,
|
|
|
|
|
Value: int(summaryToday.TotalTimeByKey(models.SummaryOS, o.Key).Seconds()),
|
|
|
|
|
Labels: []mm.Label{{Key: "name", Value: o.Key}},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, m := range summaryToday.Machines {
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_machine_seconds_total",
|
|
|
|
|
Desc: DescMachines,
|
|
|
|
|
Value: int(summaryToday.TotalTimeByKey(models.SummaryMachine, m.Key).Seconds()),
|
|
|
|
|
Labels: []mm.Label{{Key: "name", Value: m.Key}},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-11 21:59:34 +03:00
|
|
|
|
for _, m := range summaryToday.Labels {
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_label_seconds_total",
|
|
|
|
|
Desc: DescLabels,
|
|
|
|
|
Value: int(summaryToday.TotalTimeByKey(models.SummaryLabel, m.Key).Seconds()),
|
|
|
|
|
Labels: []mm.Label{{Key: "name", Value: m.Key}},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-14 11:35:01 +03:00
|
|
|
|
// Runtime metrics
|
|
|
|
|
var memStats runtime.MemStats
|
|
|
|
|
runtime.ReadMemStats(&memStats)
|
|
|
|
|
|
2021-10-14 11:22:59 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_goroutines_total",
|
|
|
|
|
Desc: DescGoroutines,
|
|
|
|
|
Value: runtime.NumGoroutine(),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
2021-10-14 11:35:01 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_mem_alloc_total",
|
|
|
|
|
Desc: DescMemAllocTotal,
|
|
|
|
|
Value: int(memStats.Alloc),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_mem_sys_total",
|
|
|
|
|
Desc: DescMemSysTotal,
|
|
|
|
|
Value: int(memStats.Sys),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
return &metrics, nil
|
|
|
|
|
}
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
func (h *MetricsHandler) getAdminMetrics(user *models.User) (*mm.Metrics, error) {
|
|
|
|
|
var metrics mm.Metrics
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
if !user.IsAdmin {
|
|
|
|
|
return nil, errors.New("unauthorized")
|
|
|
|
|
}
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
var totalSeconds int
|
|
|
|
|
if t, err := h.keyValueSrvc.GetString(conf.KeyLatestTotalTime); err == nil && t != nil && t.Value != "" {
|
|
|
|
|
if d, err := time.ParseDuration(t.Value); err == nil {
|
|
|
|
|
totalSeconds = int(d.Seconds())
|
2021-02-12 20:37:30 +03:00
|
|
|
|
}
|
2021-02-13 01:06:48 +03:00
|
|
|
|
}
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
totalUsers, _ := h.userSrvc.Count()
|
2021-02-13 01:16:20 +03:00
|
|
|
|
totalHeartbeats, _ := h.heartbeatSrvc.Count()
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-06-26 13:42:51 +03:00
|
|
|
|
activeUsers, err := h.userSrvc.GetActive(false)
|
2021-02-13 01:06:48 +03:00
|
|
|
|
if err != nil {
|
2021-02-13 14:59:59 +03:00
|
|
|
|
logbuch.Error("failed to retrieve active users for metric – %v", err)
|
2021-02-13 01:06:48 +03:00
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_admin_seconds_total",
|
|
|
|
|
Desc: DescAdminTotalTime,
|
|
|
|
|
Value: totalSeconds,
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
2021-02-12 20:37:30 +03:00
|
|
|
|
|
2021-02-13 01:16:20 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_admin_heartbeats_total",
|
|
|
|
|
Desc: DescAdminTotalHeartbeats,
|
|
|
|
|
Value: int(totalHeartbeats),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_admin_users_total",
|
|
|
|
|
Desc: DescAdminTotalUsers,
|
|
|
|
|
Value: int(totalUsers),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
|
|
|
|
Name: MetricsPrefix + "_admin_users_active_total",
|
|
|
|
|
Desc: DescAdminActiveUsers,
|
|
|
|
|
Value: len(activeUsers),
|
|
|
|
|
Labels: []mm.Label{},
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Count per-user heartbeats
|
|
|
|
|
|
2021-02-13 13:23:58 +03:00
|
|
|
|
userCounts, err := h.heartbeatSrvc.CountByUsers(activeUsers)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logbuch.Error("failed to count heartbeats for active users", err.Error())
|
|
|
|
|
return nil, err
|
2021-02-13 01:06:48 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 13:23:58 +03:00
|
|
|
|
for _, uc := range userCounts {
|
2021-02-12 20:37:30 +03:00
|
|
|
|
metrics = append(metrics, &mm.CounterMetric{
|
2021-02-13 01:16:20 +03:00
|
|
|
|
Name: MetricsPrefix + "_admin_user_heartbeats_total",
|
|
|
|
|
Desc: DescAdminUserHeartbeats,
|
2021-02-13 13:23:58 +03:00
|
|
|
|
Value: int(uc.Count),
|
|
|
|
|
Labels: []mm.Label{{Key: "user", Value: uc.User}},
|
2021-02-12 20:37:30 +03:00
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 01:06:48 +03:00
|
|
|
|
return &metrics, nil
|
2021-02-12 20:37:30 +03:00
|
|
|
|
}
|