mirror of
https://git.mirrors.martin98.com/https://github.com/SigNoz/signoz
synced 2025-07-09 03:01:48 +08:00
201 lines
5.7 KiB
Go
201 lines
5.7 KiB
Go
package alertManager
|
|
|
|
// Wrapper to connect and process alert manager functions
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
neturl "net/url"
|
|
|
|
"go.signoz.io/signoz/pkg/query-service/constants"
|
|
"go.signoz.io/signoz/pkg/query-service/model"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
const contentType = "application/json"
|
|
|
|
type Manager interface {
|
|
URL() *neturl.URL
|
|
URLPath(path string) *neturl.URL
|
|
AddRoute(receiver *Receiver) *model.ApiError
|
|
EditRoute(receiver *Receiver) *model.ApiError
|
|
DeleteRoute(name string) *model.ApiError
|
|
TestReceiver(receiver *Receiver) *model.ApiError
|
|
}
|
|
|
|
func defaultOptions() []ManagerOptions {
|
|
return []ManagerOptions{
|
|
WithURL(constants.GetAlertManagerApiPrefix()),
|
|
WithChannelApiPath(constants.AmChannelApiPath),
|
|
}
|
|
}
|
|
|
|
type ManagerOptions func(m *manager) error
|
|
|
|
func New(opts ...ManagerOptions) (Manager, error) {
|
|
m := &manager{}
|
|
|
|
newOpts := defaultOptions()
|
|
newOpts = append(newOpts, opts...)
|
|
|
|
for _, opt := range newOpts {
|
|
err := opt(m)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
|
|
func WithURL(url string) ManagerOptions {
|
|
return func(m *manager) error {
|
|
m.url = url
|
|
parsedURL, err := neturl.Parse(url)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
m.parsedURL = parsedURL
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func WithChannelApiPath(path string) ManagerOptions {
|
|
return func(m *manager) error {
|
|
m.channelApiPath = path
|
|
return nil
|
|
}
|
|
}
|
|
|
|
type manager struct {
|
|
url string
|
|
parsedURL *neturl.URL
|
|
channelApiPath string
|
|
}
|
|
|
|
func (m *manager) prepareAmChannelApiURL() string {
|
|
return fmt.Sprintf("%s%s", m.url, m.channelApiPath)
|
|
}
|
|
|
|
func (m *manager) prepareTestApiURL() string {
|
|
return fmt.Sprintf("%s%s", m.url, "v1/testReceiver")
|
|
}
|
|
|
|
func (m *manager) URL() *neturl.URL {
|
|
return m.parsedURL
|
|
}
|
|
|
|
func (m *manager) URLPath(path string) *neturl.URL {
|
|
upath, err := neturl.Parse(path)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
|
|
return m.parsedURL.ResolveReference(upath)
|
|
}
|
|
|
|
func (m *manager) AddRoute(receiver *Receiver) *model.ApiError {
|
|
|
|
receiverString, _ := json.Marshal(receiver)
|
|
|
|
amURL := m.prepareAmChannelApiURL()
|
|
response, err := http.Post(amURL, contentType, bytes.NewBuffer(receiverString))
|
|
|
|
if err != nil {
|
|
zap.L().Error("Error in getting response of API call to alertmanager", zap.String("url", amURL), zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
if response.StatusCode > 299 {
|
|
zap.L().Error("Error in getting 2xx response in API call to alertmanager", zap.String("url", amURL), zap.String("status", response.Status))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *manager) EditRoute(receiver *Receiver) *model.ApiError {
|
|
receiverString, _ := json.Marshal(receiver)
|
|
|
|
amURL := m.prepareAmChannelApiURL()
|
|
req, err := http.NewRequest(http.MethodPut, amURL, bytes.NewBuffer(receiverString))
|
|
|
|
if err != nil {
|
|
zap.L().Error("Error creating new update request for API call to alertmanager", zap.String("url", amURL), zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
req.Header.Add("Content-Type", contentType)
|
|
|
|
client := &http.Client{}
|
|
response, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
zap.L().Error("Error in getting response of API call to alertmanager", zap.String("url", amURL), zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
if response.StatusCode > 299 {
|
|
zap.L().Error("Error in getting 2xx response in PUT API call to alertmanager", zap.String("url", amURL), zap.String("status", response.Status))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *manager) DeleteRoute(name string) *model.ApiError {
|
|
values := map[string]string{"name": name}
|
|
requestData, _ := json.Marshal(values)
|
|
|
|
amURL := m.prepareAmChannelApiURL()
|
|
req, err := http.NewRequest(http.MethodDelete, amURL, bytes.NewBuffer(requestData))
|
|
|
|
if err != nil {
|
|
zap.L().Error("Error in creating new delete request to alertmanager/v1/receivers", zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
req.Header.Add("Content-Type", contentType)
|
|
|
|
client := &http.Client{}
|
|
response, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
zap.L().Error("Error in getting response of API call to alertmanager", zap.String("url", amURL), zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
if response.StatusCode > 299 {
|
|
err := fmt.Errorf(fmt.Sprintf("Error in getting 2xx response in PUT API call to alertmanager(DELETE %s)\n", amURL), response.Status)
|
|
zap.L().Error("Error in getting 2xx response in PUT API call to alertmanager", zap.String("url", amURL), zap.String("status", response.Status))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *manager) TestReceiver(receiver *Receiver) *model.ApiError {
|
|
|
|
receiverBytes, _ := json.Marshal(receiver)
|
|
|
|
amTestURL := m.prepareTestApiURL()
|
|
response, err := http.Post(amTestURL, contentType, bytes.NewBuffer(receiverBytes))
|
|
|
|
if err != nil {
|
|
zap.L().Error("Error in getting response of API call to alertmanager", zap.String("url", amTestURL), zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
if response.StatusCode > 201 && response.StatusCode < 400 {
|
|
err := fmt.Errorf(fmt.Sprintf("Invalid parameters in test alert api for alertmanager(POST %s)\n", amTestURL), response.Status)
|
|
zap.L().Error("Invalid parameters in test alert api for alertmanager", zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
if response.StatusCode > 400 {
|
|
err := fmt.Errorf(fmt.Sprintf("Received Server Error response for API call to alertmanager(POST %s)\n", amTestURL), response.Status)
|
|
zap.L().Error("Received Server Error response for API call to alertmanager", zap.Error(err))
|
|
return &model.ApiError{Typ: model.ErrorInternal, Err: err}
|
|
}
|
|
|
|
return nil
|
|
}
|