alertmanager/notify/util.go

307 lines
8.2 KiB
Go

// Copyright 2019 Prometheus Team
// 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 notify
import (
"context"
"crypto/sha256"
"errors"
"fmt"
"io"
"log/slog"
"net/http"
"net/url"
"strings"
"github.com/prometheus/common/version"
"github.com/prometheus/alertmanager/template"
"github.com/prometheus/alertmanager/types"
)
// truncationMarker is the character used to represent a truncation.
const truncationMarker = "…"
// UserAgentHeader is the default User-Agent for notification requests.
var UserAgentHeader = fmt.Sprintf("Alertmanager/%s", version.Version)
// RedactURL removes the URL part from an error of *url.Error type.
func RedactURL(err error) error {
var e *url.Error
if !errors.As(err, &e) {
return err
}
e.URL = "<redacted>"
return e
}
// Get sends a GET request to the given URL.
func Get(ctx context.Context, client *http.Client, url string) (*http.Response, error) {
return request(ctx, client, http.MethodGet, url, "", nil)
}
// PostJSON sends a POST request with JSON payload to the given URL.
func PostJSON(ctx context.Context, client *http.Client, url string, body io.Reader) (*http.Response, error) {
return post(ctx, client, url, "application/json", body)
}
// PostText sends a POST request with text payload to the given URL.
func PostText(ctx context.Context, client *http.Client, url string, body io.Reader) (*http.Response, error) {
return post(ctx, client, url, "text/plain", body)
}
func post(ctx context.Context, client *http.Client, url, bodyType string, body io.Reader) (*http.Response, error) {
return request(ctx, client, http.MethodPost, url, bodyType, body)
}
func request(ctx context.Context, client *http.Client, method, url, bodyType string, body io.Reader) (*http.Response, error) {
req, err := http.NewRequest(method, url, body)
if err != nil {
return nil, err
}
req.Header.Set("User-Agent", UserAgentHeader)
if bodyType != "" {
req.Header.Set("Content-Type", bodyType)
}
return client.Do(req.WithContext(ctx))
}
// Drain consumes and closes the response's body to make sure that the
// HTTP client can reuse existing connections.
func Drain(r *http.Response) {
io.Copy(io.Discard, r.Body)
r.Body.Close()
}
// TruncateInRunes truncates a string to fit the given size in Runes.
func TruncateInRunes(s string, n int) (string, bool) {
r := []rune(s)
if len(r) <= n {
return s, false
}
if n <= 3 {
return string(r[:n]), true
}
return string(r[:n-1]) + truncationMarker, true
}
// TruncateInBytes truncates a string to fit the given size in Bytes.
func TruncateInBytes(s string, n int) (string, bool) {
// First, measure the string the w/o a to-rune conversion.
if len(s) <= n {
return s, false
}
// The truncationMarker itself is 3 bytes, we can't return any part of the string when it's less than 3.
if n <= 3 {
switch n {
case 3:
return truncationMarker, true
default:
return strings.Repeat(".", n), true
}
}
// Now, to ensure we don't butcher the string we need to remove using runes.
r := []rune(s)
truncationTarget := n - 3
// Next, let's truncate the runes to the lower possible number.
truncatedRunes := r[:truncationTarget]
for len(string(truncatedRunes)) > truncationTarget {
truncatedRunes = r[:len(truncatedRunes)-1]
}
return string(truncatedRunes) + truncationMarker, true
}
// TmplText is using monadic error handling in order to make string templating
// less verbose. Use with care as the final error checking is easily missed.
func TmplText(tmpl *template.Template, data *template.Data, err *error) func(string) string {
return func(name string) (s string) {
if *err != nil {
return
}
s, *err = tmpl.ExecuteTextString(name, data)
return s
}
}
// TmplHTML is using monadic error handling in order to make string templating
// less verbose. Use with care as the final error checking is easily missed.
func TmplHTML(tmpl *template.Template, data *template.Data, err *error) func(string) string {
return func(name string) (s string) {
if *err != nil {
return
}
s, *err = tmpl.ExecuteHTMLString(name, data)
return s
}
}
// Key is a string that can be hashed.
type Key string
// ExtractGroupKey gets the group key from the context.
func ExtractGroupKey(ctx context.Context) (Key, error) {
key, ok := GroupKey(ctx)
if !ok {
return "", fmt.Errorf("group key missing")
}
return Key(key), nil
}
// Hash returns the sha256 for a group key as integrations may have
// maximum length requirements on deduplication keys.
func (k Key) Hash() string {
h := sha256.New()
// hash.Hash.Write never returns an error.
//nolint: errcheck
h.Write([]byte(string(k)))
return fmt.Sprintf("%x", h.Sum(nil))
}
func (k Key) String() string {
return string(k)
}
// GetTemplateData creates the template data from the context and the alerts.
func GetTemplateData(ctx context.Context, tmpl *template.Template, alerts []*types.Alert, l *slog.Logger) *template.Data {
recv, ok := ReceiverName(ctx)
if !ok {
l.Error("Missing receiver")
}
groupLabels, ok := GroupLabels(ctx)
if !ok {
l.Error("Missing group labels")
}
return tmpl.Data(recv, groupLabels, alerts...)
}
func readAll(r io.Reader) string {
if r == nil {
return ""
}
bs, err := io.ReadAll(r)
if err != nil {
return ""
}
return string(bs)
}
// Retrier knows when to retry an HTTP request to a receiver. 2xx status codes
// are successful, anything else is a failure and only 5xx status codes should
// be retried.
type Retrier struct {
// Function to return additional information in the error message.
CustomDetailsFunc func(code int, body io.Reader) string
// Additional HTTP status codes that should be retried.
RetryCodes []int
}
// Check returns a boolean indicating whether the request should be retried
// and an optional error if the request has failed. If body is not nil, it will
// be included in the error message.
func (r *Retrier) Check(statusCode int, body io.Reader) (bool, error) {
// 2xx responses are considered to be always successful.
if statusCode/100 == 2 {
return false, nil
}
// 5xx responses are considered to be always retried.
retry := statusCode/100 == 5
if !retry {
for _, code := range r.RetryCodes {
if code == statusCode {
retry = true
break
}
}
}
s := fmt.Sprintf("unexpected status code %v", statusCode)
var details string
if r.CustomDetailsFunc != nil {
details = r.CustomDetailsFunc(statusCode, body)
} else {
details = readAll(body)
}
if details != "" {
s = fmt.Sprintf("%s: %s", s, details)
}
return retry, errors.New(s)
}
type ErrorWithReason struct {
Err error
Reason Reason
}
func NewErrorWithReason(reason Reason, err error) *ErrorWithReason {
return &ErrorWithReason{
Err: err,
Reason: reason,
}
}
func (e *ErrorWithReason) Error() string {
return e.Err.Error()
}
// Reason is the failure reason.
type Reason int
const (
DefaultReason Reason = iota
ClientErrorReason
ServerErrorReason
ContextCanceledReason
ContextDeadlineExceededReason
)
func (s Reason) String() string {
switch s {
case DefaultReason:
return "other"
case ClientErrorReason:
return "clientError"
case ServerErrorReason:
return "serverError"
case ContextCanceledReason:
return "contextCanceled"
case ContextDeadlineExceededReason:
return "contextDeadlineExceeded"
default:
panic(fmt.Sprintf("unknown Reason: %d", s))
}
}
// possibleFailureReasonCategory is a list of possible failure reason.
var possibleFailureReasonCategory = []string{DefaultReason.String(), ClientErrorReason.String(), ServerErrorReason.String(), ContextCanceledReason.String(), ContextDeadlineExceededReason.String()}
// GetFailureReasonFromStatusCode returns the reason for the failure based on the status code provided.
func GetFailureReasonFromStatusCode(statusCode int) Reason {
if statusCode/100 == 4 {
return ClientErrorReason
}
if statusCode/100 == 5 {
return ServerErrorReason
}
return DefaultReason
}