mirror of
https://git.mirrors.martin98.com/https://github.com/SigNoz/signoz
synced 2025-07-27 17:21:59 +08:00

### Summary Integrate the new implementations of the alertmanager along with changes to the ruler. This change can be broadly categoried into 3 parts: #### Frontend - The earlier `/api/v1/alerts` api was double encoding the response in json and sending it to the frontend. This PR fixes the json response object. For instance, we have gone from the response `{ "status": "success", "data": "{\"status\":\"success\",\"data\":[{\"labels\":{\"alertname\":\"[platform][consumer] consumer is above 100% memory utilization\",\"bu\":\"platform\",\"...... }` to the response `{"status":"success","data":[{"labels":{"alertname":"[Metrics] Pod CP......` - `msteams` has been changed to `msteamsv2` wherever applicable #### Ruler The following changes have been done in the ruler component: - Removal of the old alertmanager and notifier - The RuleDB methods `Create`, `Edit` and `Delete` have been made transactional - Introduction of a new `testPrepareNotifyFunc` for sending test notifications - Integration with the new alertmanager #### Alertmanager Although a huge chunk of the alertmanagers have been merged in previous PRs (the list can be found at https://github.com/SigNoz/platform-pod/issues/404), this PR takes care of changes needed in order to incorporate it with the ruler - Addition of ruleId based matching - Support for marshalling the global configuration directly from the upstream alertmanager - Addition of orgId to the legacy alertmanager - Support for always adding defaults to both routes and receivers while creating them - Migration to create the required alertmanager tables - Migration for msteams to msteamsv2 has been added. We will start using msteamv2 config for the new alertmanager and keep using msteams for the old one. #### Related Issues / PR's Closes https://github.com/SigNoz/platform-pod/issues/404 Closes https://github.com/SigNoz/platform-pod/issues/176
154 lines
3.4 KiB
Go
154 lines
3.4 KiB
Go
package config
|
|
|
|
import (
|
|
"net/url"
|
|
"reflect"
|
|
|
|
"github.com/go-viper/mapstructure/v2"
|
|
"github.com/knadh/koanf/providers/confmap"
|
|
"github.com/knadh/koanf/v2"
|
|
yamlv2 "gopkg.in/yaml.v2"
|
|
)
|
|
|
|
const (
|
|
KoanfDelimiter string = "::"
|
|
)
|
|
|
|
// Conf is a wrapper around the koanf library.
|
|
type Conf struct {
|
|
*koanf.Koanf
|
|
}
|
|
|
|
// NewConf creates a new Conf instance.
|
|
func NewConf() *Conf {
|
|
return &Conf{koanf.New(KoanfDelimiter)}
|
|
}
|
|
|
|
// NewConfFromMap creates a new Conf instance from a map.
|
|
func NewConfFromMap(m map[string]any) (*Conf, error) {
|
|
conf := NewConf()
|
|
if err := conf.Koanf.Load(confmap.Provider(m, KoanfDelimiter), nil); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return conf, nil
|
|
}
|
|
|
|
// MustNewConfFromMap creates a new Conf instance from a map.
|
|
// It panics if the conf cannot be created.
|
|
func MustNewConfFromMap(m map[string]any) *Conf {
|
|
conf, err := NewConfFromMap(m)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return conf
|
|
}
|
|
|
|
// Merge merges the current configuration with the input configuration.
|
|
func (conf *Conf) Merge(input *Conf) error {
|
|
return conf.Koanf.Merge(input.Koanf)
|
|
}
|
|
|
|
// Merge merges the current configuration with the input configuration.
|
|
func (conf *Conf) MergeAt(input *Conf, path string) error {
|
|
return conf.Koanf.MergeAt(input.Koanf, path)
|
|
}
|
|
|
|
// Unmarshal unmarshals the configuration at the given path into the input.
|
|
// It uses a WeaklyTypedInput to allow for more flexible unmarshalling.
|
|
func (conf *Conf) Unmarshal(path string, input any, tags ...string) error {
|
|
tags = append([]string{"mapstructure"}, tags...)
|
|
|
|
for _, tag := range tags {
|
|
dc := &mapstructure.DecoderConfig{
|
|
TagName: tag,
|
|
WeaklyTypedInput: true,
|
|
DecodeHook: mapstructure.ComposeDecodeHookFunc(
|
|
mapstructure.StringToSliceHookFunc(","),
|
|
mapstructure.StringToTimeDurationHookFunc(),
|
|
mapstructure.TextUnmarshallerHookFunc(),
|
|
StringToURLHookFunc(),
|
|
YamlV2UnmarshalHookFunc(),
|
|
),
|
|
Result: input,
|
|
}
|
|
|
|
err := conf.Koanf.UnmarshalWithConf(path, input, koanf.UnmarshalConf{Tag: tag, DecoderConfig: dc})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Set sets the configuration at the given key.
|
|
// It decodes the input into a map as per mapstructure.Decode and then merges it into the configuration.
|
|
func (conf *Conf) Set(key string, input any) error {
|
|
m := map[string]any{}
|
|
err := mapstructure.Decode(input, &m)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
newConf := NewConf()
|
|
if err := newConf.Koanf.Load(confmap.Provider(m, KoanfDelimiter), nil); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := conf.Koanf.MergeAt(newConf.Koanf, key); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func StringToURLHookFunc() mapstructure.DecodeHookFunc {
|
|
return func(
|
|
f reflect.Type,
|
|
t reflect.Type,
|
|
data interface{},
|
|
) (interface{}, error) {
|
|
if f.Kind() != reflect.String {
|
|
return data, nil
|
|
}
|
|
if t != reflect.TypeOf(url.URL{}) {
|
|
return data, nil
|
|
}
|
|
|
|
// Convert it by parsing
|
|
u, err := url.Parse(data.(string))
|
|
return u, err
|
|
}
|
|
}
|
|
|
|
func YamlV2UnmarshalHookFunc() mapstructure.DecodeHookFunc {
|
|
return func(
|
|
f reflect.Type,
|
|
t reflect.Type,
|
|
data interface{},
|
|
) (interface{}, error) {
|
|
if f.Kind() != reflect.String {
|
|
return data, nil
|
|
}
|
|
result := reflect.New(t).Interface()
|
|
_, ok := result.(yamlv2.Unmarshaler)
|
|
if !ok {
|
|
return data, nil
|
|
}
|
|
|
|
str, ok := data.(string)
|
|
if !ok {
|
|
str = reflect.Indirect(reflect.ValueOf(&data)).Elem().String()
|
|
}
|
|
|
|
if err := yamlv2.Unmarshal([]byte(str), result); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
}
|