mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-05-13 09:38:41 +08:00
297 lines
9.2 KiB
Go
297 lines
9.2 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"
|
|
"fmt"
|
|
"time"
|
|
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/errors"
|
|
|
|
"yunion.io/x/onecloud/pkg/apis"
|
|
"yunion.io/x/onecloud/pkg/apis/monitor"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db"
|
|
"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
|
|
}
|
|
err := state.state.UpdateSendTime()
|
|
if err != nil {
|
|
return errors.Wrap(err, "notifierState UpdateSendTime err")
|
|
}
|
|
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
|
|
shouldNotify := false
|
|
for _, obj := range notis {
|
|
not, err := InitNotifier(NotificationConfig{
|
|
Ctx: evalCtx.Ctx,
|
|
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) {
|
|
shouldNotify = true
|
|
result = append(result, ¬ifierState{
|
|
notifier: not,
|
|
state: state,
|
|
})
|
|
}
|
|
}
|
|
if shouldNotify || evalCtx.Rule.State == monitor.AlertStateAlerting {
|
|
go func() {
|
|
if err := n.createAlertRecordWhenNotify(evalCtx, shouldNotify); err != nil {
|
|
log.Errorf("createAlertRecordWhenNotify error: %v", err)
|
|
}
|
|
}()
|
|
}
|
|
if !shouldNotify && evalCtx.shouldUpdateAlertState() && evalCtx.NoDataFound {
|
|
go func() {
|
|
n.detachAlertResourceWhenNodata(evalCtx)
|
|
}()
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func (n *notificationService) createAlertRecordWhenNotify(evalCtx *EvalContext, shouldNotify bool) error {
|
|
var matches []*monitor.EvalMatch
|
|
if evalCtx.Firing {
|
|
matches = evalCtx.EvalMatches
|
|
} else {
|
|
matches = evalCtx.AlertOkEvalMatches
|
|
}
|
|
n.dealNeedShieldEvalMatchs(evalCtx, matches)
|
|
recordCreateInput := monitor.AlertRecordCreateInput{
|
|
StandaloneResourceCreateInput: apis.StandaloneResourceCreateInput{
|
|
GenerateName: evalCtx.Rule.Name,
|
|
},
|
|
AlertId: evalCtx.Rule.Id,
|
|
Level: evalCtx.Rule.Level,
|
|
State: string(evalCtx.Rule.State),
|
|
SendState: monitor.SEND_STATE_OK,
|
|
EvalData: matches,
|
|
AlertRule: newAlertRecordRule(evalCtx),
|
|
}
|
|
if !shouldNotify {
|
|
recordCreateInput.SendState = monitor.SEND_STATE_SILENT
|
|
}
|
|
recordCreateInput.ResType = recordCreateInput.AlertRule.ResType
|
|
if len(recordCreateInput.ResType) == 0 {
|
|
recordCreateInput.ResType = monitor.METRIC_RES_TYPE_HOST
|
|
}
|
|
createData := recordCreateInput.JSON(recordCreateInput)
|
|
alert, _ := models.CommonAlertManager.GetAlert(evalCtx.Rule.Id)
|
|
record, err := db.DoCreate(models.AlertRecordManager, evalCtx.Ctx, evalCtx.UserCred, jsonutils.NewDict(),
|
|
createData, evalCtx.UserCred)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "db.DoCreate")
|
|
}
|
|
alertData := jsonutils.Marshal(alert)
|
|
alertData.(*jsonutils.JSONDict).Set("project_id", jsonutils.NewString(alert.GetProjectId()))
|
|
db.PerformSetScope(evalCtx.Ctx, record.(*models.SAlertRecord), evalCtx.UserCred, alertData)
|
|
dbMatches, _ := record.(*models.SAlertRecord).GetEvalData()
|
|
if !evalCtx.Firing {
|
|
evalCtx.AlertOkEvalMatches = make([]*monitor.EvalMatch, len(dbMatches))
|
|
for i := range dbMatches {
|
|
evalCtx.AlertOkEvalMatches[i] = &dbMatches[i]
|
|
}
|
|
}
|
|
record.PostCreate(evalCtx.Ctx, evalCtx.UserCred, evalCtx.UserCred, nil, createData)
|
|
return nil
|
|
}
|
|
|
|
func (n *notificationService) dealNeedShieldEvalMatchs(evalCtx *EvalContext, match []*monitor.EvalMatch) {
|
|
input := monitor.AlertRecordShieldListInput{
|
|
ResType: evalCtx.Rule.RuleDescription[0].ResType,
|
|
AlertId: evalCtx.Rule.Id,
|
|
}
|
|
filterMatch:
|
|
for i := range match {
|
|
input.ResId = match[i].Tags[monitor.MEASUREMENT_TAG_ID[input.ResType]]
|
|
alertRecordShields, err := models.AlertRecordShieldManager.GetRecordShields(input)
|
|
if err != nil {
|
|
log.Errorf("GetRecordShields byAlertId:%s,err:%v", input.AlertId, err)
|
|
return
|
|
}
|
|
if len(alertRecordShields) != 0 {
|
|
for _, shield := range alertRecordShields {
|
|
if shield.EndTime.After(time.Now().UTC()) && shield.StartTime.Before(time.Now().UTC()) {
|
|
match[i].Tags[monitor.ALERT_RESOURCE_RECORD_SHIELD_KEY] = monitor.ALERT_RESOURCE_RECORD_SHIELD_VALUE
|
|
continue filterMatch
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func (n *notificationService) detachAlertResourceWhenNodata(evalCtx *EvalContext) {
|
|
errs := models.CommonAlertManager.DetachAlertResourceByAlertId(evalCtx.Ctx, evalCtx.UserCred, evalCtx.Rule.Id)
|
|
if len(errs) != 0 {
|
|
log.Errorf("detachAlertResourceWhenNodata err:%#v", errors.NewAggregate(errs))
|
|
}
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
func newAlertRecordRule(evalCtx *EvalContext) monitor.AlertRecordRule {
|
|
alertRule := monitor.AlertRecordRule{}
|
|
if len(evalCtx.Rule.RuleDescription) != 0 {
|
|
alertRule = evalCtx.Rule.RuleDescription[0].AlertRecordRule
|
|
}
|
|
if evalCtx.Rule.Frequency < 60 {
|
|
alertRule.Period = fmt.Sprintf("%ds", evalCtx.Rule.Frequency)
|
|
} else {
|
|
alertRule.Period = fmt.Sprintf("%dm", evalCtx.Rule.Frequency/60)
|
|
}
|
|
|
|
alertRule.AlertDuration = int64(evalCtx.Rule.For) / evalCtx.Rule.Frequency
|
|
if alertRule.AlertDuration == 0 {
|
|
alertRule.AlertDuration = 1
|
|
}
|
|
|
|
if evalCtx.Rule.SilentPeriod != 0 {
|
|
alertRule.SilentPeriod = fmt.Sprintf("%dm", evalCtx.Rule.SilentPeriod/60)
|
|
}
|
|
return alertRule
|
|
}
|