pkg/rules/nacos/config/nacos_go_client_config_setup.go (161 lines of code) (raw):
// Copyright (c) 2024 Alibaba Group Holding Ltd.
//
// 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 config
import (
"context"
"errors"
"log"
"reflect"
"strconv"
"time"
"unsafe"
_ "unsafe"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/api"
"github.com/alibaba/opentelemetry-go-auto-instrumentation/pkg/inst-api-semconv/instrumenter/experimental"
"github.com/nacos-group/nacos-sdk-go/v2/clients/cache"
"github.com/nacos-group/nacos-sdk-go/v2/clients/config_client"
"github.com/nacos-group/nacos-sdk-go/v2/clients/nacos_client"
"github.com/nacos-group/nacos-sdk-go/v2/common/nacos_error"
"github.com/nacos-group/nacos-sdk-go/v2/common/nacos_server"
"github.com/nacos-group/nacos-sdk-go/v2/common/remote/rpc"
"github.com/nacos-group/nacos-sdk-go/v2/common/remote/rpc/rpc_request"
"github.com/nacos-group/nacos-sdk-go/v2/common/remote/rpc/rpc_response"
"go.opentelemetry.io/otel/attribute"
"go.opentelemetry.io/otel/metric"
)
//go:linkname beforeNewConfigClient github.com/nacos-group/nacos-sdk-go/v2/clients/config_client.beforeNewConfigClient
func beforeNewConfigClient(call api.CallContext, nc nacos_client.INacosClient) {
if !experimental.NacosEnabler.Enable() {
return
}
param, err := nc.GetClientConfig()
if err != nil {
return
}
call.SetKeyData("namespace", param.NamespaceId)
call.SetKeyData("region", param.RegionId)
call.SetKeyData("appName", param.AppName)
call.SetKeyData("appKey", param.AppKey)
call.SetKeyData("userName", param.Username)
}
//go:linkname afterNewConfigClient github.com/nacos-group/nacos-sdk-go/v2/clients/config_client.afterNewConfigClient
func afterNewConfigClient(call api.CallContext, client *config_client.ConfigClient, err error) {
if !experimental.NacosEnabler.Enable() {
return
}
if client == nil {
return
}
// get reference for cache map
t := reflect.ValueOf(client)
if t.Kind() == reflect.Ptr {
t = t.Elem()
} else {
return
}
cacheMapField := t.FieldByName("cacheMap")
if cacheMapField.IsValid() {
cf := reflect.NewAt(cacheMapField.Type(), unsafe.Pointer(cacheMapField.UnsafeAddr())).Elem()
cacheMap, ok := cf.Interface().(cache.ConcurrentMap)
if !ok {
return
}
attrSet := attribute.NewSet(attribute.KeyValue{
Key: "namespace",
Value: attribute.StringValue(call.GetKeyData("namespace").(string)),
}, attribute.KeyValue{
Key: "region",
Value: attribute.StringValue(call.GetKeyData("region").(string)),
}, attribute.KeyValue{
Key: "appName",
Value: attribute.StringValue(call.GetKeyData("appName").(string)),
}, attribute.KeyValue{
Key: "appKey",
Value: attribute.StringValue(call.GetKeyData("appKey").(string)),
}, attribute.KeyValue{
Key: "userName",
Value: attribute.StringValue(call.GetKeyData("userName").(string)),
})
reg, err := experimental.GlobalMeter.RegisterCallback(func(ctx context.Context, observer metric.Observer) error {
observer.ObserveInt64(experimental.ClientConfigCacheMapSize, int64(cacheMap.Count()), metric.WithAttributeSet(attrSet))
return nil
}, experimental.ClientConfigCacheMapSize)
if err != nil {
log.Printf("[otel nacos] failed to register metrics for config info holder, %v\n", err)
} else {
client.OtelReg = reg
}
}
}
//go:linkname beforeConfigClientClose github.com/nacos-group/nacos-sdk-go/v2/clients/config_client.beforeConfigClientClose
func beforeConfigClientClose(call api.CallContext, sc *config_client.ConfigClient) {
if !experimental.NacosEnabler.Enable() {
return
}
if sc.OtelReg == nil {
return
}
if reg, ok := sc.OtelReg.(metric.Registration); ok {
err := reg.Unregister()
if err != nil {
log.Printf("[otel nacos] failed to unregister metrics for config info holder, %v", err)
}
}
}
//go:linkname beforeCallConfigServer github.com/nacos-group/nacos-sdk-go/v2/common/nacos_server.beforeCallConfigServer
func beforeCallConfigServer(call api.CallContext, server *nacos_server.NacosServer, api string, params map[string]string, newHeaders map[string]string,
method string, curServer string, contextPath string, timeoutMS uint64) {
if !experimental.NacosEnabler.Enable() {
return
}
call.SetKeyData("ts", time.Now().UnixMilli())
call.SetKeyData("method", method)
call.SetKeyData("type", contextPath+api)
}
//go:linkname afterCallConfigServer github.com/nacos-group/nacos-sdk-go/v2/common/nacos_server.afterCallConfigServer
func afterCallConfigServer(call api.CallContext, result string, err error) {
if !experimental.NacosEnabler.Enable() {
return
}
method := call.GetKeyData("method").(string)
t := call.GetKeyData("ts").(int64)
tpe := call.GetKeyData("type").(string)
code := "200"
if err != nil {
var nacosErr *nacos_error.NacosError
errors.As(err, &nacosErr)
code = nacosErr.ErrorCode()
}
set := attribute.NewSet(attribute.KeyValue{
Key: "method",
Value: attribute.StringValue(method),
}, attribute.KeyValue{
Key: "type",
Value: attribute.StringValue(tpe),
}, attribute.KeyValue{
Key: "status",
Value: attribute.StringValue(code),
})
experimental.ClientConfigRequestDuration.Record(context.Background(), float64(time.Now().UnixMilli()-t), metric.WithAttributeSet(set))
}
//go:linkname beforeRequestProxy github.com/nacos-group/nacos-sdk-go/v2/clients/config_client.beforeRequestProxy
func beforeRequestProxy(call api.CallContext, cp *config_client.ConfigProxy, rpcClient *rpc.RpcClient, request rpc_request.IRequest, timeoutMills uint64) {
if !experimental.NacosEnabler.Enable() {
return
}
call.SetKeyData("ts", time.Now().UnixMilli())
call.SetKeyData("request", request)
}
//go:linkname afterRequestProxy github.com/nacos-group/nacos-sdk-go/v2/clients/config_client.afterRequestProxy
func afterRequestProxy(call api.CallContext, resp rpc_response.IResponse, err error) {
if !experimental.NacosEnabler.Enable() {
return
}
t := call.GetKeyData("ts").(int64)
req := call.GetKeyData("request").(rpc_request.IRequest)
code := "NA"
if resp != nil {
code = strconv.Itoa(resp.GetResultCode())
}
set := attribute.NewSet(attribute.KeyValue{
Key: "method",
Value: attribute.StringValue("GRPC"),
}, attribute.KeyValue{
Key: "type",
Value: attribute.StringValue(req.GetRequestType()),
}, attribute.KeyValue{
Key: "status",
Value: attribute.StringValue(code),
})
experimental.ClientConfigRequestDuration.Record(context.Background(), float64(time.Now().UnixMilli()-t), metric.WithAttributeSet(set))
}