Files
cloudpods/pkg/monitor/models/unifiedmonitor.go

530 lines
15 KiB
Go
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
// 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"
"fmt"
"strings"
"time"
"yunion.io/x/jsonutils"
"yunion.io/x/log"
"yunion.io/x/pkg/errors"
"yunion.io/x/onecloud/pkg/apis/monitor"
"yunion.io/x/onecloud/pkg/cloudcommon/db"
"yunion.io/x/onecloud/pkg/hostman/hostinfo/hostconsts"
"yunion.io/x/onecloud/pkg/httperrors"
"yunion.io/x/onecloud/pkg/mcclient"
merrors "yunion.io/x/onecloud/pkg/monitor/errors"
mq "yunion.io/x/onecloud/pkg/monitor/metricquery"
"yunion.io/x/onecloud/pkg/monitor/tsdb"
"yunion.io/x/onecloud/pkg/monitor/validators"
"yunion.io/x/onecloud/pkg/util/rbacutils"
)
const (
TELEGRAF_DATABASE = "telegraf"
)
var (
UnifiedMonitorManager *SUnifiedMonitorManager
)
func init() {
UnifiedMonitorManager = &SUnifiedMonitorManager{
SVirtualResourceBaseManager: db.NewVirtualResourceBaseManager(
&SUnifiedMonitorManager{},
"",
"unifiedmonitor",
"unifiedmonitors",
),
}
UnifiedMonitorManager.SetVirtualObject(UnifiedMonitorManager)
}
type SUnifiedMonitorManager struct {
db.SVirtualResourceBaseManager
}
type SUnifiedMonitorModel struct {
}
func (self *SUnifiedMonitorManager) AllowGetPropertyDatabases(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject) bool {
return true
}
func (self *SUnifiedMonitorManager) GetPropertyDatabases(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject) (jsonutils.JSONObject, error) {
return DataSourceManager.GetDatabases()
}
func (self *SUnifiedMonitorManager) AllowGetPropertyMeasurements(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject) bool {
return true
}
func (self *SUnifiedMonitorManager) GetPropertyMeasurements(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject) (jsonutils.JSONObject, error) {
filter, err := getTagFilterByRequestQuery(ctx, userCred, query)
if err != nil {
return nil, err
}
return DataSourceManager.GetMeasurementsWithDescriptionInfos(query, "", filter)
}
func getTagFilterByRequestQuery(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject) (filter string, err error) {
scope, _ := query.GetString("scope")
filter, err = filterByScope(ctx, userCred, scope, query)
return
}
func filterByScope(ctx context.Context, userCred mcclient.TokenCredential, scope string, data jsonutils.JSONObject) (string, error) {
domainId := jsonutils.GetAnyString(data, []string{"domain_id", "domain", "project_domain_id", "project_domain"})
projectId := jsonutils.GetAnyString(data, []string{"project_id", "project"})
if projectId != "" {
project, err := db.DefaultProjectFetcher(ctx, projectId)
if err != nil {
return "", err
}
projectId = project.GetProjectId()
domainId = project.GetProjectDomainId()
}
if domainId != "" {
domain, err := db.DefaultDomainFetcher(ctx, domainId)
if err != nil {
return "", err
}
domainId = domain.GetProjectDomainId()
domain.GetProjectId()
}
switch scope {
case "system":
return "", nil
case "domain":
if domainId == "" {
domainId = userCred.GetProjectDomainId()
}
return getProjectIdsFilterByDomain(domainId)
default:
if projectId == "" {
projectId = userCred.GetProjectId()
}
return getProjectIdFilterByProject(projectId)
}
}
func getTenantIdStr(role string, userCred mcclient.TokenCredential) (string, error) {
if role == "admin" {
return "", nil
}
if role == "domainadmin" {
domainId := userCred.GetDomainId()
return getProjectIdsFilterByDomain(domainId)
}
if role == "member" {
tenantId := userCred.GetProjectId()
return getProjectIdFilterByProject(tenantId)
}
return "", errors.ErrNotFound
}
func getProjectIdsFilterByDomain(domainId string) (string, error) {
//s := auth.GetAdminSession(context.Background(), "", "")
//params := jsonutils.Marshal(map[string]string{"domain_id": domainId})
//tenants, err := modules.Projects.List(s, params)
//if err != nil {
// return "", errors.Wrap(err, "Projects.List")
//}
//var buffer bytes.Buffer
//buffer.WriteString("( ")
//for index, tenant := range tenants.Data {
// tenantId, _ := tenant.GetString("id")
// if index != len(tenants.Data)-1 {
// buffer.WriteString(fmt.Sprintf(" %s =~ /%s/ %s ", "tenant_id", tenantId, "OR"))
// } else {
// buffer.WriteString(fmt.Sprintf(" %s =~ /%s/ ", "tenant_id", tenantId))
// }
//}
//buffer.WriteString(" )")
//return buffer.String(), nil
return fmt.Sprintf(`"%s" =~ /%s/`, "domain_id", domainId), nil
}
func getProjectIdFilterByProject(projectId string) (string, error) {
return fmt.Sprintf(`"%s" =~ /%s/`, "tenant_id", projectId), nil
}
func (self *SUnifiedMonitorManager) AllowGetPropertyMetricMeasurement(ctx context.Context,
userCred mcclient.TokenCredential,
query jsonutils.JSONObject) bool {
return true
}
func (self *SUnifiedMonitorManager) GetPropertyMetricMeasurement(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject) (jsonutils.JSONObject, error) {
metricFunc := monitor.MetricFunc{
FieldOptType: monitor.UNIFIED_MONITOR_FIELD_OPT_TYPE,
FieldOptValue: monitor.UNIFIED_MONITOR_FIELD_OPT_VALUE,
GroupOptType: monitor.UNIFIED_MONITOR_GROUPBY_OPT_TYPE,
GroupOptValue: monitor.UNIFIED_MONITOR_GROUPBY_OPT_VALUE,
}
filter, err := getTagFilterByRequestQuery(ctx, userCred, query)
if err != nil {
return nil, err
}
rtn, err := DataSourceManager.GetMetricMeasurement(query, filter)
if err != nil {
return nil, err
}
rtn.(*jsonutils.JSONDict).Add(jsonutils.Marshal(&metricFunc), "func")
return rtn, nil
}
func (self *SUnifiedMonitorManager) AllowPerformQuery(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject, data jsonutils.JSONObject) bool {
return true
}
func (self *SUnifiedMonitorManager) PerformQuery(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
tmp := jsonutils.DeepCopy(data)
self.handleDataPreSignature(ctx, tmp)
if err := ValidateQuerySignature(tmp); err != nil {
return nil, errors.Wrap(err, "ValidateQuerySignature")
}
inputQuery := new(monitor.MetricInputQuery)
err := data.Unmarshal(inputQuery)
if err != nil {
return jsonutils.NewDict(), err
}
if len(inputQuery.MetricQuery) == 0 {
return nil, merrors.NewArgIsEmptyErr("metric_query")
}
for _, q := range inputQuery.MetricQuery {
scope, _ := data.GetString("scope")
ownId, _ := self.FetchOwnerId(ctx, data)
if ownId == nil {
ownId = userCred
}
setDefaultValue(q, inputQuery, scope, ownId)
err = self.ValidateInputQuery(q)
if err != nil {
return jsonutils.NewDict(), err
}
}
var groupByTag = make([]string, 0)
for _, query := range inputQuery.MetricQuery {
for _, group := range query.Model.GroupBy {
if group.Type == "tag" {
groupByTag = append(groupByTag, group.Params[0])
}
}
}
rtn, err := doQuery(*inputQuery)
if err != nil {
return jsonutils.NewDict(), err
}
if len(inputQuery.Soffset) != 0 && len(inputQuery.Slimit) != 0 {
seriesTotal := self.fillSearchSeriesTotalQuery(*inputQuery.MetricQuery[0])
rtn.SeriesTotal = seriesTotal
}
setSerieRowName(&rtn.Series, groupByTag)
fillSerieTags(&rtn.Series)
return jsonutils.Marshal(rtn), nil
}
func (self *SUnifiedMonitorManager) fillSearchSeriesTotalQuery(fork monitor.AlertQuery) int64 {
newGroupByPart := make([]monitor.MetricQueryPart, 0)
newGroupByPart = append(newGroupByPart, fork.Model.GroupBy[0])
fork.Model.GroupBy = newGroupByPart
forkInputQury := new(monitor.MetricInputQuery)
forkInputQury.MetricQuery = []*monitor.AlertQuery{&fork}
rtn, err := doQuery(*forkInputQury)
if err != nil {
log.Errorf("exec forkInputQury err:%v", err)
return 0
}
log.Errorf("series len:%d", len(rtn.Series))
return int64(len(rtn.Series))
}
func (self *SUnifiedMonitorManager) handleDataPreSignature(ctx context.Context, data jsonutils.JSONObject) {
scope, _ := data.GetString("scope")
isIdentityName, _ := data.Bool("identity_name")
switch scope {
case "system":
case "domain":
domain, err := data.GetString("project_domain")
if err == nil {
domainObj, _ := db.DefaultDomainFetcher(ctx, domain)
if isIdentityName {
domain = domainObj.Name
}
data.(*jsonutils.JSONDict).Remove("project_domain")
data.(*jsonutils.JSONDict).Set("domain_id", jsonutils.NewString(domain))
}
default:
project, err := data.GetString("project")
if err == nil {
tenant, _ := db.DefaultProjectFetcher(ctx, project)
if isIdentityName {
project = tenant.Name
}
data.(*jsonutils.JSONDict).Remove("project")
data.(*jsonutils.JSONDict).Set("project_id", jsonutils.NewString(project))
}
}
}
func doQuery(query monitor.MetricInputQuery) (*mq.Metrics, error) {
conditions := make([]*monitor.AlertCondition, 0)
for _, q := range query.MetricQuery {
condition := monitor.AlertCondition{
Type: "metricquery",
Query: *q,
}
conditions = append(conditions, &condition)
}
factory := mq.GetQueryFactories()["metricquery"]
metricQ, err := factory(conditions)
if err != nil {
return nil, err
}
metrics, err := metricQ.ExecuteQuery()
if err != nil {
return nil, err
}
// drop metas contains raw_query
if !query.ShowMeta {
metrics.Metas = nil
}
return metrics, nil
}
func (self *SUnifiedMonitorManager) ValidateInputQuery(query *monitor.AlertQuery) error {
if query.From == "" {
query.From = "1h"
}
if query.Model.Interval == "" {
query.Model.Interval = "5m"
}
if query.To == "" {
query.To = "now"
}
if _, err := time.ParseDuration(query.Model.Interval); err != nil {
return httperrors.NewInputParameterError("Invalid interval format: %s", query.Model.Interval)
}
return validators.ValidateSelectOfMetricQuery(*query)
}
func setDefaultValue(query *monitor.AlertQuery, inputQuery *monitor.MetricInputQuery,
scope string, ownerId mcclient.IIdentityProvider) {
setDataSourceId(query)
query.From = inputQuery.From
query.To = inputQuery.To
query.Model.Interval = inputQuery.Interval
metricMeasurement, _ := MetricMeasurementManager.GetCache().Get(query.Model.Measurement)
checkQueryGroupBy(query, inputQuery)
if len(inputQuery.Interval) != 0 {
query.Model.GroupBy = append(query.Model.GroupBy,
monitor.MetricQueryPart{
Type: "time",
Params: []string{"$interval"},
},
monitor.MetricQueryPart{
Type: "fill",
Params: []string{"none"},
})
}
if len(inputQuery.Slimit) != 0 && len(inputQuery.Soffset) != 0 {
query.Model.GroupBy = append(query.Model.GroupBy,
monitor.MetricQueryPart{Type: "slimit", Params: []string{inputQuery.Slimit}},
monitor.MetricQueryPart{Type: "soffset", Params: []string{inputQuery.Soffset}},
)
}
if query.Model.Database == "" {
database := ""
if metricMeasurement == nil {
log.Warningf("Not found measurement %s from metrics measurement cache", query.Model.Measurement)
} else {
database = metricMeasurement.Database
}
if database == "" {
// hack: query from default telegraf database if no metric measurement matched
database = TELEGRAF_DATABASE
}
query.Model.Database = database
}
for i, sel := range query.Model.Selects {
if len(sel) > 1 {
continue
}
sel = append(sel, monitor.MetricQueryPart{
Type: "mean",
Params: []string{},
})
query.Model.Selects[i] = sel
}
var projectId, domainId string
switch rbacutils.TRbacScope(scope) {
case rbacutils.ScopeProject:
projectId = ownerId.GetProjectId()
containId := false
for _, tagFilter := range query.Model.Tags {
if tagFilter.Key == "tenant_id" {
containId = true
break
}
}
if !containId {
query.Model.Tags = append(query.Model.Tags, monitor.MetricQueryTag{
Key: "tenant_id",
Operator: "=",
Value: projectId,
Condition: "and",
})
}
case rbacutils.ScopeDomain:
domainId = ownerId.GetProjectDomainId()
containId := false
for _, tagFilter := range query.Model.Tags {
if tagFilter.Key == "domain_id" {
containId = true
break
}
}
if !containId {
query.Model.Tags = append(query.Model.Tags, monitor.MetricQueryTag{
Key: "domain_id",
Operator: "=",
Value: domainId,
Condition: "and",
})
}
}
if metricMeasurement != nil && metricMeasurement.ResType == hostconsts.TELEGRAF_TAG_ONECLOUD_RES_TYPE {
query.Model.Tags = append(query.Model.Tags, monitor.MetricQueryTag{
Key: hostconsts.TELEGRAF_TAG_KEY_RES_TYPE,
Operator: "=",
Value: hostconsts.TELEGRAF_TAG_ONECLOUD_RES_TYPE,
Condition: "and",
})
}
}
func setDataSourceId(query *monitor.AlertQuery) {
datasource, _ := DataSourceManager.GetDefaultSource()
query.DataSourceId = datasource.Id
}
func checkQueryGroupBy(query *monitor.AlertQuery, inputQuery *monitor.MetricInputQuery) {
if len(query.Model.GroupBy) != 0 {
return
}
metricMeasurement, _ := MetricMeasurementManager.GetCache().Get(query.Model.Measurement)
if inputQuery.Unit || metricMeasurement == nil && query.Model.Database == monitor.METRIC_DATABASE_METER {
return
}
tagId := ""
if metricMeasurement != nil {
tagId = monitor.MEASUREMENT_TAG_ID[metricMeasurement.ResType]
}
if len(tagId) == 0 {
tagId = "*"
}
query.Model.GroupBy = append(query.Model.GroupBy,
monitor.MetricQueryPart{
Type: "field",
Params: []string{tagId},
})
}
func setSerieRowName(series *tsdb.TimeSeriesSlice, groupTag []string) {
//Add rownameThe front end displays the curve according to rowname
var index, unknownIndex = 1, 1
for i, serie := range *series {
//setRowName by groupTag
if len(groupTag) != 0 {
for key, val := range serie.Tags {
if strings.Contains(strings.Join(groupTag, ","), key) {
serie.RawName = fmt.Sprintf("%s", val)
(*series)[i] = serie
break
}
}
continue
}
measurement := strings.Split(serie.Name, ".")[0]
//sep measurement set RowName by spe param
measurements, _ := MetricMeasurementManager.getMeasurementByName(measurement)
if len(measurements) != 0 {
if key, ok := monitor.MEASUREMENT_TAG_KEYWORD[measurements[0].ResType]; ok {
serie.RawName = fmt.Sprintf("%d: %s", index, serie.Tags[key])
(*series)[i] = serie
index++
continue
}
}
//other condition set RowName
for key, val := range serie.Tags {
if strings.Contains(key, "id") {
serie.RawName = fmt.Sprintf("%d: %s", index, val)
(*series)[i] = serie
index++
break
}
}
if serie.RawName == "" {
serie.RawName = fmt.Sprintf("unknown-%d", unknownIndex)
(*series)[i] = serie
unknownIndex++
}
}
}
func fillSerieTags(series *tsdb.TimeSeriesSlice) {
for i, serie := range *series {
for _, tag := range []string{"brand", "platform", "hypervisor"} {
if val, ok := serie.Tags[tag]; ok {
serie.Tags["brand"] = val
break
}
}
for _, tag := range []string{"source", "status", hostconsts.TELEGRAF_TAG_KEY_HOST_TYPE,
hostconsts.TELEGRAF_TAG_KEY_RES_TYPE, "cpu", "is_vm", "os_type", "domain_name", "region"} {
if _, ok := serie.Tags[tag]; ok {
delete(serie.Tags, tag)
}
}
(*series)[i] = serie
}
}