mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-06-22 21:33:17 +08:00
174 lines
5.0 KiB
Go
174 lines
5.0 KiB
Go
// Copyright 2019 Yunion
|
|
//
|
|
// 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 alerting
|
|
|
|
import (
|
|
"database/sql"
|
|
"time"
|
|
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/errors"
|
|
|
|
"yunion.io/x/onecloud/pkg/apis/monitor"
|
|
"yunion.io/x/onecloud/pkg/mcclient"
|
|
"yunion.io/x/onecloud/pkg/monitor/models"
|
|
"yunion.io/x/onecloud/pkg/monitor/notifydrivers"
|
|
)
|
|
|
|
type notificationService struct {
|
|
}
|
|
|
|
func newNotificationService() *notificationService {
|
|
return ¬ificationService{}
|
|
}
|
|
|
|
func (n *notificationService) SendIfNeeded(evalCtx *EvalContext) error {
|
|
notifierStates, err := n.getNeededNotifiers(evalCtx.Rule.Notifications, evalCtx)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to get alert notifiers")
|
|
}
|
|
|
|
if len(notifierStates) == 0 {
|
|
return nil
|
|
}
|
|
|
|
return n.sendNotifications(evalCtx, notifierStates)
|
|
}
|
|
|
|
type notifierState struct {
|
|
notifier Notifier
|
|
state *models.SAlertnotification
|
|
}
|
|
|
|
type notifierStateSlice []*notifierState
|
|
|
|
func (n *notificationService) sendNotification(evalCtx *EvalContext, state *notifierState) error {
|
|
if !evalCtx.IsTestRun {
|
|
if err := state.state.SetToPending(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return n.sendAndMarkAsComplete(evalCtx, state)
|
|
}
|
|
|
|
func (n *notificationService) sendAndMarkAsComplete(evalCtx *EvalContext, state *notifierState) error {
|
|
notifier := state.notifier
|
|
|
|
log.Debugf("Sending notification, type %s, id %s", notifier.GetType(), notifier.GetNotifierId())
|
|
|
|
if err := notifier.Notify(evalCtx, state.state.GetParams()); err != nil {
|
|
log.Errorf("failed to send notification %s: %v", notifier.GetNotifierId(), err)
|
|
return err
|
|
}
|
|
|
|
if evalCtx.IsTestRun {
|
|
return nil
|
|
}
|
|
|
|
return state.state.SetToCompleted()
|
|
}
|
|
|
|
func (n *notificationService) sendNotifications(evalCtx *EvalContext, states notifierStateSlice) error {
|
|
for _, state := range states {
|
|
if err := n.sendNotification(evalCtx, state); err != nil {
|
|
log.Errorf("failed to send %s notification: %v", state.notifier.GetNotifierId(), err)
|
|
if evalCtx.IsTestRun {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (n *notificationService) getNeededNotifiers(nIds []string, evalCtx *EvalContext) (notifierStateSlice, error) {
|
|
notis, err := models.NotificationManager.GetNotificationsWithDefault(nIds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var result notifierStateSlice
|
|
for _, obj := range notis {
|
|
not, err := InitNotifier(NotificationConfig{
|
|
Id: obj.GetId(),
|
|
Name: obj.GetName(),
|
|
Type: obj.Type,
|
|
Frequency: time.Duration(obj.Frequency),
|
|
SendReminder: obj.SendReminder,
|
|
DisableResolveMessage: obj.DisableResolveMessage,
|
|
Settings: obj.Settings,
|
|
})
|
|
if err != nil {
|
|
log.Errorf("Could not create notifier %s, error: %v", obj.GetId(), err)
|
|
continue
|
|
}
|
|
state, err := models.AlertNotificationManager.Get(evalCtx.Rule.Id, obj.GetId())
|
|
if err != nil {
|
|
if errors.Cause(err) == sql.ErrNoRows {
|
|
state, err = obj.AttachToAlert(evalCtx.Ctx, evalCtx.UserCred, evalCtx.Rule.Id)
|
|
if err != nil {
|
|
log.Errorf("Attach notification %s to alert %s error: %v", obj.GetName(), evalCtx.Rule.Id, err)
|
|
continue
|
|
}
|
|
} else {
|
|
log.Errorf("Get alert state: %v, alertId %s, notifierId: %s", err, evalCtx.Rule.Id, obj.GetId())
|
|
continue
|
|
}
|
|
}
|
|
|
|
if not.ShouldNotify(evalCtx.Ctx, evalCtx, state) {
|
|
result = append(result, ¬ifierState{
|
|
notifier: not,
|
|
state: state,
|
|
})
|
|
}
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
type NotifierPlugin struct {
|
|
Type string
|
|
Factory NotifierFactory
|
|
ValidateCreateData func(cred mcclient.IIdentityProvider, input monitor.NotificationCreateInput) (monitor.NotificationCreateInput, error)
|
|
}
|
|
|
|
type NotificationConfig notifydrivers.NotificationConfig
|
|
|
|
// NotifierFactory is a signature for creating notifiers
|
|
type NotifierFactory func(config NotificationConfig) (Notifier, error)
|
|
|
|
func RegisterNotifier(plug *NotifierPlugin) {
|
|
notifydrivers.RegisterNotifier(¬ifydrivers.NotifierPlugin{
|
|
Type: plug.Type,
|
|
Factory: func(cfg notifydrivers.NotificationConfig) (notifydrivers.Notifier, error) {
|
|
ret, err := plug.Factory(NotificationConfig(cfg))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return ret.(notifydrivers.Notifier), nil
|
|
},
|
|
ValidateCreateData: plug.ValidateCreateData,
|
|
})
|
|
}
|
|
|
|
// InitNotifier construct a new notifier
|
|
func InitNotifier(config NotificationConfig) (Notifier, error) {
|
|
plug, err := notifydrivers.InitNotifier(notifydrivers.NotificationConfig(config))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return plug.(Notifier), nil
|
|
}
|