Files
cloudpods/pkg/monitor/validators/validators.go
zhaoxiangchun 4888f7d819 feat(monitor): add nodata alert policy
1.add queryConditon子类:nodataQueryCondition处理nodata类型策略
2.遍历Onecloud类型宿主机,得到nodata宿主机信息
2020-11-23 10:10:14 +08:00

237 lines
6.4 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 validators
import (
"strings"
"time"
"yunion.io/x/pkg/errors"
"yunion.io/x/pkg/utils"
"yunion.io/x/onecloud/pkg/apis/monitor"
"yunion.io/x/onecloud/pkg/httperrors"
merrors "yunion.io/x/onecloud/pkg/monitor/errors"
)
const (
ErrMissingParameterThreshold = errors.Error("Condition is missing the threshold parameter")
ErrMissingParameterType = errors.Error("Condition is missing the type parameter")
ErrInvalidEvaluatorType = errors.Error("Invalid condition evaluator type")
ErrAlertConditionUnknown = errors.Error("Unknown alert condition")
ErrAlertConditionEmpty = errors.Error("Alert is missing conditions")
)
var (
EvaluatorDefaultTypes = []string{"gt", "lt", "eq"}
EvaluatorRangedTypes = []string{"within_range", "outside_range"}
CommonAlertType = []string{monitor.CommonAlertNomalAlertType, monitor.CommonAlertSystemAlertType}
CommonAlertReducerFieldOpts = []string{"/"}
CommonAlertNotifyTypes = []string{"email", "mobile", "dingtalk", "webconsole", "feishu"}
ConditionTypes = []string{"query", "nodata_query"}
)
func ValidateAlertCreateInput(input monitor.AlertCreateInput) error {
if len(input.Settings.Conditions) == 0 {
return httperrors.NewInputParameterError("input condition is empty")
}
for _, condition := range input.Settings.Conditions {
if err := ValidateAlertCondition(condition); err != nil {
return err
}
}
return nil
}
func ValidateAlertCondition(input monitor.AlertCondition) error {
condType := input.Type
if err := ValidateAlertConditionType(condType); err != nil {
return err
}
if err := ValidateAlertConditionQuery(input.Query); err != nil {
return err
}
if err := ValidateAlertConditionReducer(input.Reducer); err != nil {
return err
}
if err := ValidateAlertConditionEvaluator(input.Evaluator); err != nil {
return err
}
if input.Operator == "" {
input.Operator = "and"
}
if !utils.IsInStringArray(input.Operator, []string{"and", "or"}) {
return httperrors.NewInputParameterError("Unkown operator %s", input.Operator)
}
return nil
}
func ValidateAlertConditionQuery(input monitor.AlertQuery) error {
if err := ValidateFromValue(input.From); err != nil {
return err
}
if err := ValidateToValue(input.To); err != nil {
return err
}
if err := ValidateAlertQueryModel(input.Model); err != nil {
return err
}
return nil
}
func ValidateAlertQueryModel(input monitor.MetricQuery) error {
if len(input.Selects) == 0 {
return merrors.NewArgIsEmptyErr("select")
}
if len(input.Database) == 0 {
return merrors.NewArgIsEmptyErr("database")
}
if len(input.Measurement) == 0 {
return merrors.NewArgIsEmptyErr("measurement")
}
return nil
}
func ValidateSelectOfMetricQuery(input monitor.AlertQuery) error {
if err := ValidateFromAndToValue(input); err != nil {
return err
}
if err := ValidateAlertQueryModel(input.Model); err != nil {
return err
}
for _, sel := range input.Model.Selects {
if len(sel) == 0 {
return httperrors.NewInputParameterError("select for nothing in query")
}
}
return nil
}
func ValidateFromAndToValue(input monitor.AlertQuery) error {
fromRaw := strings.Replace(input.From, "now-", "", 1)
fromDur, err := time.ParseDuration("-" + fromRaw)
if err != nil {
return err
}
if input.To == "now" {
return nil
} else if strings.HasPrefix(input.To, "now-") {
withoutNow := strings.Replace(input.To, "now-", "", 1)
toDur, err := time.ParseDuration("-" + withoutNow)
if err == nil {
if toDur >= fromDur {
return nil
}
return httperrors.NewInputParameterError("query duration err: from: %s, to:%s", input.From, input.To)
}
return err
}
return httperrors.NewInputParameterError("query duration `to` err: %s", input.To)
}
func ValidateAlertConditionReducer(input monitor.Condition) error {
return nil
}
func ValidateAlertConditionEvaluator(input monitor.Condition) error {
typ := input.Type
if typ == "" {
return ErrMissingParameterType
}
if utils.IsInStringArray(typ, EvaluatorDefaultTypes) {
return ValidateAlertConditionThresholdEvaluator(input)
}
if utils.IsInStringArray(typ, EvaluatorRangedTypes) {
return ValidateAlertConditionRangedEvaluator(input)
}
if typ != "no_value" {
return errors.Wrapf(ErrInvalidEvaluatorType, "type: %s", typ)
}
return nil
}
func ValidateAlertConditionType(typ string) error {
if typ == "" {
return httperrors.NewInputParameterError("alert condition type is empty")
}
if !utils.IsInStringArray(typ, ConditionTypes) {
return httperrors.NewInputParameterError("Unkown alert condition type: %s", typ)
}
return nil
}
func ValidateAlertConditionThresholdEvaluator(input monitor.Condition) error {
if len(input.Params) == 0 {
return errors.Wrapf(ErrMissingParameterThreshold, "Evaluator %s", HumanThresholdType(input.Type))
}
return nil
}
func ValidateAlertConditionRangedEvaluator(input monitor.Condition) error {
if len(input.Params) == 0 {
return errors.Wrapf(ErrMissingParameterThreshold, "Evaluator %s", HumanThresholdType(input.Type))
}
if len(input.Params) == 1 {
return errors.Wrap(ErrMissingParameterThreshold, "RangedEvaluator parameter second parameter is missing")
}
return nil
}
// HumanThresholdType converts a threshold "type" string to a string that matches the UI
// so errors are less confusing.
func HumanThresholdType(typ string) string {
switch typ {
case "gt":
return "IS ABOVE"
case "lt":
return "IS BELOW"
case "within_range":
return "IS WITHIN RANGE"
case "outside_range":
return "IS OUTSIDE RANGE"
}
return ""
}
func ValidateFromValue(from string) error {
fromRaw := strings.Replace(from, "now-", "", 1)
_, err := time.ParseDuration("-" + fromRaw)
return err
}
func ValidateToValue(to string) error {
if to == "now" {
return nil
} else if strings.HasPrefix(to, "now-") {
withoutNow := strings.Replace(to, "now-", "", 1)
_, err := time.ParseDuration("-" + withoutNow)
if err == nil {
return nil
}
}
_, err := time.ParseDuration(to)
return err
}