mirror of
https://github.com/prometheus/prometheus.git
synced 2025-03-05 20:59:13 -08:00
web/api: Expose rule health and last error (#4501)
Expose rule health and last evaluation error on `/api/v1/rules`. Signed-off-by: Max Leonard Inden <IndenML@gmail.com>
This commit is contained in:
parent
12fe204ea6
commit
ecf676cf97
|
@ -406,6 +406,7 @@ $ curl http://localhost:9090/api/v1/rules
|
||||||
"summary": "High request latency"
|
"summary": "High request latency"
|
||||||
},
|
},
|
||||||
"duration": 600,
|
"duration": 600,
|
||||||
|
"health": "ok",
|
||||||
"labels": {
|
"labels": {
|
||||||
"severity": "page"
|
"severity": "page"
|
||||||
},
|
},
|
||||||
|
@ -414,6 +415,7 @@ $ curl http://localhost:9090/api/v1/rules
|
||||||
"type": "alerting"
|
"type": "alerting"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"health": "ok",
|
||||||
"name": "job:http_inprogress_requests:sum",
|
"name": "job:http_inprogress_requests:sum",
|
||||||
"query": "sum(http_inprogress_requests) by (job)",
|
"query": "sum(http_inprogress_requests) by (job)",
|
||||||
"type": "recording"
|
"type": "recording"
|
||||||
|
|
|
@ -655,20 +655,24 @@ type RuleGroup struct {
|
||||||
type rule interface{}
|
type rule interface{}
|
||||||
|
|
||||||
type alertingRule struct {
|
type alertingRule struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Query string `json:"query"`
|
Query string `json:"query"`
|
||||||
Duration float64 `json:"duration"`
|
Duration float64 `json:"duration"`
|
||||||
Labels labels.Labels `json:"labels"`
|
Labels labels.Labels `json:"labels"`
|
||||||
Annotations labels.Labels `json:"annotations"`
|
Annotations labels.Labels `json:"annotations"`
|
||||||
Alerts []*Alert `json:"alerts"`
|
Alerts []*Alert `json:"alerts"`
|
||||||
|
Health rules.RuleHealth `json:"health"`
|
||||||
|
LastError string `json:"lastError,omitempty"`
|
||||||
// Type of an alertingRule is always "alerting".
|
// Type of an alertingRule is always "alerting".
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type recordingRule struct {
|
type recordingRule struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Query string `json:"query"`
|
Query string `json:"query"`
|
||||||
Labels labels.Labels `json:"labels,omitempty"`
|
Labels labels.Labels `json:"labels,omitempty"`
|
||||||
|
Health rules.RuleHealth `json:"health"`
|
||||||
|
LastError string `json:"lastError,omitempty"`
|
||||||
// Type of a recordingRule is always "recording".
|
// Type of a recordingRule is always "recording".
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
}
|
}
|
||||||
|
@ -687,6 +691,11 @@ func (api *API) rules(r *http.Request) (interface{}, *apiError, func()) {
|
||||||
for _, r := range grp.Rules() {
|
for _, r := range grp.Rules() {
|
||||||
var enrichedRule rule
|
var enrichedRule rule
|
||||||
|
|
||||||
|
lastError := ""
|
||||||
|
if r.LastError() != nil {
|
||||||
|
lastError = r.LastError().Error()
|
||||||
|
}
|
||||||
|
|
||||||
switch rule := r.(type) {
|
switch rule := r.(type) {
|
||||||
case *rules.AlertingRule:
|
case *rules.AlertingRule:
|
||||||
enrichedRule = alertingRule{
|
enrichedRule = alertingRule{
|
||||||
|
@ -696,14 +705,18 @@ func (api *API) rules(r *http.Request) (interface{}, *apiError, func()) {
|
||||||
Labels: rule.Labels(),
|
Labels: rule.Labels(),
|
||||||
Annotations: rule.Annotations(),
|
Annotations: rule.Annotations(),
|
||||||
Alerts: rulesAlertsToAPIAlerts(rule.ActiveAlerts()),
|
Alerts: rulesAlertsToAPIAlerts(rule.ActiveAlerts()),
|
||||||
|
Health: rule.Health(),
|
||||||
|
LastError: lastError,
|
||||||
Type: "alerting",
|
Type: "alerting",
|
||||||
}
|
}
|
||||||
case *rules.RecordingRule:
|
case *rules.RecordingRule:
|
||||||
enrichedRule = recordingRule{
|
enrichedRule = recordingRule{
|
||||||
Name: rule.Name(),
|
Name: rule.Name(),
|
||||||
Query: rule.Query().String(),
|
Query: rule.Query().String(),
|
||||||
Labels: rule.Labels(),
|
Labels: rule.Labels(),
|
||||||
Type: "recording",
|
Health: rule.Health(),
|
||||||
|
LastError: lastError,
|
||||||
|
Type: "recording",
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
err := fmt.Errorf("failed to assert type of rule '%v'", rule.Name())
|
err := fmt.Errorf("failed to assert type of rule '%v'", rule.Name())
|
||||||
|
|
|
@ -681,6 +681,7 @@ func testEndpoints(t *testing.T, api *API, testLabelAPI bool) {
|
||||||
Labels: labels.Labels{},
|
Labels: labels.Labels{},
|
||||||
Annotations: labels.Labels{},
|
Annotations: labels.Labels{},
|
||||||
Alerts: []*Alert{},
|
Alerts: []*Alert{},
|
||||||
|
Health: "unknown",
|
||||||
Type: "alerting",
|
Type: "alerting",
|
||||||
},
|
},
|
||||||
alertingRule{
|
alertingRule{
|
||||||
|
@ -690,12 +691,14 @@ func testEndpoints(t *testing.T, api *API, testLabelAPI bool) {
|
||||||
Labels: labels.Labels{},
|
Labels: labels.Labels{},
|
||||||
Annotations: labels.Labels{},
|
Annotations: labels.Labels{},
|
||||||
Alerts: []*Alert{},
|
Alerts: []*Alert{},
|
||||||
|
Health: "unknown",
|
||||||
Type: "alerting",
|
Type: "alerting",
|
||||||
},
|
},
|
||||||
recordingRule{
|
recordingRule{
|
||||||
Name: "recording-rule-1",
|
Name: "recording-rule-1",
|
||||||
Query: "vector(1)",
|
Query: "vector(1)",
|
||||||
Labels: labels.Labels{},
|
Labels: labels.Labels{},
|
||||||
|
Health: "unknown",
|
||||||
Type: "recording",
|
Type: "recording",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in a new issue