mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-05-13 01:17:42 +08:00
279 lines
9.9 KiB
Go
279 lines
9.9 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 models
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/errors"
|
|
"yunion.io/x/sqlchemy"
|
|
|
|
"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/util/rbacutils"
|
|
"yunion.io/x/onecloud/pkg/util/stringutils2"
|
|
)
|
|
|
|
var (
|
|
MonitorResourceAlertManager *SMonitorResourceAlertManager
|
|
)
|
|
|
|
func init() {
|
|
MonitorResourceAlertManager = &SMonitorResourceAlertManager{
|
|
SJointResourceBaseManager: db.NewJointResourceBaseManager(
|
|
SMonitorResourceAlert{},
|
|
"monitor_resource_alert_tbl",
|
|
"monitorresourcealert",
|
|
"monitorresourcealerts",
|
|
MonitorResourceManager, CommonAlertManager),
|
|
}
|
|
MonitorResourceAlertManager.SetVirtualObject(MonitorResourceAlertManager)
|
|
}
|
|
|
|
type SMonitorResourceAlertManager struct {
|
|
db.SJointResourceBaseManager
|
|
SMonitorScopedResourceManager
|
|
}
|
|
|
|
type SMonitorResourceAlert struct {
|
|
db.SJointResourceBase
|
|
|
|
MonitorResourceId string `width:"36" charset:"ascii" nullable:"false" list:"user" create:"required" index:"true" json:"monitor_resource_id"`
|
|
AlertId string `width:"36" charset:"ascii" list:"user" create:"required" index:"true"`
|
|
AlertRecordId string `width:"36" charset:"ascii" list:"user" update:"user"`
|
|
ResType string `width:"36" charset:"ascii" list:"user" update:"user" json:"res_type"`
|
|
AlertState string `width:"18" charset:"ascii" default:"init" list:"user" update:"user"`
|
|
SendState string `width:"18" charset:"ascii" default:"ok" list:"user" update:"user"`
|
|
TriggerTime time.Time `list:"user" update:"user" json:"trigger_time"`
|
|
Data jsonutils.JSONObject `list:"user" update:"user"`
|
|
}
|
|
|
|
func (manager *SMonitorResourceAlertManager) GetMasterFieldName() string {
|
|
return "monitor_resource_id"
|
|
}
|
|
|
|
func (manager *SMonitorResourceAlertManager) GetSlaveFieldName() string {
|
|
return "alert_id"
|
|
}
|
|
|
|
func (manager *SMonitorResourceAlertManager) DetachJoint(ctx context.Context, userCred mcclient.TokenCredential,
|
|
input monitor.MonitorResourceJointListInput) error {
|
|
joints, err := manager.GetJoinsByListInput(input)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "SMonitorResourceAlertManager DetachJoint when GetJoinsByListInput err,input:%v", input)
|
|
}
|
|
errs := make([]error, 0)
|
|
for _, joint := range joints {
|
|
err := joint.Delete(ctx, nil)
|
|
if err != nil {
|
|
errs = append(errs, errors.Wrapf(err, "joint %s:%s ,%s:%s", manager.GetMasterFieldName(),
|
|
joint.MonitorResourceId, manager.GetSlaveFieldName(), joint.AlertId))
|
|
continue
|
|
}
|
|
resources, err := MonitorResourceManager.GetMonitorResources(monitor.MonitorResourceListInput{ResId: []string{joint.
|
|
MonitorResourceId}})
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
for _, res := range resources {
|
|
err := res.UpdateAlertState()
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
}
|
|
}
|
|
return errors.NewAggregate(errs)
|
|
}
|
|
|
|
func (obj *SMonitorResourceAlert) Detach(ctx context.Context, userCred mcclient.TokenCredential) error {
|
|
return db.DetachJoint(ctx, userCred, obj)
|
|
}
|
|
|
|
func (obj *SMonitorResourceAlert) GetAlert() (*SCommonAlert, error) {
|
|
sObj, err := CommonAlertManager.FetchById(obj.AlertId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return sObj.(*SCommonAlert), nil
|
|
}
|
|
|
|
func (manager *SMonitorResourceAlertManager) GetJoinsByListInput(input monitor.MonitorResourceJointListInput) ([]SMonitorResourceAlert, error) {
|
|
joints := make([]SMonitorResourceAlert, 0)
|
|
query := manager.Query()
|
|
|
|
if len(input.MonitorResourceId) != 0 {
|
|
query.Equals(manager.GetMasterFieldName(), input.MonitorResourceId)
|
|
}
|
|
if len(input.AlertId) != 0 {
|
|
query.Equals(manager.GetSlaveFieldName(), input.AlertId)
|
|
}
|
|
if len(input.JointId) != 0 {
|
|
query.In("row_id", input.JointId)
|
|
}
|
|
err := db.FetchModelObjects(manager, query, &joints)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "FetchModelObjects by GetJoinsByMasterId:%s err", input.MonitorResourceId)
|
|
}
|
|
return joints, nil
|
|
}
|
|
|
|
func (obj *SMonitorResourceAlert) UpdateAlertRecordData(record *SAlertRecord, match *monitor.EvalMatch) error {
|
|
sendState := record.SendState
|
|
if _, ok := match.Tags[monitor.ALERT_RESOURCE_RECORD_SHIELD_KEY]; ok {
|
|
sendState = monitor.SEND_STATE_SHIELD
|
|
}
|
|
if _, err := db.Update(obj, func() error {
|
|
obj.AlertRecordId = record.GetId()
|
|
obj.ResType = record.ResType
|
|
obj.AlertState = record.State
|
|
obj.SendState = sendState
|
|
obj.TriggerTime = record.CreatedAt
|
|
obj.Data = jsonutils.Marshal(match)
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *SMonitorResourceAlertManager) ListItemFilter(ctx context.Context, q *sqlchemy.SQuery, userCred mcclient.TokenCredential, input *monitor.MonitorResourceJointListInput) (*sqlchemy.SQuery, error) {
|
|
var err error
|
|
q, err = m.SJointResourceBaseManager.ListItemFilter(ctx, q, userCred, input.JointResourceBaseListInput)
|
|
if err != nil {
|
|
return q, errors.Wrap(err, "SJointResourceBaseManager ListItemFilter err")
|
|
}
|
|
if input.Alerting {
|
|
q.Equals("alert_state", monitor.AlertStateAlerting)
|
|
resQ := MonitorResourceManager.Query("res_id")
|
|
resQ, err = MonitorResourceManager.ListItemFilter(ctx, resQ, userCred, monitor.MonitorResourceListInput{})
|
|
if err != nil {
|
|
return q, errors.Wrap(err, "Get monitor in Query err")
|
|
}
|
|
m.SMonitorScopedResourceManager.FilterByOwner(resQ, userCred, rbacutils.TRbacScope(input.Scope))
|
|
q.Filter(sqlchemy.In(q.Field("monitor_resource_id"), resQ.SubQuery()))
|
|
}
|
|
if len(input.SendState) != 0 {
|
|
q.Equals("send_state", input.SendState)
|
|
}
|
|
if len(input.ResType) != 0 {
|
|
q.Equals("res_type", input.ResType)
|
|
}
|
|
if len(input.ResName) != 0 {
|
|
resQ := MonitorResourceManager.Query("res_id")
|
|
resQ, err = MonitorResourceManager.ListItemFilter(ctx, resQ, userCred,
|
|
monitor.MonitorResourceListInput{ResName: input.ResName})
|
|
if err != nil {
|
|
return q, errors.Wrap(err, "Get monitor in Query err")
|
|
}
|
|
q.Filter(sqlchemy.In(q.Field("monitor_resource_id"), resQ.SubQuery()))
|
|
}
|
|
alertQuery := CommonAlertManager.Query("id")
|
|
m.SMonitorScopedResourceManager.FilterByOwner(alertQuery, userCred, rbacutils.TRbacScope(input.Scope))
|
|
if len(input.AlertName) != 0 {
|
|
CommonAlertManager.FieldListFilter(alertQuery, monitor.CommonAlertListInput{Name: input.AlertName})
|
|
q.Filter(sqlchemy.In(q.Field(m.GetSlaveFieldName()), alertQuery.SubQuery()))
|
|
}
|
|
if len(input.Level) != 0 {
|
|
CommonAlertManager.FieldListFilter(alertQuery, monitor.CommonAlertListInput{Level: input.Level})
|
|
q.Filter(sqlchemy.In(q.Field(m.GetSlaveFieldName()), alertQuery.SubQuery()))
|
|
}
|
|
q.Filter(sqlchemy.In(q.Field(m.GetSlaveFieldName()), alertQuery.SubQuery()))
|
|
q.Filter(sqlchemy.In(q.Field("alert_record_id"), AlertRecordManager.Query("id").SubQuery()))
|
|
return q, nil
|
|
}
|
|
|
|
func (man *SMonitorResourceAlertManager) FetchCustomizeColumns(
|
|
ctx context.Context,
|
|
userCred mcclient.TokenCredential,
|
|
query jsonutils.JSONObject,
|
|
objs []interface{},
|
|
fields stringutils2.SSortedStrings,
|
|
isList bool,
|
|
) []monitor.MonitorResourceJointDetails {
|
|
rows := make([]monitor.MonitorResourceJointDetails, len(objs))
|
|
for i := range rows {
|
|
rows[i] = monitor.MonitorResourceJointDetails{}
|
|
rows[i] = objs[i].(*SMonitorResourceAlert).getMoreDetails(rows[i])
|
|
}
|
|
return rows
|
|
}
|
|
|
|
func (obj *SMonitorResourceAlert) getMoreDetails(detail monitor.MonitorResourceJointDetails) monitor.
|
|
MonitorResourceJointDetails {
|
|
resources, err := MonitorResourceManager.GetMonitorResources(monitor.MonitorResourceListInput{ResId: []string{obj.
|
|
MonitorResourceId}})
|
|
if err != nil {
|
|
log.Errorf("getMonitorResource:%s err:%v", obj.MonitorResourceId, err)
|
|
return detail
|
|
}
|
|
if len(resources) == 0 {
|
|
return detail
|
|
}
|
|
detail.ResType = resources[0].ResType
|
|
detail.ResName = resources[0].Name
|
|
detail.ResId = resources[0].ResId
|
|
|
|
if len(obj.AlertRecordId) != 0 {
|
|
record, err := AlertRecordManager.GetAlertRecord(obj.AlertRecordId)
|
|
if err != nil {
|
|
log.Errorf("get alertRecord:%s err:%v", obj.AlertRecordId, err)
|
|
return detail
|
|
}
|
|
detail.Level = record.Level
|
|
detail.AlertRule = record.AlertRule
|
|
detail.SendState = record.SendState
|
|
detail.State = record.State
|
|
}
|
|
alert, err := CommonAlertManager.GetAlert(obj.AlertId)
|
|
if err != nil {
|
|
log.Errorf("SMonitorResourceAlert get alert by id :%s err:%v", obj.AlertId, err)
|
|
return detail
|
|
}
|
|
detail.AlertName = alert.Name
|
|
|
|
now := time.Now()
|
|
shields, err := AlertRecordShieldManager.GetRecordShields(monitor.AlertRecordShieldListInput{ResId: obj.MonitorResourceId,
|
|
AlertId: obj.AlertId, EndTime: &now})
|
|
if err != nil {
|
|
log.Errorf("SMonitorResourceAlert get GetRecordShields by resId: %s,alertId: %s, err: %v",
|
|
obj.MonitorResourceId, obj.AlertId, err)
|
|
return detail
|
|
}
|
|
if len(shields) != 0 {
|
|
detail.IsSetShield = true
|
|
}
|
|
return detail
|
|
}
|
|
|
|
func (manager *SMonitorResourceAlertManager) ResourceScope() rbacutils.TRbacScope {
|
|
return manager.SScopedResourceBaseManager.ResourceScope()
|
|
}
|
|
|
|
func (manager *SMonitorResourceAlertManager) ListItemExportKeys(ctx context.Context, q *sqlchemy.SQuery, userCred mcclient.TokenCredential, keys stringutils2.SSortedStrings) (*sqlchemy.SQuery, error) {
|
|
q, err := manager.SScopedResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SScopedResourceBaseManager.ListItemExportKeys")
|
|
}
|
|
return q, nil
|
|
}
|
|
|
|
func (m *SMonitorResourceAlertManager) FilterByOwner(q *sqlchemy.SQuery, userCred mcclient.IIdentityProvider, scope rbacutils.TRbacScope) *sqlchemy.SQuery {
|
|
return q
|
|
}
|