2024-10-23 03:29:28 +00:00
|
|
|
// Copyright (C) 2019-2020 Zilliz. All rights reserved.
|
|
|
|
//
|
|
|
|
// 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 metricsinfo
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2024-10-28 02:13:29 +00:00
|
|
|
"sync"
|
2024-10-23 03:29:28 +00:00
|
|
|
|
|
|
|
"github.com/cockroachdb/errors"
|
|
|
|
"github.com/tidwall/gjson"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/milvuspb"
|
|
|
|
"github.com/milvus-io/milvus/pkg/common"
|
|
|
|
"github.com/milvus-io/milvus/pkg/log"
|
|
|
|
"github.com/milvus-io/milvus/pkg/util/commonpbutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// MetricTypeKey are the key of metric type in GetMetrics request.
|
|
|
|
MetricTypeKey = common.MetricTypeKey
|
|
|
|
|
|
|
|
// SystemInfoMetrics means users request for system information metrics.
|
|
|
|
SystemInfoMetrics = "system_info"
|
|
|
|
|
|
|
|
// CollectionStorageMetrics means users request for collection storage metrics.
|
|
|
|
CollectionStorageMetrics = "collection_storage"
|
|
|
|
|
|
|
|
// MetricRequestTypeKey is a key for identify request type.
|
|
|
|
MetricRequestTypeKey = "req_type"
|
|
|
|
|
|
|
|
// MetricRequestParamsSeparator is a separator that parameter value will be joined be separator
|
|
|
|
MetricRequestParamsSeparator = ","
|
|
|
|
|
|
|
|
// QuerySegmentDist request for segment distribution on the query node
|
2024-11-07 03:52:25 +00:00
|
|
|
QuerySegments = "qn_segments"
|
2024-10-23 03:29:28 +00:00
|
|
|
|
|
|
|
// QueryChannelDist request for channel distribution on the query node
|
2024-11-07 03:52:25 +00:00
|
|
|
QueryChannels = "qn_channels"
|
|
|
|
|
|
|
|
// QueryDist request for segment/channel/leader view distribution on querycoord
|
|
|
|
QueryDist = "qc_dist"
|
|
|
|
|
|
|
|
// QueryTarget request for segment/channel target on the querycoord
|
|
|
|
QueryTarget = "qc_target"
|
2024-10-23 03:29:28 +00:00
|
|
|
|
2024-10-28 02:13:29 +00:00
|
|
|
// QueryCoordAllTasks request for get tasks on the querycoord
|
|
|
|
QueryCoordAllTasks = "qc_tasks_all"
|
2024-10-23 03:29:28 +00:00
|
|
|
|
|
|
|
// QueryReplicas request for get replica on the querycoord
|
2024-10-28 02:13:29 +00:00
|
|
|
QueryReplicas = "qc_replica"
|
2024-10-23 03:29:28 +00:00
|
|
|
|
|
|
|
// QueryResourceGroups request for get resource groups on the querycoord
|
2024-10-28 02:13:29 +00:00
|
|
|
QueryResourceGroups = "qc_resource_group"
|
|
|
|
|
2024-11-07 03:52:25 +00:00
|
|
|
// DataDist request for get segments on the datacoord
|
|
|
|
DataDist = "dc_segments"
|
2024-10-28 02:13:29 +00:00
|
|
|
|
|
|
|
// ImportTasks request for get import tasks from the datacoord
|
|
|
|
ImportTasks = "dc_import_tasks"
|
|
|
|
|
|
|
|
// CompactionTasks request for get compaction tasks from the datacoord
|
|
|
|
CompactionTasks = "dc_compaction_tasks"
|
|
|
|
|
|
|
|
// BuildIndexTasks request for get building index tasks from the datacoord
|
|
|
|
BuildIndexTasks = "dc_build_index_tasks"
|
|
|
|
|
|
|
|
// SyncTasks request for get sync tasks from the datanode
|
|
|
|
SyncTasks = "dn_sync_tasks"
|
2024-10-23 03:29:28 +00:00
|
|
|
|
2024-11-07 03:52:25 +00:00
|
|
|
// DataSegments request for get segments from the datanode
|
|
|
|
DataSegments = "dn_segments"
|
|
|
|
|
|
|
|
// DataChannels request for get channels from the datanode
|
|
|
|
DataChannels = "dn_channels"
|
|
|
|
|
2024-10-23 03:29:28 +00:00
|
|
|
// MetricRequestParamVerboseKey as a request parameter decide to whether return verbose value
|
|
|
|
MetricRequestParamVerboseKey = "verbose"
|
2024-11-12 09:44:29 +00:00
|
|
|
|
|
|
|
MetricRequestParamTargetScopeKey = "target_scope"
|
2024-10-23 03:29:28 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type MetricsRequestAction func(ctx context.Context, req *milvuspb.GetMetricsRequest, jsonReq gjson.Result) (string, error)
|
|
|
|
|
2024-10-28 02:13:29 +00:00
|
|
|
type MetricsRequest struct {
|
|
|
|
metricsReqType2Action map[string]MetricsRequestAction
|
|
|
|
lock sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewMetricsRequest() *MetricsRequest {
|
|
|
|
return &MetricsRequest{
|
|
|
|
metricsReqType2Action: make(map[string]MetricsRequestAction),
|
|
|
|
}
|
|
|
|
}
|
2024-10-23 03:29:28 +00:00
|
|
|
|
2024-10-28 02:13:29 +00:00
|
|
|
func (mr *MetricsRequest) RegisterMetricsRequest(reqType string, action MetricsRequestAction) {
|
|
|
|
mr.lock.Lock()
|
|
|
|
defer mr.lock.Unlock()
|
|
|
|
_, ok := mr.metricsReqType2Action[reqType]
|
2024-10-23 03:29:28 +00:00
|
|
|
if ok {
|
|
|
|
log.Info("metrics request type already exists", zap.String("reqType", reqType))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-10-28 02:13:29 +00:00
|
|
|
mr.metricsReqType2Action[reqType] = action
|
2024-10-23 03:29:28 +00:00
|
|
|
}
|
|
|
|
|
2024-10-28 02:13:29 +00:00
|
|
|
func (mr *MetricsRequest) ExecuteMetricsRequest(ctx context.Context, req *milvuspb.GetMetricsRequest) (string, error) {
|
2024-10-23 03:29:28 +00:00
|
|
|
jsonReq := gjson.Parse(req.Request)
|
|
|
|
reqType, err := ParseMetricRequestType(jsonReq)
|
|
|
|
if err != nil {
|
|
|
|
log.Warn("failed to parse metric type", zap.Error(err))
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2024-10-28 02:13:29 +00:00
|
|
|
mr.lock.Lock()
|
|
|
|
action, ok := mr.metricsReqType2Action[reqType]
|
2024-10-23 03:29:28 +00:00
|
|
|
if !ok {
|
2024-10-28 02:13:29 +00:00
|
|
|
mr.lock.Unlock()
|
2024-10-23 03:29:28 +00:00
|
|
|
log.Warn("unimplemented metric request type", zap.String("req_type", reqType))
|
|
|
|
return "", errors.New(MsgUnimplementedMetric)
|
|
|
|
}
|
2024-10-28 02:13:29 +00:00
|
|
|
mr.lock.Unlock()
|
2024-10-23 03:29:28 +00:00
|
|
|
|
|
|
|
actionRet, err := action(ctx, req, jsonReq)
|
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("failed to execute %s", reqType)
|
|
|
|
log.Warn(msg, zap.Error(err))
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return actionRet, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func RequestWithVerbose(jsonReq gjson.Result) bool {
|
|
|
|
v := jsonReq.Get(MetricRequestParamVerboseKey)
|
|
|
|
if !v.Exists() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return v.Bool()
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseMetricRequestType returns the metric type of req
|
|
|
|
func ParseMetricRequestType(jsonRet gjson.Result) (string, error) {
|
|
|
|
v := jsonRet.Get(MetricRequestTypeKey)
|
|
|
|
if v.Exists() {
|
|
|
|
return v.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
v = jsonRet.Get(MetricTypeKey)
|
|
|
|
if v.Exists() {
|
|
|
|
return v.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", fmt.Errorf("%s or %s not found in request", MetricTypeKey, MetricRequestTypeKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConstructRequestByMetricType constructs a request according to the metric type
|
|
|
|
func ConstructRequestByMetricType(metricType string) (*milvuspb.GetMetricsRequest, error) {
|
|
|
|
m := make(map[string]interface{})
|
|
|
|
m[MetricTypeKey] = metricType
|
|
|
|
binary, err := json.Marshal(m)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to construct request by metric type %s: %s", metricType, err.Error())
|
|
|
|
}
|
|
|
|
// TODO:: switch metricType to different msgType and return err when metricType is not supported
|
|
|
|
return &milvuspb.GetMetricsRequest{
|
|
|
|
Base: commonpbutil.NewMsgBase(
|
|
|
|
commonpbutil.WithMsgType(commonpb.MsgType_SystemInfo),
|
|
|
|
),
|
|
|
|
Request: string(binary),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ConstructGetMetricsRequest(m map[string]interface{}) (*milvuspb.GetMetricsRequest, error) {
|
|
|
|
binary, err := json.Marshal(m)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to construct request: %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return &milvuspb.GetMetricsRequest{
|
|
|
|
Base: commonpbutil.NewMsgBase(
|
|
|
|
commonpbutil.WithMsgType(commonpb.MsgType_SystemInfo),
|
|
|
|
),
|
|
|
|
Request: string(binary),
|
|
|
|
}, nil
|
|
|
|
}
|