From 39647af9ab8741bf229b31ca821734ac01cd096d Mon Sep 17 00:00:00 2001 From: Mohammed Faraaz Date: Tue, 16 Jan 2024 14:14:05 +0530 Subject: [PATCH] Go Code format checker and formatter Also formatted the files which dont adhere to go formats. Build will fail if there exists any formatting issue. Formatting is supported for bookworm as well. --- Makefile | 23 +- common_utils/constants.go | 7 +- common_utils/context.go | 6 +- common_utils/shareMem.go | 5 +- dialout/dialout_client/dialout_client.go | 6 +- dialout/dialout_client/dialout_client_test.go | 3 +- .../dialout_client_cli/dialout_client_cli.go | 2 +- dialout/dialout_server/dialout_server.go | 2 +- gnmi_server/basicAuth.go | 2 +- gnmi_server/clear_neighbor_dummy_test.go | 49 + gnmi_server/clientCertAuth.go | 2 +- gnmi_server/client_subscribe.go | 20 +- gnmi_server/connection_manager.go | 16 +- gnmi_server/constants_native.go | 11 +- gnmi_server/constants_native_write.go | 11 +- gnmi_server/constants_translib.go | 11 +- gnmi_server/constants_translib_write.go | 11 +- gnmi_server/gnoi.go | 142 +- gnmi_server/jwtAuth.go | 4 +- gnmi_server/pamAuth.go | 4 +- gnmi_server/server.go | 28 +- gnmi_server/server_test.go | 1347 ++++++++--------- gnoi_client/gnoi_client.go | 121 +- sonic_data_client/client_test.go | 119 +- sonic_data_client/db_client.go | 12 +- sonic_data_client/dummy_client_test.go | 50 + sonic_data_client/events_client.go | 741 +++++---- sonic_data_client/json_client.go | 14 +- sonic_data_client/mixed_db_client.go | 116 +- sonic_data_client/non_db_client.go | 7 +- sonic_data_client/virtual_db.go | 8 +- sonic_db_config/db_config.go | 8 +- sonic_db_config/db_config_test.go | 5 +- sonic_service_client/dbus_client.go | 10 +- sonic_service_client/dbus_client_test.go | 818 +++++----- telemetry/telemetry.go | 130 +- transl_utils/transl_utils.go | 159 +- 37 files changed, 2059 insertions(+), 1971 deletions(-) create mode 100644 gnmi_server/clear_neighbor_dummy_test.go create mode 100644 sonic_data_client/dummy_client_test.go diff --git a/Makefile b/Makefile index a1aed09c..42354603 100644 --- a/Makefile +++ b/Makefile @@ -6,9 +6,14 @@ export PATH := $(PATH):$(GOPATH)/bin INSTALL := /usr/bin/install DBDIR := /var/run/redis/sonic-db/ GO ?= /usr/local/go/bin/go +GOROOT ?= $(shell $(GO) env GOROOT) TOP_DIR := $(abspath ..) MGMT_COMMON_DIR := $(TOP_DIR)/sonic-mgmt-common BUILD_DIR := build/bin +FORMAT_CHECK = $(BUILD_DIR)/.formatcheck +FORMAT_LOG = $(BUILD_DIR)/go_format.log +# Find all .go files excluding vendor, build, and patches files +GO_FILES := $(shell find . -type f -name '*.go' ! -path './vendor/*' ! -path './build/*' ! -path './patches/*' ! -path './proto/*' ! -path './swsscommon/*') export CVL_SCHEMA_PATH := $(MGMT_COMMON_DIR)/build/cvl/schema export GOBIN := $(abspath $(BUILD_DIR)) export PATH := $(PATH):$(GOBIN):$(shell dirname $(GO)) @@ -61,7 +66,7 @@ go-deps: $(GO_DEPS) go-deps-clean: $(RM) -r vendor -sonic-gnmi: $(GO_DEPS) +sonic-gnmi: $(GO_DEPS) $(FORMAT_CHECK) ifeq ($(CROSS_BUILD_ENVIRON),y) $(GO) build -o ${GOBIN}/telemetry -mod=vendor $(BLD_FLAGS) github.com/sonic-net/sonic-gnmi/telemetry ifneq ($(ENABLE_DIALOUT_VALUE),0) @@ -136,6 +141,22 @@ clean: $(RM) -r build $(RM) -r vendor +# File target that generates a diff file if formatting is incorrect +$(FORMAT_CHECK): $(GO_FILES) + @echo "Checking Go file formatting..." + @echo $(GO_FILES) + mkdir -p $(@D) + @$(GOROOT)/bin/gofmt -l -d $(GO_FILES) > $(FORMAT_LOG) + @if [ -s $(FORMAT_LOG) ]; then \ + cat $(FORMAT_LOG); \ + echo "Formatting issues found. Please run 'gofmt -w .' on the above files and commit the changes."; \ + exit 1; \ + else \ + echo "All files are properly formatted."; \ + rm -f $(FORMAT_LOG); \ + fi + touch $@ + install: $(INSTALL) -D $(BUILD_DIR)/telemetry $(DESTDIR)/usr/sbin/telemetry ifneq ($(ENABLE_DIALOUT_VALUE),0) diff --git a/common_utils/constants.go b/common_utils/constants.go index db38b0e6..86c335bc 100644 --- a/common_utils/constants.go +++ b/common_utils/constants.go @@ -1,4 +1,3 @@ - -package common_utils - -const GNMI_WORK_PATH = "/tmp" +package common_utils + +const GNMI_WORK_PATH = "/tmp" diff --git a/common_utils/context.go b/common_utils/context.go index e181b1d4..202047d6 100644 --- a/common_utils/context.go +++ b/common_utils/context.go @@ -6,11 +6,10 @@ import ( "sync/atomic" ) - // AuthInfo holds data about the authenticated user type AuthInfo struct { // Username - User string + User string AuthEnabled bool // Roles Roles []string @@ -37,6 +36,7 @@ const requestContextKey contextkey = 0 var requestCounter uint64 type CounterType int + const ( GNMI_GET CounterType = iota GNMI_GET_FAIL @@ -89,7 +89,6 @@ func (c CounterType) String() string { var globalCounters [COUNTER_SIZE]uint64 - // GetContext function returns the RequestContext object for a // gRPC request. RequestContext is maintained as a context value of // the request. Creates a new RequestContext object is not already @@ -125,4 +124,3 @@ func IncCounter(cnt CounterType) { atomic.AddUint64(&globalCounters[cnt], 1) SetMemCounters(&globalCounters) } - diff --git a/common_utils/shareMem.go b/common_utils/shareMem.go index ad0caeb0..843760f0 100644 --- a/common_utils/shareMem.go +++ b/common_utils/shareMem.go @@ -9,11 +9,11 @@ import ( // Use share memory to dump GNMI internal counters, // GNMI server and gnmi_dump should use memKey to access the share memory, // memSize is 1024 bytes, so we can support 128 counters -// memMode is 0x380, this value is O_RDWR|IPC_CREAT, +// memMode is 0x380, this value is O_RDWR|IPC_CREAT, // O_RDWR means: Owner can write and read the file, everyone else can't. // IPC_CREAT means: Create a shared memory segment if a shared memory identifier does not exist for memKey. var ( - memKey = 7749 + memKey = 7749 memSize = 1024 memMode = 0x380 ) @@ -61,4 +61,3 @@ func GetMemCounters(counters *[int(COUNTER_SIZE)]uint64) error { } return nil } - diff --git a/dialout/dialout_client/dialout_client.go b/dialout/dialout_client/dialout_client.go index 8c3678e5..7eb51445 100644 --- a/dialout/dialout_client/dialout_client.go +++ b/dialout/dialout_client/dialout_client.go @@ -5,14 +5,14 @@ import ( "crypto/tls" "errors" "fmt" - spb "github.com/sonic-net/sonic-gnmi/proto" - sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" - sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" "github.com/Workiva/go-datastructures/queue" "github.com/go-redis/redis" log "github.com/golang/glog" gpb "github.com/openconfig/gnmi/proto/gnmi" "github.com/openconfig/ygot/ygot" + spb "github.com/sonic-net/sonic-gnmi/proto" + sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" + sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" "golang.org/x/net/context" "google.golang.org/grpc" "google.golang.org/grpc/credentials" diff --git a/dialout/dialout_client/dialout_client_test.go b/dialout/dialout_client/dialout_client_test.go index db37aaa4..a7b7a30e 100644 --- a/dialout/dialout_client/dialout_client_test.go +++ b/dialout/dialout_client/dialout_client_test.go @@ -28,10 +28,10 @@ import ( "testing" "time" + gclient "github.com/openconfig/gnmi/client/gnmi" sds "github.com/sonic-net/sonic-gnmi/dialout/dialout_server" sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" - gclient "github.com/openconfig/gnmi/client/gnmi" ) var clientTypes = []string{gclient.Type} @@ -326,7 +326,6 @@ func serverOp(t *testing.T, sop ServerOp) { } } -// func TestGNMIDialOutPublish(t *testing.T) { fileName := "../../testdata/COUNTERS_PORT_NAME_MAP.txt" diff --git a/dialout/dialout_client_cli/dialout_client_cli.go b/dialout/dialout_client_cli/dialout_client_cli.go index de8b5033..b6156fcc 100644 --- a/dialout/dialout_client_cli/dialout_client_cli.go +++ b/dialout/dialout_client_cli/dialout_client_cli.go @@ -4,9 +4,9 @@ package main import ( "crypto/tls" "flag" - dc "github.com/sonic-net/sonic-gnmi/dialout/dialout_client" log "github.com/golang/glog" gpb "github.com/openconfig/gnmi/proto/gnmi" + dc "github.com/sonic-net/sonic-gnmi/dialout/dialout_client" "golang.org/x/net/context" "os" "os/signal" diff --git a/dialout/dialout_server/dialout_server.go b/dialout/dialout_server/dialout_server.go index 4fbde056..74a09b96 100644 --- a/dialout/dialout_server/dialout_server.go +++ b/dialout/dialout_server/dialout_server.go @@ -3,10 +3,10 @@ package dialout_server import ( "errors" "fmt" - spb "github.com/sonic-net/sonic-gnmi/proto" log "github.com/golang/glog" "github.com/google/gnxi/utils" gpb "github.com/openconfig/gnmi/proto/gnmi" + spb "github.com/sonic-net/sonic-gnmi/proto" "google.golang.org/grpc" "google.golang.org/grpc/codes" "google.golang.org/grpc/peer" diff --git a/gnmi_server/basicAuth.go b/gnmi_server/basicAuth.go index 78330ac4..fa08c7f4 100644 --- a/gnmi_server/basicAuth.go +++ b/gnmi_server/basicAuth.go @@ -1,8 +1,8 @@ package gnmi import ( - "github.com/sonic-net/sonic-gnmi/common_utils" "github.com/golang/glog" + "github.com/sonic-net/sonic-gnmi/common_utils" "golang.org/x/net/context" "google.golang.org/grpc/codes" "google.golang.org/grpc/metadata" diff --git a/gnmi_server/clear_neighbor_dummy_test.go b/gnmi_server/clear_neighbor_dummy_test.go new file mode 100644 index 00000000..9ec7a5f2 --- /dev/null +++ b/gnmi_server/clear_neighbor_dummy_test.go @@ -0,0 +1,49 @@ +package gnmi + +import ( + "testing" + "time" + + spb_gnoi "github.com/sonic-net/sonic-gnmi/proto/gnoi" + "golang.org/x/net/context" +) + +func TestDummyClearNeighbor(t *testing.T) { + // Start server + s := createServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + // Run Client + client := createClient(t, 8081) + sc := spb_gnoi.NewSonicServiceClient(client) + req := &spb_gnoi.ClearNeighborsRequest{ + Input: &spb_gnoi.ClearNeighborsRequest_Input{}, + } + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + sc.ClearNeighbors(ctx, req) + ctx2, cancel2 := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel2() + req2 := &spb_gnoi.CopyConfigRequest{ + Input: &spb_gnoi.CopyConfigRequest_Input{}, + } + sc.CopyConfig(ctx2, req2) +} + +func TestDummyCopyConfig(t *testing.T) { + // Start server + s := createServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + // Run Client + client := createClient(t, 8081) + sc := spb_gnoi.NewSonicServiceClient(client) + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + req := &spb_gnoi.CopyConfigRequest{ + Input: &spb_gnoi.CopyConfigRequest_Input{}, + } + sc.CopyConfig(ctx, req) +} diff --git a/gnmi_server/clientCertAuth.go b/gnmi_server/clientCertAuth.go index 1c44d9c5..b140223a 100644 --- a/gnmi_server/clientCertAuth.go +++ b/gnmi_server/clientCertAuth.go @@ -1,8 +1,8 @@ package gnmi import ( - "github.com/sonic-net/sonic-gnmi/common_utils" "github.com/golang/glog" + "github.com/sonic-net/sonic-gnmi/common_utils" "golang.org/x/net/context" "google.golang.org/grpc/codes" "google.golang.org/grpc/credentials" diff --git a/gnmi_server/client_subscribe.go b/gnmi_server/client_subscribe.go index 5d27177a..966b5699 100644 --- a/gnmi_server/client_subscribe.go +++ b/gnmi_server/client_subscribe.go @@ -8,11 +8,11 @@ import ( "github.com/Workiva/go-datastructures/queue" log "github.com/golang/glog" + gnmipb "github.com/openconfig/gnmi/proto/gnmi" + sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" "google.golang.org/grpc" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" - sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" - gnmipb "github.com/openconfig/gnmi/proto/gnmi" ) // Client contains information about a subscribe client that has connected to the server. @@ -28,9 +28,9 @@ type Client struct { q *queue.PriorityQueue subscribe *gnmipb.SubscriptionList // Wait for all sub go routine to finish - w sync.WaitGroup - fatal bool - logLevel int + w sync.WaitGroup + fatal bool + logLevel int } // Syslog level for error @@ -44,8 +44,8 @@ var connectionManager *ConnectionManager func NewClient(addr net.Addr) *Client { pq := queue.NewPriorityQueue(1, false) return &Client{ - addr: addr, - q: pq, + addr: addr, + q: pq, logLevel: logLevelError, } } @@ -58,7 +58,7 @@ func (c *Client) setConnectionManager(threshold int) { if connectionManager != nil && threshold == connectionManager.GetThreshold() { return } - connectionManager = &ConnectionManager { + connectionManager = &ConnectionManager{ connections: make(map[string]struct{}), threshold: threshold, } @@ -169,7 +169,7 @@ func (c *Client) Run(stream gnmipb.GNMI_SubscribeServer) (err error) { return grpc.Errorf(codes.Unimplemented, "Empty target data not supported") } else if target == "OTHERS" { dc, err = sdc.NewNonDbClient(paths, prefix) - } else if ((target == "EVENTS") && (mode == gnmipb.SubscriptionList_STREAM)) { + } else if (target == "EVENTS") && (mode == gnmipb.SubscriptionList_STREAM) { dc, err = sdc.NewEventClient(paths, prefix, c.logLevel) } else if _, ok, _, _ := sdc.IsTargetDb(target); ok { dc, err = sdc.NewDbClient(paths, prefix) @@ -297,7 +297,7 @@ func (c *Client) send(stream gnmipb.GNMI_SubscribeServer, dc sdc.Client) error { c.errors++ return err } - val = &v; + val = &v default: log.V(1).Infof("Unknown data type %v for %s in queue", items[0], c) c.errors++ diff --git a/gnmi_server/connection_manager.go b/gnmi_server/connection_manager.go index cc0b2cde..f4406f8f 100644 --- a/gnmi_server/connection_manager.go +++ b/gnmi_server/connection_manager.go @@ -1,11 +1,11 @@ package gnmi import ( - "sync" - "time" + log "github.com/golang/glog" "net" "regexp" - log "github.com/golang/glog" + "sync" + "time" "github.com/go-redis/redis" sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" @@ -16,9 +16,9 @@ const table = "TELEMETRY_CONNECTIONS" var rclient *redis.Client type ConnectionManager struct { - connections map[string]struct{} - mu sync.RWMutex - threshold int + connections map[string]struct{} + mu sync.RWMutex + threshold int } func (cm *ConnectionManager) GetThreshold() int { @@ -57,7 +57,7 @@ func (cm *ConnectionManager) PrepareRedis() { } func (cm *ConnectionManager) Add(addr net.Addr, query string) (string, bool) { - cm.mu.RLock() // reading + cm.mu.RLock() // reading if len(cm.connections) >= cm.threshold && cm.threshold != 0 { // 0 is defined as no threshold log.V(1).Infof("Cannot add another client connection as threshold is already at limit") cm.mu.RUnlock() @@ -73,7 +73,7 @@ func (cm *ConnectionManager) Add(addr net.Addr, query string) (string, bool) { return key, true } -func (cm *ConnectionManager) Remove(key string) (bool) { +func (cm *ConnectionManager) Remove(key string) bool { cm.mu.RLock() // reading _, exists := cm.connections[key] cm.mu.RUnlock() diff --git a/gnmi_server/constants_native.go b/gnmi_server/constants_native.go index 450899c9..fef2677d 100644 --- a/gnmi_server/constants_native.go +++ b/gnmi_server/constants_native.go @@ -1,5 +1,6 @@ -// +build !gnmi_native_write - -package gnmi - -const ENABLE_NATIVE_WRITE = false +//go:build !gnmi_native_write +// +build !gnmi_native_write + +package gnmi + +const ENABLE_NATIVE_WRITE = false diff --git a/gnmi_server/constants_native_write.go b/gnmi_server/constants_native_write.go index 8bb98fae..162cc6b0 100644 --- a/gnmi_server/constants_native_write.go +++ b/gnmi_server/constants_native_write.go @@ -1,5 +1,6 @@ -// +build gnmi_native_write - -package gnmi - -const ENABLE_NATIVE_WRITE = true +//go:build gnmi_native_write +// +build gnmi_native_write + +package gnmi + +const ENABLE_NATIVE_WRITE = true diff --git a/gnmi_server/constants_translib.go b/gnmi_server/constants_translib.go index d3ef6e75..21cd7610 100644 --- a/gnmi_server/constants_translib.go +++ b/gnmi_server/constants_translib.go @@ -1,5 +1,6 @@ -// +build !gnmi_translib_write - -package gnmi - -const ENABLE_TRANSLIB_WRITE = false +//go:build !gnmi_translib_write +// +build !gnmi_translib_write + +package gnmi + +const ENABLE_TRANSLIB_WRITE = false diff --git a/gnmi_server/constants_translib_write.go b/gnmi_server/constants_translib_write.go index 662a9930..26a7b5ca 100644 --- a/gnmi_server/constants_translib_write.go +++ b/gnmi_server/constants_translib_write.go @@ -1,5 +1,6 @@ -// +build gnmi_translib_write - -package gnmi - -const ENABLE_TRANSLIB_WRITE = true +//go:build gnmi_translib_write +// +build gnmi_translib_write + +package gnmi + +const ENABLE_TRANSLIB_WRITE = true diff --git a/gnmi_server/gnoi.go b/gnmi_server/gnoi.go index 8bd96536..a44c1347 100644 --- a/gnmi_server/gnoi.go +++ b/gnmi_server/gnoi.go @@ -2,22 +2,22 @@ package gnmi import ( "context" + "encoding/json" "errors" - "os" - gnoi_system_pb "github.com/openconfig/gnoi/system" + jwt "github.com/dgrijalva/jwt-go" log "github.com/golang/glog" - "time" + gnoi_system_pb "github.com/openconfig/gnoi/system" + "github.com/sonic-net/sonic-gnmi/common_utils" spb "github.com/sonic-net/sonic-gnmi/proto/gnoi" - transutil "github.com/sonic-net/sonic-gnmi/transl_utils" - io "io/ioutil" - ssc "github.com/sonic-net/sonic-gnmi/sonic_service_client" spb_jwt "github.com/sonic-net/sonic-gnmi/proto/gnoi/jwt" - "github.com/sonic-net/sonic-gnmi/common_utils" - "google.golang.org/grpc/status" + ssc "github.com/sonic-net/sonic-gnmi/sonic_service_client" + transutil "github.com/sonic-net/sonic-gnmi/transl_utils" "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + io "io/ioutil" + "os" "os/user" - "encoding/json" - jwt "github.com/dgrijalva/jwt-go" + "time" ) func RebootSystem(fileName string) error { @@ -128,12 +128,11 @@ func (srv *Server) Authenticate(ctx context.Context, req *spb_jwt.AuthenticateRe // } log.V(1).Info("gNOI: Sonic Authenticate") - if !srv.config.UserAuth.Enabled("jwt") { return nil, status.Errorf(codes.Unimplemented, "") } auth_success, _ := UserPwAuth(req.Username, req.Password) - if auth_success { + if auth_success { usr, err := user.Lookup(req.Username) if err == nil { roles, err := GetUserRoles(usr) @@ -141,7 +140,7 @@ func (srv *Server) Authenticate(ctx context.Context, req *spb_jwt.AuthenticateRe return &spb_jwt.AuthenticateResponse{Token: tokenResp(req.Username, roles)}, nil } } - + } return nil, status.Errorf(codes.PermissionDenied, "Invalid Username or Password") @@ -169,41 +168,40 @@ func (srv *Server) Refresh(ctx context.Context, req *spb_jwt.RefreshRequest) (*s if time.Unix(claims.ExpiresAt, 0).Sub(time.Now()) > JwtRefreshInt { return nil, status.Errorf(codes.InvalidArgument, "Invalid JWT Token") } - + return &spb_jwt.RefreshResponse{Token: tokenResp(claims.Username, claims.Roles)}, nil } func (srv *Server) ClearNeighbors(ctx context.Context, req *spb.ClearNeighborsRequest) (*spb.ClearNeighborsResponse, error) { - ctx, err := authenticate(srv.config.UserAuth, ctx) - if err != nil { - return nil, err - } - log.V(1).Info("gNOI: Sonic ClearNeighbors") - log.V(1).Info("Request: ", req) - - resp := &spb.ClearNeighborsResponse{ - Output: &spb.ClearNeighborsResponse_Output { - }, - } - - reqstr, err := json.Marshal(req) - if err != nil { - return nil, status.Error(codes.Unknown, err.Error()) - } - - jsresp, err:= transutil.TranslProcessAction("/sonic-neighbor:clear-neighbors", []byte(reqstr), ctx) - - if err != nil { - return nil, status.Error(codes.Unknown, err.Error()) - } - - err = json.Unmarshal(jsresp, resp) - if err != nil { - return nil, status.Error(codes.Unknown, err.Error()) - } - - return resp, nil + ctx, err := authenticate(srv.config.UserAuth, ctx) + if err != nil { + return nil, err + } + log.V(1).Info("gNOI: Sonic ClearNeighbors") + log.V(1).Info("Request: ", req) + + resp := &spb.ClearNeighborsResponse{ + Output: &spb.ClearNeighborsResponse_Output{}, + } + + reqstr, err := json.Marshal(req) + if err != nil { + return nil, status.Error(codes.Unknown, err.Error()) + } + + jsresp, err := transutil.TranslProcessAction("/sonic-neighbor:clear-neighbors", []byte(reqstr), ctx) + + if err != nil { + return nil, status.Error(codes.Unknown, err.Error()) + } + + err = json.Unmarshal(jsresp, resp) + if err != nil { + return nil, status.Error(codes.Unknown, err.Error()) + } + + return resp, nil } func (srv *Server) CopyConfig(ctx context.Context, req *spb.CopyConfigRequest) (*spb.CopyConfigResponse, error) { @@ -212,28 +210,26 @@ func (srv *Server) CopyConfig(ctx context.Context, req *spb.CopyConfigRequest) ( return nil, err } log.V(1).Info("gNOI: Sonic CopyConfig") - - resp := &spb.CopyConfigResponse{ - Output: &spb.SonicOutput { - }, + resp := &spb.CopyConfigResponse{ + Output: &spb.SonicOutput{}, } - + reqstr, err := json.Marshal(req) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - jsresp, err:= transutil.TranslProcessAction("/sonic-config-mgmt:copy", []byte(reqstr), ctx) + jsresp, err := transutil.TranslProcessAction("/sonic-config-mgmt:copy", []byte(reqstr), ctx) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - + err = json.Unmarshal(jsresp, resp) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - + return resp, nil } @@ -243,29 +239,26 @@ func (srv *Server) ShowTechsupport(ctx context.Context, req *spb.TechsupportRequ return nil, err } log.V(1).Info("gNOI: Sonic ShowTechsupport") - - resp := &spb.TechsupportResponse{ - Output: &spb.TechsupportResponse_Output { - }, + resp := &spb.TechsupportResponse{ + Output: &spb.TechsupportResponse_Output{}, } reqstr, err := json.Marshal(req) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - jsresp, err:= transutil.TranslProcessAction("/sonic-show-techsupport:sonic-show-techsupport-info", []byte(reqstr), ctx) + jsresp, err := transutil.TranslProcessAction("/sonic-show-techsupport:sonic-show-techsupport-info", []byte(reqstr), ctx) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - + err = json.Unmarshal(jsresp, resp) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - - + return resp, nil } @@ -275,29 +268,26 @@ func (srv *Server) ImageInstall(ctx context.Context, req *spb.ImageInstallReques return nil, err } log.V(1).Info("gNOI: Sonic ImageInstall") - - resp := &spb.ImageInstallResponse{ - Output: &spb.SonicOutput { - }, + resp := &spb.ImageInstallResponse{ + Output: &spb.SonicOutput{}, } reqstr, err := json.Marshal(req) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - jsresp, err:= transutil.TranslProcessAction("/sonic-image-management:image-install", []byte(reqstr), ctx) + jsresp, err := transutil.TranslProcessAction("/sonic-image-management:image-install", []byte(reqstr), ctx) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - + err = json.Unmarshal(jsresp, resp) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - return resp, nil } @@ -307,11 +297,9 @@ func (srv *Server) ImageRemove(ctx context.Context, req *spb.ImageRemoveRequest) return nil, err } log.V(1).Info("gNOI: Sonic ImageRemove") - - resp := &spb.ImageRemoveResponse{ - Output: &spb.SonicOutput { - }, + resp := &spb.ImageRemoveResponse{ + Output: &spb.SonicOutput{}, } reqstr, err := json.Marshal(req) @@ -319,17 +307,16 @@ func (srv *Server) ImageRemove(ctx context.Context, req *spb.ImageRemoveRequest) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - jsresp, err:= transutil.TranslProcessAction("/sonic-image-management:image-remove", []byte(reqstr), ctx) + jsresp, err := transutil.TranslProcessAction("/sonic-image-management:image-remove", []byte(reqstr), ctx) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - + err = json.Unmarshal(jsresp, resp) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - return resp, nil } @@ -339,11 +326,9 @@ func (srv *Server) ImageDefault(ctx context.Context, req *spb.ImageDefaultReques return nil, err } log.V(1).Info("gNOI: Sonic ImageDefault") - - resp := &spb.ImageDefaultResponse{ - Output: &spb.SonicOutput { - }, + resp := &spb.ImageDefaultResponse{ + Output: &spb.SonicOutput{}, } reqstr, err := json.Marshal(req) @@ -351,7 +336,7 @@ func (srv *Server) ImageDefault(ctx context.Context, req *spb.ImageDefaultReques if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } - jsresp, err:= transutil.TranslProcessAction("/sonic-image-management:image-default", []byte(reqstr), ctx) + jsresp, err := transutil.TranslProcessAction("/sonic-image-management:image-default", []byte(reqstr), ctx) if err != nil { return nil, status.Error(codes.Unknown, err.Error()) } @@ -361,6 +346,5 @@ func (srv *Server) ImageDefault(ctx context.Context, req *spb.ImageDefaultReques return nil, status.Error(codes.Unknown, err.Error()) } - return resp, nil } diff --git a/gnmi_server/jwtAuth.go b/gnmi_server/jwtAuth.go index 0b6c2187..2ac4ecb3 100644 --- a/gnmi_server/jwtAuth.go +++ b/gnmi_server/jwtAuth.go @@ -1,15 +1,15 @@ package gnmi import ( - "github.com/sonic-net/sonic-gnmi/common_utils" "crypto/rand" jwt "github.com/dgrijalva/jwt-go" "github.com/golang/glog" + "github.com/sonic-net/sonic-gnmi/common_utils" + spb "github.com/sonic-net/sonic-gnmi/proto/gnoi/jwt" "golang.org/x/net/context" "google.golang.org/grpc/codes" "google.golang.org/grpc/metadata" "google.golang.org/grpc/status" - spb "github.com/sonic-net/sonic-gnmi/proto/gnoi/jwt" "time" ) diff --git a/gnmi_server/pamAuth.go b/gnmi_server/pamAuth.go index 73ce1ed5..e69b0d25 100644 --- a/gnmi_server/pamAuth.go +++ b/gnmi_server/pamAuth.go @@ -1,10 +1,10 @@ package gnmi import ( - "github.com/sonic-net/sonic-gnmi/common_utils" "errors" "github.com/golang/glog" "github.com/msteinert/pam" + "github.com/sonic-net/sonic-gnmi/common_utils" "golang.org/x/crypto/ssh" "os/user" ) @@ -14,7 +14,7 @@ type UserCredential struct { Password string } -//PAM conversation handler. +// PAM conversation handler. func (u UserCredential) PAMConvHandler(s pam.Style, msg string) (string, error) { switch s { diff --git a/gnmi_server/server.go b/gnmi_server/server.go index 4ae32c18..553aad5a 100644 --- a/gnmi_server/server.go +++ b/gnmi_server/server.go @@ -5,16 +5,16 @@ import ( "errors" "fmt" "github.com/Azure/sonic-mgmt-common/translib" - "github.com/sonic-net/sonic-gnmi/common_utils" - spb "github.com/sonic-net/sonic-gnmi/proto" - spb_gnoi "github.com/sonic-net/sonic-gnmi/proto/gnoi" - spb_jwt_gnoi "github.com/sonic-net/sonic-gnmi/proto/gnoi/jwt" - sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" log "github.com/golang/glog" "github.com/golang/protobuf/proto" gnmipb "github.com/openconfig/gnmi/proto/gnmi" gnmi_extpb "github.com/openconfig/gnmi/proto/gnmi_ext" gnoi_system_pb "github.com/openconfig/gnoi/system" + "github.com/sonic-net/sonic-gnmi/common_utils" + spb "github.com/sonic-net/sonic-gnmi/proto" + spb_gnoi "github.com/sonic-net/sonic-gnmi/proto/gnoi" + spb_jwt_gnoi "github.com/sonic-net/sonic-gnmi/proto/gnoi/jwt" + sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" "golang.org/x/net/context" "google.golang.org/grpc" "google.golang.org/grpc/codes" @@ -50,14 +50,14 @@ type AuthTypes map[string]bool type Config struct { // Port for the Server to listen on. If 0 or unset the Server will pick a port // for this Server. - Port int64 - LogLevel int - Threshold int - UserAuth AuthTypes + Port int64 + LogLevel int + Threshold int + UserAuth AuthTypes EnableTranslibWrite bool - EnableNativeWrite bool - ZmqAddress string - IdleConnDuration int + EnableNativeWrite bool + ZmqAddress string + IdleConnDuration int } var AuthLock sync.Mutex @@ -160,7 +160,7 @@ func NewServer(config *Config, opts []grpc.ServerOption) (*Server, error) { if srv.config.EnableTranslibWrite || srv.config.EnableNativeWrite { gnoi_system_pb.RegisterSystemServer(srv.s, srv) } - if srv.config.EnableTranslibWrite { + if srv.config.EnableTranslibWrite { spb_gnoi.RegisterSonicServiceServer(srv.s, srv) } spb_gnoi.RegisterDebugServer(srv.s, srv) @@ -591,7 +591,7 @@ func ReqFromMasterEnabledMA(req *gnmipb.SetRequest, masterEID *uint128) error { // Role will be implemented later. return status.Errorf(codes.Unimplemented, "MA: Role is not implemented") } - + reqEID = uint128{High: ma.ElectionId.High, Low: ma.ElectionId.Low} // Use the election ID that is in the last extension, so, no 'break' here. } diff --git a/gnmi_server/server_test.go b/gnmi_server/server_test.go index 6af1d58e..b14aa13e 100644 --- a/gnmi_server/server_test.go +++ b/gnmi_server/server_test.go @@ -5,25 +5,25 @@ package gnmi import ( "crypto/tls" "encoding/json" - "path/filepath" "flag" "fmt" -"sync" + "path/filepath" "strings" + "sync" "unsafe" - testcert "github.com/sonic-net/sonic-gnmi/testdata/tls" "github.com/go-redis/redis" "github.com/golang/protobuf/proto" + testcert "github.com/sonic-net/sonic-gnmi/testdata/tls" "io/ioutil" "os" "os/exec" "os/user" "reflect" + "runtime" "testing" "time" - "runtime" "github.com/kylelemons/godebug/pretty" "github.com/openconfig/gnmi/client" @@ -36,26 +36,25 @@ import ( "google.golang.org/grpc" "google.golang.org/grpc/codes" "google.golang.org/grpc/credentials" - "google.golang.org/grpc/status" "google.golang.org/grpc/keepalive" + "google.golang.org/grpc/status" // Register supported client types. + "github.com/Workiva/go-datastructures/queue" + "github.com/agiledragon/gomonkey/v2" + linuxproc "github.com/c9s/goprocinfo/linux" + "github.com/godbus/dbus/v5" + gclient "github.com/jipanyang/gnmi/client/gnmi" + "github.com/jipanyang/gnxi/utils/xpath" + cacheclient "github.com/openconfig/gnmi/client" + gnmipb "github.com/openconfig/gnmi/proto/gnmi" + gnoi_system_pb "github.com/openconfig/gnoi/system" + "github.com/sonic-net/sonic-gnmi/common_utils" spb "github.com/sonic-net/sonic-gnmi/proto" sgpb "github.com/sonic-net/sonic-gnmi/proto/gnoi" - gnmipb "github.com/openconfig/gnmi/proto/gnmi" sdc "github.com/sonic-net/sonic-gnmi/sonic_data_client" sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" - "github.com/Workiva/go-datastructures/queue" - linuxproc "github.com/c9s/goprocinfo/linux" - "github.com/sonic-net/sonic-gnmi/common_utils" "github.com/sonic-net/sonic-gnmi/test_utils" - gclient "github.com/jipanyang/gnmi/client/gnmi" - "github.com/jipanyang/gnxi/utils/xpath" - gnoi_system_pb "github.com/openconfig/gnoi/system" - "github.com/agiledragon/gomonkey/v2" - "github.com/godbus/dbus/v5" - cacheclient "github.com/openconfig/gnmi/client" - ) var clientTypes = []string{gclient.Type} @@ -163,7 +162,7 @@ func createRejectServer(t *testing.T, port int64) *Server { } opts := []grpc.ServerOption{grpc.Creds(credentials.NewTLS(tlsCfg))} - cfg := &Config{Port: port, EnableTranslibWrite: true, Threshold: 2} + cfg := &Config{Port: port, EnableTranslibWrite: true, Threshold: 2} s, err := NewServer(cfg, opts) if err != nil { t.Fatalf("Failed to create gNMI server: %v", err) @@ -356,7 +355,7 @@ func runTestSet(t *testing.T, ctx context.Context, gClient pb.GNMIClient, pathTa runTestSetRaw(t, ctx, gClient, req, wantRetCode) } -func runTestSetRaw(t *testing.T, ctx context.Context, gClient pb.GNMIClient, req *pb.SetRequest, +func runTestSetRaw(t *testing.T, ctx context.Context, gClient pb.GNMIClient, req *pb.SetRequest, wantRetCode codes.Code) { t.Helper() @@ -799,7 +798,7 @@ func createEventsQuery(t *testing.T, paths ...string) client.Query { func createStateDbQueryOnChangeMode(t *testing.T, paths ...string) client.Query { return createQueryOrFail(t, - pb.SubscriptionList_STREAM, + pb.SubscriptionList_STREAM, "STATE_DB", []subscriptionQuery{ { @@ -953,17 +952,17 @@ func TestGnmiSet(t *testing.T) { operation: Delete, }, { - desc: "Set OC Interface MTU", - pathTarget: "OC_YANG", + desc: "Set OC Interface MTU", + pathTarget: "OC_YANG", textPbPath: pathToPb("openconfig-interfaces:interfaces/interface[name=Ethernet4]/config"), attributeData: "../testdata/set_interface_mtu.json", wantRetCode: codes.OK, operation: Update, }, { - desc: "Set OC Interface IP", - pathTarget: "OC_YANG", - textPbPath: pathToPb("/openconfig-interfaces:interfaces/interface[name=Ethernet4]/subinterfaces/subinterface[index=0]/openconfig-if-ip:ipv4"), + desc: "Set OC Interface IP", + pathTarget: "OC_YANG", + textPbPath: pathToPb("/openconfig-interfaces:interfaces/interface[name=Ethernet4]/subinterfaces/subinterface[index=0]/openconfig-if-ip:ipv4"), attributeData: "../testdata/set_interface_ipv4.json", wantRetCode: codes.OK, operation: Update, @@ -990,9 +989,9 @@ func TestGnmiSet(t *testing.T) { valTest: false, }, { - desc: "Set OC Interface IPv6 (unprefixed path)", - pathTarget: "OC_YANG", - textPbPath: pathToPb("/interfaces/interface[name=Ethernet0]/subinterfaces/subinterface[index=0]/ipv6/addresses/address"), + desc: "Set OC Interface IPv6 (unprefixed path)", + pathTarget: "OC_YANG", + textPbPath: pathToPb("/interfaces/interface[name=Ethernet0]/subinterfaces/subinterface[index=0]/ipv6/addresses/address"), attributeData: `{"address": [{"ip": "150::1","config": {"ip": "150::1","prefix-length": 80}}]}`, wantRetCode: codes.OK, operation: Update, @@ -1005,13 +1004,13 @@ func TestGnmiSet(t *testing.T) { operation: Delete, }, { - desc: "Create ACL (unprefixed path)", - pathTarget: "OC_YANG", - textPbPath: pathToPb("/acl/acl-sets/acl-set"), + desc: "Create ACL (unprefixed path)", + pathTarget: "OC_YANG", + textPbPath: pathToPb("/acl/acl-sets/acl-set"), attributeData: `{"acl-set": [{"name": "A001", "type": "ACL_IPV4", "config": {"name": "A001", "type": "ACL_IPV4", "description": "hello, world!"}}]}`, - wantRetCode: codes.OK, - operation: Update, + wantRetCode: codes.OK, + operation: Update, }, { desc: "Verify Create ACL", @@ -1058,7 +1057,7 @@ func TestGnmiSet(t *testing.T) { t.Run(td.desc, func(t *testing.T) { runTestGet(t, ctx, gClient, td.pathTarget, td.textPbPath, td.wantRetCode, td.wantRespVal, td.valTest) }) - t.Run(td.desc + " (unprefixed path)", func(t *testing.T) { + t.Run(td.desc+" (unprefixed path)", func(t *testing.T) { p := removeModulePrefixFromPathPb(t, td.textPbPath) runTestGet(t, ctx, gClient, td.pathTarget, p, td.wantRetCode, td.wantRespVal, td.valTest) }) @@ -1411,9 +1410,9 @@ func runGnmiTestGet(t *testing.T, namespace string) { wantRetCode: codes.OK, wantRespVal: []byte(`{"test_field": "test_value"}`), }, { - desc: "Invalid DBKey of length 1", - pathTarget: stateDBPath, - textPbPath: ``, + desc: "Invalid DBKey of length 1", + pathTarget: stateDBPath, + textPbPath: ``, valTest: true, wantRetCode: codes.NotFound, }, @@ -1820,7 +1819,7 @@ func runTestSubscribe(t *testing.T, namespace string) { generateIntervals bool } - tests := []TestExec { + tests := []TestExec{ { desc: "stream query for table COUNTERS_PORT_NAME_MAP with new test_field field", q: createCountersDbQueryOnChangeMode(t, "COUNTERS_PORT_NAME_MAP"), @@ -2663,7 +2662,7 @@ func runTestSubscribe(t *testing.T, namespace string) { mutexGotNoti.Unlock() return nil } - go func(t2 TestExec) { + go func(t2 TestExec) { defer wg.Done() err := c.Subscribe(context.Background(), q) if t2.wantSubErr != nil && t2.wantSubErr.Error() != err.Error() { @@ -3025,7 +3024,7 @@ func TestBulkSet(t *testing.T) { t.Run("Invalid Replace Path", func(t *testing.T) { req := &pb.SetRequest{ - Delete: []*pb.Path{aclPath1, aclPath2}, + Delete: []*pb.Path{aclPath1, aclPath2}, Replace: []*pb.Update{ newPbUpdate("interface[name=Ethernet0]/config/mtu", `{"mtu": 9104}`), }} @@ -3047,23 +3046,23 @@ func newPbUpdate(path, value string) *pb.Update { v := &pb.TypedValue_JsonIetfVal{JsonIetfVal: extractJSON(value)} return &pb.Update{ Path: p, - Val: &pb.TypedValue{Value: v}, + Val: &pb.TypedValue{Value: v}, } } type loginCreds struct { - Username, Password string + Username, Password string } func (c *loginCreds) GetRequestMetadata(context.Context, ...string) (map[string]string, error) { - return map[string]string{ - "username": c.Username, - "password": c.Password, - }, nil + return map[string]string{ + "username": c.Username, + "password": c.Password, + }, nil } func (c *loginCreds) RequireTransportSecurity() bool { - return true + return true } func TestAuthCapabilities(t *testing.T) { @@ -3103,659 +3102,659 @@ func TestAuthCapabilities(t *testing.T) { } func TestTableKeyOnDeletion(t *testing.T) { - s := createKeepAliveServer(t, 8081) - go runServer(t, s) - defer s.s.Stop() - - fileName := "../testdata/NEIGH_STATE_TABLE_MAP.txt" - neighStateTableByte, err := ioutil.ReadFile(fileName) - if err != nil { - t.Fatalf("read file %v err: %v", fileName, err) - } - var neighStateTableJson interface{} - json.Unmarshal(neighStateTableByte, &neighStateTableJson) - - fileName = "../testdata/NEIGH_STATE_TABLE_key_deletion_57.txt" - neighStateTableDeletedByte57, err := ioutil.ReadFile(fileName) - if err != nil { - t.Fatalf("read file %v err: %v", fileName, err) - } - var neighStateTableDeletedJson57 interface{} - json.Unmarshal(neighStateTableDeletedByte57, &neighStateTableDeletedJson57) - - fileName = "../testdata/NEIGH_STATE_TABLE_MAP_2.txt" - neighStateTableByteTwo, err := ioutil.ReadFile(fileName) - if err != nil { - t.Fatalf("read file %v err: %v", fileName, err) - } - var neighStateTableJsonTwo interface{} - json.Unmarshal(neighStateTableByteTwo, &neighStateTableJsonTwo) - - fileName = "../testdata/NEIGH_STATE_TABLE_key_deletion_59.txt" - neighStateTableDeletedByte59, err := ioutil.ReadFile(fileName) - if err != nil { - t.Fatalf("read file %v err: %v", fileName, err) - } - var neighStateTableDeletedJson59 interface{} - json.Unmarshal(neighStateTableDeletedByte59, &neighStateTableDeletedJson59) - - fileName = "../testdata/NEIGH_STATE_TABLE_key_deletion_61.txt" - neighStateTableDeletedByte61, err := ioutil.ReadFile(fileName) - if err != nil { - t.Fatalf("read file %v err: %v", fileName, err) - } - var neighStateTableDeletedJson61 interface{} - json.Unmarshal(neighStateTableDeletedByte61, &neighStateTableDeletedJson61) - - namespace, _ := sdcfg.GetDbDefaultNamespace() - rclient := getRedisClientN(t, 6, namespace) - defer rclient.Close() - prepareStateDb(t, namespace) - - tests := []struct { - desc string - q client.Query - wantNoti []client.Notification - paths []string - }{ - { - desc: "Testing deletion of NEIGH_STATE_TABLE:10.0.0.57", - q: createStateDbQueryOnChangeMode(t, "NEIGH_STATE_TABLE"), - wantNoti: []client.Notification { - client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableJson}, - client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableDeletedJson57}, - }, - paths: []string { - "NEIGH_STATE_TABLE|10.0.0.57", - }, - }, - { - desc: "Testing deletion of NEIGH_STATE_TABLE:10.0.0.59 and NEIGH_STATE_TABLE 10.0.0.61", - q: createStateDbQueryOnChangeMode(t, "NEIGH_STATE_TABLE"), - wantNoti: []client.Notification { - client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableJsonTwo}, - client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableDeletedJson59}, - client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableDeletedJson61}, - }, - paths: []string { - "NEIGH_STATE_TABLE|10.0.0.59", - "NEIGH_STATE_TABLE|10.0.0.61", - }, - }, - } - - var mutexNoti sync.RWMutex - var mutexPaths sync.Mutex - for _, tt := range tests { - t.Run(tt.desc, func(t *testing.T) { - q := tt.q - q.Addrs = []string{"127.0.0.1:8081"} - c := client.New() - defer c.Close() - var gotNoti []client.Notification - q.NotificationHandler = func(n client.Notification) error { - if nn, ok := n.(client.Update); ok { - nn.TS = time.Unix(0, 200) - mutexNoti.Lock() - currentNoti := gotNoti - mutexNoti.Unlock() - - mutexNoti.RLock() - gotNoti = append(currentNoti, nn) - mutexNoti.RUnlock() - } - return nil - } - - go func() { - c.Subscribe(context.Background(), q) - }() - - time.Sleep(time.Millisecond * 500) // half a second for subscribe request to sync - - mutexPaths.Lock() - paths := tt.paths - mutexPaths.Unlock() - - rclient.Del(paths...) - - time.Sleep(time.Millisecond * 1500) - - mutexNoti.Lock() - if diff := pretty.Compare(tt.wantNoti, gotNoti); diff != "" { - t.Log("\n Want: \n", tt.wantNoti) - t.Log("\n Got : \n", gotNoti) - t.Errorf("unexpected updates:\n%s", diff) - } - mutexNoti.Unlock() - }) - } + s := createKeepAliveServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + fileName := "../testdata/NEIGH_STATE_TABLE_MAP.txt" + neighStateTableByte, err := ioutil.ReadFile(fileName) + if err != nil { + t.Fatalf("read file %v err: %v", fileName, err) + } + var neighStateTableJson interface{} + json.Unmarshal(neighStateTableByte, &neighStateTableJson) + + fileName = "../testdata/NEIGH_STATE_TABLE_key_deletion_57.txt" + neighStateTableDeletedByte57, err := ioutil.ReadFile(fileName) + if err != nil { + t.Fatalf("read file %v err: %v", fileName, err) + } + var neighStateTableDeletedJson57 interface{} + json.Unmarshal(neighStateTableDeletedByte57, &neighStateTableDeletedJson57) + + fileName = "../testdata/NEIGH_STATE_TABLE_MAP_2.txt" + neighStateTableByteTwo, err := ioutil.ReadFile(fileName) + if err != nil { + t.Fatalf("read file %v err: %v", fileName, err) + } + var neighStateTableJsonTwo interface{} + json.Unmarshal(neighStateTableByteTwo, &neighStateTableJsonTwo) + + fileName = "../testdata/NEIGH_STATE_TABLE_key_deletion_59.txt" + neighStateTableDeletedByte59, err := ioutil.ReadFile(fileName) + if err != nil { + t.Fatalf("read file %v err: %v", fileName, err) + } + var neighStateTableDeletedJson59 interface{} + json.Unmarshal(neighStateTableDeletedByte59, &neighStateTableDeletedJson59) + + fileName = "../testdata/NEIGH_STATE_TABLE_key_deletion_61.txt" + neighStateTableDeletedByte61, err := ioutil.ReadFile(fileName) + if err != nil { + t.Fatalf("read file %v err: %v", fileName, err) + } + var neighStateTableDeletedJson61 interface{} + json.Unmarshal(neighStateTableDeletedByte61, &neighStateTableDeletedJson61) + + namespace, _ := sdcfg.GetDbDefaultNamespace() + rclient := getRedisClientN(t, 6, namespace) + defer rclient.Close() + prepareStateDb(t, namespace) + + tests := []struct { + desc string + q client.Query + wantNoti []client.Notification + paths []string + }{ + { + desc: "Testing deletion of NEIGH_STATE_TABLE:10.0.0.57", + q: createStateDbQueryOnChangeMode(t, "NEIGH_STATE_TABLE"), + wantNoti: []client.Notification{ + client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableJson}, + client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableDeletedJson57}, + }, + paths: []string{ + "NEIGH_STATE_TABLE|10.0.0.57", + }, + }, + { + desc: "Testing deletion of NEIGH_STATE_TABLE:10.0.0.59 and NEIGH_STATE_TABLE 10.0.0.61", + q: createStateDbQueryOnChangeMode(t, "NEIGH_STATE_TABLE"), + wantNoti: []client.Notification{ + client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableJsonTwo}, + client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableDeletedJson59}, + client.Update{Path: []string{"NEIGH_STATE_TABLE"}, TS: time.Unix(0, 200), Val: neighStateTableDeletedJson61}, + }, + paths: []string{ + "NEIGH_STATE_TABLE|10.0.0.59", + "NEIGH_STATE_TABLE|10.0.0.61", + }, + }, + } + + var mutexNoti sync.RWMutex + var mutexPaths sync.Mutex + for _, tt := range tests { + t.Run(tt.desc, func(t *testing.T) { + q := tt.q + q.Addrs = []string{"127.0.0.1:8081"} + c := client.New() + defer c.Close() + var gotNoti []client.Notification + q.NotificationHandler = func(n client.Notification) error { + if nn, ok := n.(client.Update); ok { + nn.TS = time.Unix(0, 200) + mutexNoti.Lock() + currentNoti := gotNoti + mutexNoti.Unlock() + + mutexNoti.RLock() + gotNoti = append(currentNoti, nn) + mutexNoti.RUnlock() + } + return nil + } + + go func() { + c.Subscribe(context.Background(), q) + }() + + time.Sleep(time.Millisecond * 500) // half a second for subscribe request to sync + + mutexPaths.Lock() + paths := tt.paths + mutexPaths.Unlock() + + rclient.Del(paths...) + + time.Sleep(time.Millisecond * 1500) + + mutexNoti.Lock() + if diff := pretty.Compare(tt.wantNoti, gotNoti); diff != "" { + t.Log("\n Want: \n", tt.wantNoti) + t.Log("\n Got : \n", gotNoti) + t.Errorf("unexpected updates:\n%s", diff) + } + mutexNoti.Unlock() + }) + } } func TestCPUUtilization(t *testing.T) { - mock := gomonkey.ApplyFunc(sdc.PollStats, func() { - var i uint64 - for i = 0; i < 3000; i++ { - sdc.WriteStatsToBuffer(&linuxproc.Stat{}) - } - }) - - defer mock.Reset() - s := createServer(t, 8081) - go runServer(t, s) - defer s.s.Stop() - - tests := []struct { - desc string - q client.Query - want []client.Notification - poll int - }{ - { - desc: "poll query for CPU Utilization", - poll: 10, - q: client.Query{ - Target: "OTHERS", - Type: client.Poll, - Queries: []client.Path{{"platform", "cpu"}}, - TLS: &tls.Config{InsecureSkipVerify: true}, - }, - want: []client.Notification{ - client.Connected{}, - client.Sync{}, - }, - }, - } - - for _, tt := range tests { - t.Run(tt.desc, func(t *testing.T) { - q := tt.q - q.Addrs = []string{"127.0.0.1:8081"} - c := client.New() - var gotNoti []client.Notification - q.NotificationHandler = func(n client.Notification) error { - if nn, ok := n.(client.Update); ok { - nn.TS = time.Unix(0, 200) - gotNoti = append(gotNoti, nn) - } else { - gotNoti = append(gotNoti, n) - } - return nil - } - - wg := new(sync.WaitGroup) - wg.Add(1) - - go func() { - defer wg.Done() - if err := c.Subscribe(context.Background(), q); err != nil { - t.Errorf("c.Subscribe(): got error %v, expected nil", err) - } - }() - - wg.Wait() - - for i := 0; i < tt.poll; i++ { - if err := c.Poll(); err != nil { - t.Errorf("c.Poll(): got error %v, expected nil", err) - } - } - - if len(gotNoti) == 0 { - t.Errorf("expected non zero notifications") - } - - c.Close() - }) - } + mock := gomonkey.ApplyFunc(sdc.PollStats, func() { + var i uint64 + for i = 0; i < 3000; i++ { + sdc.WriteStatsToBuffer(&linuxproc.Stat{}) + } + }) + + defer mock.Reset() + s := createServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + tests := []struct { + desc string + q client.Query + want []client.Notification + poll int + }{ + { + desc: "poll query for CPU Utilization", + poll: 10, + q: client.Query{ + Target: "OTHERS", + Type: client.Poll, + Queries: []client.Path{{"platform", "cpu"}}, + TLS: &tls.Config{InsecureSkipVerify: true}, + }, + want: []client.Notification{ + client.Connected{}, + client.Sync{}, + }, + }, + } + + for _, tt := range tests { + t.Run(tt.desc, func(t *testing.T) { + q := tt.q + q.Addrs = []string{"127.0.0.1:8081"} + c := client.New() + var gotNoti []client.Notification + q.NotificationHandler = func(n client.Notification) error { + if nn, ok := n.(client.Update); ok { + nn.TS = time.Unix(0, 200) + gotNoti = append(gotNoti, nn) + } else { + gotNoti = append(gotNoti, n) + } + return nil + } + + wg := new(sync.WaitGroup) + wg.Add(1) + + go func() { + defer wg.Done() + if err := c.Subscribe(context.Background(), q); err != nil { + t.Errorf("c.Subscribe(): got error %v, expected nil", err) + } + }() + + wg.Wait() + + for i := 0; i < tt.poll; i++ { + if err := c.Poll(); err != nil { + t.Errorf("c.Poll(): got error %v, expected nil", err) + } + } + + if len(gotNoti) == 0 { + t.Errorf("expected non zero notifications") + } + + c.Close() + }) + } } func TestClientConnections(t *testing.T) { - s := createRejectServer(t, 8081) - go runServer(t, s) - defer s.s.Stop() - - tests := []struct { - desc string - q client.Query - want []client.Notification - poll int - }{ - { - desc: "Reject OTHERS/proc/uptime", - poll: 10, - q: client.Query{ - Target: "OTHERS", - Type: client.Poll, - Queries: []client.Path{{"proc", "uptime"}}, - TLS: &tls.Config{InsecureSkipVerify: true}, - }, - want: []client.Notification{ - client.Connected{}, - client.Sync{}, - }, - }, - { - desc: "Reject COUNTERS/Ethernet*", - poll: 10, - q: client.Query{ - Target: "COUNTERS_DB", - Type: client.Poll, - Queries: []client.Path{{"COUNTERS", "Ethernet*"}}, - TLS: &tls.Config{InsecureSkipVerify: true}, - }, - want: []client.Notification{ - client.Connected{}, - client.Sync{}, - }, - }, - { - desc: "Reject COUNTERS/Ethernet68", - poll: 10, - q: client.Query{ - Target: "COUNTERS_DB", - Type: client.Poll, - Queries: []client.Path{{"COUNTERS", "Ethernet68"}}, - TLS: &tls.Config{InsecureSkipVerify: true}, - }, - want: []client.Notification{ - client.Connected{}, - client.Sync{}, - }, - }, - } - - var clients []*cacheclient.CacheClient - - for i, tt := range tests { - t.Run(tt.desc, func(t *testing.T) { - q := tt.q - q.Addrs = []string{"127.0.0.1:8081"} - var gotNoti []client.Notification - q.NotificationHandler = func(n client.Notification) error { - if nn, ok := n.(client.Update); ok { - nn.TS = time.Unix(0, 200) - gotNoti = append(gotNoti, nn) - } else { - gotNoti = append(gotNoti, n) - } - return nil - } - - wg := new(sync.WaitGroup) - wg.Add(1) - - go func() { - defer wg.Done() - c := client.New() - clients = append(clients, c) - err := c.Subscribe(context.Background(), q) - if err == nil && i == len(tests) - 1 { // reject third - t.Errorf("Expecting rejection message as no connections are allowed") - } - if err != nil && i < len(tests) - 1 { // accept first two - t.Errorf("Expecting accepts for first two connections") - } - }() - - wg.Wait() - }) - } - - for _, cacheClient := range(clients) { - cacheClient.Close() - } + s := createRejectServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + tests := []struct { + desc string + q client.Query + want []client.Notification + poll int + }{ + { + desc: "Reject OTHERS/proc/uptime", + poll: 10, + q: client.Query{ + Target: "OTHERS", + Type: client.Poll, + Queries: []client.Path{{"proc", "uptime"}}, + TLS: &tls.Config{InsecureSkipVerify: true}, + }, + want: []client.Notification{ + client.Connected{}, + client.Sync{}, + }, + }, + { + desc: "Reject COUNTERS/Ethernet*", + poll: 10, + q: client.Query{ + Target: "COUNTERS_DB", + Type: client.Poll, + Queries: []client.Path{{"COUNTERS", "Ethernet*"}}, + TLS: &tls.Config{InsecureSkipVerify: true}, + }, + want: []client.Notification{ + client.Connected{}, + client.Sync{}, + }, + }, + { + desc: "Reject COUNTERS/Ethernet68", + poll: 10, + q: client.Query{ + Target: "COUNTERS_DB", + Type: client.Poll, + Queries: []client.Path{{"COUNTERS", "Ethernet68"}}, + TLS: &tls.Config{InsecureSkipVerify: true}, + }, + want: []client.Notification{ + client.Connected{}, + client.Sync{}, + }, + }, + } + + var clients []*cacheclient.CacheClient + + for i, tt := range tests { + t.Run(tt.desc, func(t *testing.T) { + q := tt.q + q.Addrs = []string{"127.0.0.1:8081"} + var gotNoti []client.Notification + q.NotificationHandler = func(n client.Notification) error { + if nn, ok := n.(client.Update); ok { + nn.TS = time.Unix(0, 200) + gotNoti = append(gotNoti, nn) + } else { + gotNoti = append(gotNoti, n) + } + return nil + } + + wg := new(sync.WaitGroup) + wg.Add(1) + + go func() { + defer wg.Done() + c := client.New() + clients = append(clients, c) + err := c.Subscribe(context.Background(), q) + if err == nil && i == len(tests)-1 { // reject third + t.Errorf("Expecting rejection message as no connections are allowed") + } + if err != nil && i < len(tests)-1 { // accept first two + t.Errorf("Expecting accepts for first two connections") + } + }() + + wg.Wait() + }) + } + + for _, cacheClient := range clients { + cacheClient.Close() + } } func TestConnectionDataSet(t *testing.T) { - s := createServer(t, 8081) - go runServer(t, s) - defer s.s.Stop() - - tests := []struct { - desc string - q client.Query - want []client.Notification - poll int - }{ - { - desc: "poll query for COUNTERS/Ethernet*", - poll: 10, - q: client.Query{ - Target: "COUNTERS_DB", - Type: client.Poll, - Queries: []client.Path{{"COUNTERS", "Ethernet*"}}, - TLS: &tls.Config{InsecureSkipVerify: true}, - }, - want: []client.Notification{ - client.Connected{}, - client.Sync{}, - }, - }, - } - namespace, _ := sdcfg.GetDbDefaultNamespace() - rclient := getRedisClientN(t, 6, namespace) - defer rclient.Close() - - for _, tt := range tests { - prepareStateDb(t, namespace) - t.Run(tt.desc, func(t *testing.T) { - q := tt.q - q.Addrs = []string{"127.0.0.1:8081"} - c := client.New() - - wg := new(sync.WaitGroup) - wg.Add(1) - - go func() { - defer wg.Done() - if err := c.Subscribe(context.Background(), q); err != nil { - t.Errorf("c.Subscribe(): got error %v, expected nil", err) - } - }() - - wg.Wait() - - resultMap, err := rclient.HGetAll("TELEMETRY_CONNECTIONS").Result() - - if resultMap == nil { - t.Errorf("result Map is nil, expected non nil, err: %v", err) - } - if len(resultMap) != 1 { - t.Errorf("result for TELEMETRY_CONNECTIONS should be 1") - } - - for key, _ := range resultMap { - if !strings.Contains(key, "COUNTERS_DB|COUNTERS|Ethernet*") { - t.Errorf("key is expected to contain correct query, received: %s", key) - } - } - - c.Close() - }) - } + s := createServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + tests := []struct { + desc string + q client.Query + want []client.Notification + poll int + }{ + { + desc: "poll query for COUNTERS/Ethernet*", + poll: 10, + q: client.Query{ + Target: "COUNTERS_DB", + Type: client.Poll, + Queries: []client.Path{{"COUNTERS", "Ethernet*"}}, + TLS: &tls.Config{InsecureSkipVerify: true}, + }, + want: []client.Notification{ + client.Connected{}, + client.Sync{}, + }, + }, + } + namespace, _ := sdcfg.GetDbDefaultNamespace() + rclient := getRedisClientN(t, 6, namespace) + defer rclient.Close() + + for _, tt := range tests { + prepareStateDb(t, namespace) + t.Run(tt.desc, func(t *testing.T) { + q := tt.q + q.Addrs = []string{"127.0.0.1:8081"} + c := client.New() + + wg := new(sync.WaitGroup) + wg.Add(1) + + go func() { + defer wg.Done() + if err := c.Subscribe(context.Background(), q); err != nil { + t.Errorf("c.Subscribe(): got error %v, expected nil", err) + } + }() + + wg.Wait() + + resultMap, err := rclient.HGetAll("TELEMETRY_CONNECTIONS").Result() + + if resultMap == nil { + t.Errorf("result Map is nil, expected non nil, err: %v", err) + } + if len(resultMap) != 1 { + t.Errorf("result for TELEMETRY_CONNECTIONS should be 1") + } + + for key, _ := range resultMap { + if !strings.Contains(key, "COUNTERS_DB|COUNTERS|Ethernet*") { + t.Errorf("key is expected to contain correct query, received: %s", key) + } + } + + c.Close() + }) + } } func TestConnectionsKeepAlive(t *testing.T) { - s := createKeepAliveServer(t, 8081) - go runServer(t, s) - defer s.s.Stop() - - tests := []struct { - desc string - q client.Query - want []client.Notification - poll int - }{ - { - desc: "Testing KeepAlive with goroutine count", - poll: 3, - q: client.Query{ - Target: "COUNTERS_DB", - Type: client.Poll, - Queries: []client.Path{{"COUNTERS", "Ethernet*"}}, - TLS: &tls.Config{InsecureSkipVerify: true}, - }, - want: []client.Notification{ - client.Connected{}, - client.Sync{}, - }, - }, - } - for _, tt := range(tests) { - for i := 0; i < 5; i++ { - t.Run(tt.desc, func(t *testing.T) { - q := tt.q - q.Addrs = []string{"127.0.0.1:8081"} - c := client.New() - wg := new(sync.WaitGroup) - wg.Add(1) - - go func() { - defer wg.Done() - if err := c.Subscribe(context.Background(), q); err != nil { - t.Errorf("c.Subscribe(): got error %v, expected nil", err) - } - }() - - wg.Wait() - after_subscribe := runtime.NumGoroutine() - t.Logf("Num go routines after client subscribe: %d", after_subscribe) - time.Sleep(10 * time.Second) - after_sleep := runtime.NumGoroutine() - t.Logf("Num go routines after sleep, should be less, as keepalive should close idle connections: %d", after_sleep) - if after_sleep > after_subscribe { - t.Errorf("Expecting goroutine after sleep to be less than or equal to after subscribe, after_subscribe: %d, after_sleep: %d", after_subscribe, after_sleep) - } - }) - } - } + s := createKeepAliveServer(t, 8081) + go runServer(t, s) + defer s.s.Stop() + + tests := []struct { + desc string + q client.Query + want []client.Notification + poll int + }{ + { + desc: "Testing KeepAlive with goroutine count", + poll: 3, + q: client.Query{ + Target: "COUNTERS_DB", + Type: client.Poll, + Queries: []client.Path{{"COUNTERS", "Ethernet*"}}, + TLS: &tls.Config{InsecureSkipVerify: true}, + }, + want: []client.Notification{ + client.Connected{}, + client.Sync{}, + }, + }, + } + for _, tt := range tests { + for i := 0; i < 5; i++ { + t.Run(tt.desc, func(t *testing.T) { + q := tt.q + q.Addrs = []string{"127.0.0.1:8081"} + c := client.New() + wg := new(sync.WaitGroup) + wg.Add(1) + + go func() { + defer wg.Done() + if err := c.Subscribe(context.Background(), q); err != nil { + t.Errorf("c.Subscribe(): got error %v, expected nil", err) + } + }() + + wg.Wait() + after_subscribe := runtime.NumGoroutine() + t.Logf("Num go routines after client subscribe: %d", after_subscribe) + time.Sleep(10 * time.Second) + after_sleep := runtime.NumGoroutine() + t.Logf("Num go routines after sleep, should be less, as keepalive should close idle connections: %d", after_sleep) + if after_sleep > after_subscribe { + t.Errorf("Expecting goroutine after sleep to be less than or equal to after subscribe, after_subscribe: %d, after_sleep: %d", after_subscribe, after_sleep) + } + }) + } + } } func TestClient(t *testing.T) { - var mutexDeInit sync.RWMutex - var mutexHB sync.RWMutex - var mutexIdx sync.RWMutex - - // sonic-host:device-test-event is a test event. - // Events client will drop it on floor. - events := [] sdc.Evt_rcvd { - { "test0", 7, 777 }, - { "test1", 6, 677 }, - { "{\"sonic-host:device-test-event\"", 5, 577 }, - { "test2", 5, 577 }, - { "test3", 4, 477 }, - } - - HEARTBEAT_SET := 5 - heartbeat := 0 - event_index := 0 - rcv_timeout := sdc.SUBSCRIBER_TIMEOUT - deinit_done := false - - mock1 := gomonkey.ApplyFunc(sdc.C_init_subs, func(use_cache bool) unsafe.Pointer { - return nil - }) - defer mock1.Reset() - - mock2 := gomonkey.ApplyFunc(sdc.C_recv_evt, func(h unsafe.Pointer) (int, sdc.Evt_rcvd) { - rc := (int)(0) - var evt sdc.Evt_rcvd - mutexIdx.Lock() - current_index := event_index - mutexIdx.Unlock() - if current_index < len(events) { - evt = events[current_index] - mutexIdx.RLock() - event_index = current_index + 1 - mutexIdx.RUnlock() - } else { - time.Sleep(time.Millisecond * time.Duration(rcv_timeout)) - rc = -1 - } - return rc, evt - }) - defer mock2.Reset() - - mock3 := gomonkey.ApplyFunc(sdc.Set_heartbeat, func(val int) { - mutexHB.RLock() - heartbeat = val - mutexHB.RUnlock() - }) - defer mock3.Reset() - - mock4 := gomonkey.ApplyFunc(sdc.C_deinit_subs, func(h unsafe.Pointer) { - mutexDeInit.RLock() - deinit_done = true - mutexDeInit.RUnlock() - }) - defer mock4.Reset() - - mock5 := gomonkey.ApplyMethod(reflect.TypeOf(&queue.PriorityQueue{}), "Put", func(pq *queue.PriorityQueue, item ...queue.Item) error { - return fmt.Errorf("Queue error") - }) - defer mock5.Reset() - - mock6 := gomonkey.ApplyMethod(reflect.TypeOf(&queue.PriorityQueue{}), "Len", func(pq *queue.PriorityQueue) int { - return 150000 // Max size for pending events in PQ is 102400 - }) - defer mock6.Reset() - - s := createServer(t, 8081) - go runServer(t, s) - - qstr := fmt.Sprintf("all[heartbeat=%d]", HEARTBEAT_SET) - q := createEventsQuery(t, qstr) - q.Addrs = []string{"127.0.0.1:8081"} - - tests := []struct { - desc string - pub_data []string - wantErr bool - wantNoti []client.Notification - pause int - poll int - } { - { - desc: "dropped event", - poll: 3, - }, - { - desc: "queue error", - poll: 3, - }, - { - desc: "base client create", - poll: 3, - }, - } - - sdc.C_init_subs(true) - - var mutexNoti sync.RWMutex - - for testNum, tt := range tests { - mutexHB.RLock() - heartbeat = 0 - mutexHB.RUnlock() - - mutexIdx.RLock() - event_index = 0 - mutexIdx.RUnlock() - - mutexDeInit.RLock() - deinit_done = false - mutexDeInit.RUnlock() - - t.Run(tt.desc, func(t *testing.T) { - c := client.New() - defer c.Close() - - var gotNoti []string - q.NotificationHandler = func(n client.Notification) error { - if nn, ok := n.(client.Update); ok { - nn.TS = time.Unix(0, 200) - str := fmt.Sprintf("%v", nn.Val) - - mutexNoti.Lock() - currentNoti := gotNoti - mutexNoti.Unlock() - - mutexNoti.RLock() - gotNoti = append(currentNoti, str) - mutexNoti.RUnlock() - } - return nil - } - - go func() { - c.Subscribe(context.Background(), q) - }() - - // wait for half second for subscribeRequest to sync - // and to receive events via notification handler. - - time.Sleep(time.Millisecond * 2000) - - if testNum > 1 { - mutexNoti.Lock() - // -1 to discount test event, which receiver would drop. - if (len(events) - 1) != len(gotNoti) { - t.Errorf("noti[%d] != events[%d]", len(gotNoti), len(events)-1) - } - - mutexHB.Lock() - if (heartbeat != HEARTBEAT_SET) { - t.Errorf("Heartbeat is not set %d != expected:%d", heartbeat, HEARTBEAT_SET) - } - mutexHB.Unlock() - - fmt.Printf("DONE: Expect events:%d - 1 gotNoti=%d\n", len(events), len(gotNoti)) - mutexNoti.Unlock() - } - }) - - if testNum == 0 { - mock6.Reset() - } - - if testNum == 1 { - mock5.Reset() - } - time.Sleep(time.Millisecond * 1000) - - mutexDeInit.Lock() - if deinit_done == false { - t.Errorf("Events client deinit *NOT* called.") - } - mutexDeInit.Unlock() - // t.Log("END of a TEST") - } - - s.s.Stop() + var mutexDeInit sync.RWMutex + var mutexHB sync.RWMutex + var mutexIdx sync.RWMutex + + // sonic-host:device-test-event is a test event. + // Events client will drop it on floor. + events := []sdc.Evt_rcvd{ + {"test0", 7, 777}, + {"test1", 6, 677}, + {"{\"sonic-host:device-test-event\"", 5, 577}, + {"test2", 5, 577}, + {"test3", 4, 477}, + } + + HEARTBEAT_SET := 5 + heartbeat := 0 + event_index := 0 + rcv_timeout := sdc.SUBSCRIBER_TIMEOUT + deinit_done := false + + mock1 := gomonkey.ApplyFunc(sdc.C_init_subs, func(use_cache bool) unsafe.Pointer { + return nil + }) + defer mock1.Reset() + + mock2 := gomonkey.ApplyFunc(sdc.C_recv_evt, func(h unsafe.Pointer) (int, sdc.Evt_rcvd) { + rc := (int)(0) + var evt sdc.Evt_rcvd + mutexIdx.Lock() + current_index := event_index + mutexIdx.Unlock() + if current_index < len(events) { + evt = events[current_index] + mutexIdx.RLock() + event_index = current_index + 1 + mutexIdx.RUnlock() + } else { + time.Sleep(time.Millisecond * time.Duration(rcv_timeout)) + rc = -1 + } + return rc, evt + }) + defer mock2.Reset() + + mock3 := gomonkey.ApplyFunc(sdc.Set_heartbeat, func(val int) { + mutexHB.RLock() + heartbeat = val + mutexHB.RUnlock() + }) + defer mock3.Reset() + + mock4 := gomonkey.ApplyFunc(sdc.C_deinit_subs, func(h unsafe.Pointer) { + mutexDeInit.RLock() + deinit_done = true + mutexDeInit.RUnlock() + }) + defer mock4.Reset() + + mock5 := gomonkey.ApplyMethod(reflect.TypeOf(&queue.PriorityQueue{}), "Put", func(pq *queue.PriorityQueue, item ...queue.Item) error { + return fmt.Errorf("Queue error") + }) + defer mock5.Reset() + + mock6 := gomonkey.ApplyMethod(reflect.TypeOf(&queue.PriorityQueue{}), "Len", func(pq *queue.PriorityQueue) int { + return 150000 // Max size for pending events in PQ is 102400 + }) + defer mock6.Reset() + + s := createServer(t, 8081) + go runServer(t, s) + + qstr := fmt.Sprintf("all[heartbeat=%d]", HEARTBEAT_SET) + q := createEventsQuery(t, qstr) + q.Addrs = []string{"127.0.0.1:8081"} + + tests := []struct { + desc string + pub_data []string + wantErr bool + wantNoti []client.Notification + pause int + poll int + }{ + { + desc: "dropped event", + poll: 3, + }, + { + desc: "queue error", + poll: 3, + }, + { + desc: "base client create", + poll: 3, + }, + } + + sdc.C_init_subs(true) + + var mutexNoti sync.RWMutex + + for testNum, tt := range tests { + mutexHB.RLock() + heartbeat = 0 + mutexHB.RUnlock() + + mutexIdx.RLock() + event_index = 0 + mutexIdx.RUnlock() + + mutexDeInit.RLock() + deinit_done = false + mutexDeInit.RUnlock() + + t.Run(tt.desc, func(t *testing.T) { + c := client.New() + defer c.Close() + + var gotNoti []string + q.NotificationHandler = func(n client.Notification) error { + if nn, ok := n.(client.Update); ok { + nn.TS = time.Unix(0, 200) + str := fmt.Sprintf("%v", nn.Val) + + mutexNoti.Lock() + currentNoti := gotNoti + mutexNoti.Unlock() + + mutexNoti.RLock() + gotNoti = append(currentNoti, str) + mutexNoti.RUnlock() + } + return nil + } + + go func() { + c.Subscribe(context.Background(), q) + }() + + // wait for half second for subscribeRequest to sync + // and to receive events via notification handler. + + time.Sleep(time.Millisecond * 2000) + + if testNum > 1 { + mutexNoti.Lock() + // -1 to discount test event, which receiver would drop. + if (len(events) - 1) != len(gotNoti) { + t.Errorf("noti[%d] != events[%d]", len(gotNoti), len(events)-1) + } + + mutexHB.Lock() + if heartbeat != HEARTBEAT_SET { + t.Errorf("Heartbeat is not set %d != expected:%d", heartbeat, HEARTBEAT_SET) + } + mutexHB.Unlock() + + fmt.Printf("DONE: Expect events:%d - 1 gotNoti=%d\n", len(events), len(gotNoti)) + mutexNoti.Unlock() + } + }) + + if testNum == 0 { + mock6.Reset() + } + + if testNum == 1 { + mock5.Reset() + } + time.Sleep(time.Millisecond * 1000) + + mutexDeInit.Lock() + if deinit_done == false { + t.Errorf("Events client deinit *NOT* called.") + } + mutexDeInit.Unlock() + // t.Log("END of a TEST") + } + + s.s.Stop() } func TestTableData2MsiUseKey(t *testing.T) { - tblPath := sdc.CreateTablePath("STATE_DB", "NEIGH_STATE_TABLE", "|", "10.0.0.57") - newMsi := make(map[string]interface{}) - sdc.TableData2Msi(&tblPath, true, nil, &newMsi) - newMsiData, _ := json.MarshalIndent(newMsi, "", " ") - t.Logf(string(newMsiData)) - expectedMsi := map[string]interface{} { - "10.0.0.57": map[string]interface{} { - "peerType": "e-BGP", - "state": "Established", - }, - } - expectedMsiData, _ := json.MarshalIndent(expectedMsi, "", " ") - t.Logf(string(expectedMsiData)) - - if !reflect.DeepEqual(newMsi, expectedMsi) { - t.Errorf("Msi data does not match for use key = true") - } + tblPath := sdc.CreateTablePath("STATE_DB", "NEIGH_STATE_TABLE", "|", "10.0.0.57") + newMsi := make(map[string]interface{}) + sdc.TableData2Msi(&tblPath, true, nil, &newMsi) + newMsiData, _ := json.MarshalIndent(newMsi, "", " ") + t.Logf(string(newMsiData)) + expectedMsi := map[string]interface{}{ + "10.0.0.57": map[string]interface{}{ + "peerType": "e-BGP", + "state": "Established", + }, + } + expectedMsiData, _ := json.MarshalIndent(expectedMsi, "", " ") + t.Logf(string(expectedMsiData)) + + if !reflect.DeepEqual(newMsi, expectedMsi) { + t.Errorf("Msi data does not match for use key = true") + } } func TestRecoverFromJSONSerializationPanic(t *testing.T) { - panicMarshal := func(v interface{}) ([]byte, error) { - panic("json.Marshal panics and is unable to serialize JSON") - } - mock := gomonkey.ApplyFunc(json.Marshal, panicMarshal) - defer mock.Reset() + panicMarshal := func(v interface{}) ([]byte, error) { + panic("json.Marshal panics and is unable to serialize JSON") + } + mock := gomonkey.ApplyFunc(json.Marshal, panicMarshal) + defer mock.Reset() - tblPath := sdc.CreateTablePath("STATE_DB", "NEIGH_STATE_TABLE", "|", "10.0.0.57") - msi := make(map[string]interface{}) - sdc.TableData2Msi(&tblPath, true, nil, &msi) + tblPath := sdc.CreateTablePath("STATE_DB", "NEIGH_STATE_TABLE", "|", "10.0.0.57") + msi := make(map[string]interface{}) + sdc.TableData2Msi(&tblPath, true, nil, &msi) - typedValue, err := sdc.Msi2TypedValue(msi) - if typedValue != nil && err != nil { - t.Errorf("Test should recover from panic and have nil TypedValue/Error after attempting JSON serialization") - } + typedValue, err := sdc.Msi2TypedValue(msi) + if typedValue != nil && err != nil { + t.Errorf("Test should recover from panic and have nil TypedValue/Error after attempting JSON serialization") + } } func TestGnmiSetBatch(t *testing.T) { - mockCode := -` + mockCode := + ` print('No Yang validation for test mode...') print('%s') ` @@ -3839,8 +3838,8 @@ func TestGNMINative(t *testing.T) { return &dbus.Call{} }) defer mock2.Reset() - mockCode := -` + mockCode := + ` print('No Yang validation for test mode...') print('%s') ` diff --git a/gnoi_client/gnoi_client.go b/gnoi_client/gnoi_client.go index 31ea33da..fe082cf3 100644 --- a/gnoi_client/gnoi_client.go +++ b/gnoi_client/gnoi_client.go @@ -1,28 +1,29 @@ package main import ( - "google.golang.org/grpc" + "context" + "encoding/json" + "flag" + "fmt" + "github.com/google/gnxi/utils/credentials" gnoi_system_pb "github.com/openconfig/gnoi/system" spb "github.com/sonic-net/sonic-gnmi/proto/gnoi" spb_jwt "github.com/sonic-net/sonic-gnmi/proto/gnoi/jwt" - "context" + "google.golang.org/grpc" + "google.golang.org/grpc/metadata" "os" "os/signal" - "fmt" - "flag" - "google.golang.org/grpc/metadata" - "github.com/google/gnxi/utils/credentials" - "encoding/json" ) var ( - module = flag.String("module", "System", "gNOI Module") - rpc = flag.String("rpc", "Time", "rpc call in specified module to call") - target = flag.String("target", "localhost:8080", "Address:port of gNOI Server") - args = flag.String("jsonin", "", "RPC Arguments in json format") - jwtToken = flag.String("jwt_token", "", "JWT Token if required") + module = flag.String("module", "System", "gNOI Module") + rpc = flag.String("rpc", "Time", "rpc call in specified module to call") + target = flag.String("target", "localhost:8080", "Address:port of gNOI Server") + args = flag.String("jsonin", "", "RPC Arguments in json format") + jwtToken = flag.String("jwt_token", "", "JWT Token if required") targetName = flag.String("target_name", "hostname.com", "The target name use to verify the hostname returned by TLS handshake") ) + func setUserCreds(ctx context.Context) context.Context { if len(*jwtToken) > 0 { ctx = metadata.AppendToOutgoingContext(ctx, "access_token", *jwtToken) @@ -33,18 +34,18 @@ func main() { flag.Parse() opts := credentials.ClientCredentials(*targetName) - ctx, cancel := context.WithCancel(context.Background()) - go func() { - c := make(chan os.Signal, 1) - signal.Notify(c, os.Interrupt) - <-c - cancel() - }() + ctx, cancel := context.WithCancel(context.Background()) + go func() { + c := make(chan os.Signal, 1) + signal.Notify(c, os.Interrupt) + <-c + cancel() + }() conn, err := grpc.Dial(*target, opts...) if err != nil { panic(err.Error()) } - + switch *module { case "System": sc := gnoi_system_pb.NewSystemClient(conn) @@ -98,7 +99,7 @@ func main() { func systemTime(sc gnoi_system_pb.SystemClient, ctx context.Context) { fmt.Println("System Time") ctx = setUserCreds(ctx) - resp,err := sc.Time(ctx, new(gnoi_system_pb.TimeRequest)) + resp, err := sc.Time(ctx, new(gnoi_system_pb.TimeRequest)) if err != nil { panic(err.Error()) } @@ -112,9 +113,9 @@ func systemTime(sc gnoi_system_pb.SystemClient, ctx context.Context) { func systemReboot(sc gnoi_system_pb.SystemClient, ctx context.Context) { fmt.Println("System Reboot") ctx = setUserCreds(ctx) - req := &gnoi_system_pb.RebootRequest {} + req := &gnoi_system_pb.RebootRequest{} json.Unmarshal([]byte(*args), req) - _,err := sc.Reboot(ctx, req) + _, err := sc.Reboot(ctx, req) if err != nil { panic(err.Error()) } @@ -123,9 +124,9 @@ func systemReboot(sc gnoi_system_pb.SystemClient, ctx context.Context) { func systemCancelReboot(sc gnoi_system_pb.SystemClient, ctx context.Context) { fmt.Println("System CancelReboot") ctx = setUserCreds(ctx) - req := &gnoi_system_pb.CancelRebootRequest {} + req := &gnoi_system_pb.CancelRebootRequest{} json.Unmarshal([]byte(*args), req) - resp,err := sc.CancelReboot(ctx, req) + resp, err := sc.CancelReboot(ctx, req) if err != nil { panic(err.Error()) } @@ -139,8 +140,8 @@ func systemCancelReboot(sc gnoi_system_pb.SystemClient, ctx context.Context) { func systemRebootStatus(sc gnoi_system_pb.SystemClient, ctx context.Context) { fmt.Println("System RebootStatus") ctx = setUserCreds(ctx) - req := &gnoi_system_pb.RebootStatusRequest {} - resp,err := sc.RebootStatus(ctx, req) + req := &gnoi_system_pb.RebootStatusRequest{} + resp, err := sc.RebootStatus(ctx, req) if err != nil { panic(err.Error()) } @@ -154,15 +155,13 @@ func systemRebootStatus(sc gnoi_system_pb.SystemClient, ctx context.Context) { func sonicShowTechSupport(sc spb.SonicServiceClient, ctx context.Context) { fmt.Println("Sonic ShowTechsupport") ctx = setUserCreds(ctx) - req := &spb.TechsupportRequest { - Input: &spb.TechsupportRequest_Input{ - - }, + req := &spb.TechsupportRequest{ + Input: &spb.TechsupportRequest_Input{}, } json.Unmarshal([]byte(*args), req) - - resp,err := sc.ShowTechsupport(ctx, req) + + resp, err := sc.ShowTechsupport(ctx, req) if err != nil { panic(err.Error()) } @@ -181,7 +180,7 @@ func copyConfig(sc spb.SonicServiceClient, ctx context.Context) { } json.Unmarshal([]byte(*args), req) - resp,err := sc.CopyConfig(ctx, req) + resp, err := sc.CopyConfig(ctx, req) if err != nil { panic(err.Error()) @@ -200,7 +199,7 @@ func imageInstall(sc spb.SonicServiceClient, ctx context.Context) { } json.Unmarshal([]byte(*args), req) - resp,err := sc.ImageInstall(ctx, req) + resp, err := sc.ImageInstall(ctx, req) if err != nil { panic(err.Error()) @@ -219,7 +218,7 @@ func imageRemove(sc spb.SonicServiceClient, ctx context.Context) { } json.Unmarshal([]byte(*args), req) - resp,err := sc.ImageRemove(ctx, req) + resp, err := sc.ImageRemove(ctx, req) if err != nil { panic(err.Error()) @@ -239,7 +238,7 @@ func imageDefault(sc spb.SonicServiceClient, ctx context.Context) { } json.Unmarshal([]byte(*args), req) - resp,err := sc.ImageDefault(ctx, req) + resp, err := sc.ImageDefault(ctx, req) if err != nil { panic(err.Error()) @@ -254,11 +253,11 @@ func imageDefault(sc spb.SonicServiceClient, ctx context.Context) { func authenticate(sc spb_jwt.SonicJwtServiceClient, ctx context.Context) { fmt.Println("Sonic Authenticate") ctx = setUserCreds(ctx) - req := &spb_jwt.AuthenticateRequest {} - + req := &spb_jwt.AuthenticateRequest{} + json.Unmarshal([]byte(*args), req) - - resp,err := sc.Authenticate(ctx, req) + + resp, err := sc.Authenticate(ctx, req) if err != nil { panic(err.Error()) } @@ -272,11 +271,11 @@ func authenticate(sc spb_jwt.SonicJwtServiceClient, ctx context.Context) { func refresh(sc spb_jwt.SonicJwtServiceClient, ctx context.Context) { fmt.Println("Sonic Refresh") ctx = setUserCreds(ctx) - req := &spb_jwt.RefreshRequest {} - + req := &spb_jwt.RefreshRequest{} + json.Unmarshal([]byte(*args), req) - resp,err := sc.Refresh(ctx, req) + resp, err := sc.Refresh(ctx, req) if err != nil { panic(err.Error()) } @@ -288,21 +287,21 @@ func refresh(sc spb_jwt.SonicJwtServiceClient, ctx context.Context) { } func clearNeighbors(sc spb.SonicServiceClient, ctx context.Context) { - fmt.Println("Sonic ClearNeighbors") - ctx = setUserCreds(ctx) - req := &spb.ClearNeighborsRequest{ - Input: &spb.ClearNeighborsRequest_Input{}, - } - json.Unmarshal([]byte(*args), req) - - resp,err := sc.ClearNeighbors(ctx, req) - - if err != nil { - panic(err.Error()) - } - respstr, err := json.Marshal(resp) - if err != nil { - panic(err.Error()) - } - fmt.Println(string(respstr)) + fmt.Println("Sonic ClearNeighbors") + ctx = setUserCreds(ctx) + req := &spb.ClearNeighborsRequest{ + Input: &spb.ClearNeighborsRequest_Input{}, + } + json.Unmarshal([]byte(*args), req) + + resp, err := sc.ClearNeighbors(ctx, req) + + if err != nil { + panic(err.Error()) + } + respstr, err := json.Marshal(resp) + if err != nil { + panic(err.Error()) + } + fmt.Println(string(respstr)) } diff --git a/sonic_data_client/client_test.go b/sonic_data_client/client_test.go index 6f864faf..f95d0313 100644 --- a/sonic_data_client/client_test.go +++ b/sonic_data_client/client_test.go @@ -1,19 +1,19 @@ package client import ( - "sync" - "errors" - "testing" - "os" - "time" - "reflect" - "io/ioutil" "encoding/json" + "errors" "fmt" + "io/ioutil" + "os" + "reflect" + "sync" + "testing" + "time" "github.com/jipanyang/gnxi/utils/xpath" - "github.com/sonic-net/sonic-gnmi/swsscommon" gnmipb "github.com/openconfig/gnmi/proto/gnmi" + "github.com/sonic-net/sonic-gnmi/swsscommon" ) var testFile string = "/etc/sonic/ut.cp.json" @@ -58,32 +58,32 @@ func TestJsonAdd(t *testing.T) { if err != nil { t.Errorf("Create client fail: %v", err) } - path_list := [][]string { - []string { + path_list := [][]string{ + []string{ "DASH_QOS", }, - []string { + []string{ "DASH_QOS", "qos_02", }, - []string { + []string{ "DASH_QOS", "qos_03", "bw", }, - []string { + []string{ "DASH_VNET", "vnet001", "address_spaces", }, - []string { + []string{ "DASH_VNET", "vnet002", "address_spaces", "0", }, } - value_list := []string { + value_list := []string{ `{"qos_01": {"bw": "54321", "cps": "1000", "flows": "300"}}`, `{"bw": "10001", "cps": "1001", "flows": "101"}`, `"20001"`, @@ -128,44 +128,44 @@ func TestJsonAddNegative(t *testing.T) { if err != nil { t.Errorf("Create client fail: %v", err) } - path_list := [][]string { - []string { + path_list := [][]string{ + []string{ "DASH_QOS", }, - []string { + []string{ "DASH_QOS", "qos_02", }, - []string { + []string{ "DASH_QOS", "qos_03", "bw", }, - []string { + []string{ "DASH_VNET", "vnet001", "address_spaces", }, - []string { + []string{ "DASH_VNET", "vnet002", "address_spaces", "0", }, - []string { + []string{ "DASH_VNET", "vnet002", "address_spaces", "abc", }, - []string { + []string{ "DASH_VNET", "vnet002", "address_spaces", "100", }, } - value_list := []string { + value_list := []string{ `{"qos_01": {"bw": "54321", "cps": "1000", "flows": "300"}`, `{"bw": "10001", "cps": "1001", "flows": "101"`, `20001`, @@ -194,32 +194,32 @@ func TestJsonRemove(t *testing.T) { if err != nil { t.Errorf("Create client fail: %v", err) } - path_list := [][]string { - []string { + path_list := [][]string{ + []string{ "DASH_QOS", }, - []string { + []string{ "DASH_QOS", "qos_02", }, - []string { + []string{ "DASH_QOS", "qos_03", "bw", }, - []string { + []string{ "DASH_VNET", "vnet001", "address_spaces", }, - []string { + []string{ "DASH_VNET", "vnet002", "address_spaces", "0", }, } - value_list := []string { + value_list := []string{ `{"qos_01": {"bw": "54321", "cps": "1000", "flows": "300"}}`, `{"bw": "10001", "cps": "1001", "flows": "101"}`, `"20001"`, @@ -254,17 +254,17 @@ func TestJsonRemoveNegative(t *testing.T) { if err != nil { t.Errorf("Create client fail: %v", err) } - path_list := [][]string { - []string { + path_list := [][]string{ + []string{ "DASH_QOS", }, - []string { + []string{ "DASH_VNET", "vnet001", "address_spaces", }, } - value_list := []string { + value_list := []string{ `{"qos_01": {"bw": "54321", "cps": "1000", "flows": "300"}}`, `["10.250.0.0", "192.168.3.0", "139.66.72.9"]`, } @@ -277,23 +277,23 @@ func TestJsonRemoveNegative(t *testing.T) { } } - remove_list := [][]string { - []string { + remove_list := [][]string{ + []string{ "DASH_QOS", "qos_02", }, - []string { + []string{ "DASH_QOS", "qos_03", "bw", }, - []string { + []string{ "DASH_VNET", "vnet001", "address_spaces", "abc", }, - []string { + []string{ "DASH_VNET", "vnet001", "address_spaces", @@ -360,17 +360,18 @@ func TestNonDbClientGetError(t *testing.T) { } /* - Helper method for receive data from ZmqConsumerStateTable - consumer: Receive data from consumer - return: - true: data received - false: not receive any data after retry +Helper method for receive data from ZmqConsumerStateTable + + consumer: Receive data from consumer + return: + true: data received + false: not receive any data after retry */ -func ReceiveFromZmq(consumer swsscommon.ZmqConsumerStateTable) (bool) { +func ReceiveFromZmq(consumer swsscommon.ZmqConsumerStateTable) bool { receivedData := swsscommon.NewKeyOpFieldsValuesQueue() - retry := 0; + retry := 0 for { - // sender's ZMQ may disconnect, wait and retry for reconnect + // sender's ZMQ may disconnect, wait and retry for reconnect time.Sleep(time.Duration(1000) * time.Millisecond) consumer.Pops(receivedData) if receivedData.Size() == 0 { @@ -389,17 +390,17 @@ func TestZmqReconnect(t *testing.T) { db := swsscommon.NewDBConnector(APPL_DB_NAME, SWSS_TIMEOUT, false) zmqServer := swsscommon.NewZmqServer("tcp://*:1234") var TEST_TABLE string = "DASH_ROUTE" - consumer := swsscommon.NewZmqConsumerStateTable(db, TEST_TABLE, zmqServer) + consumer := swsscommon.NewZmqConsumerStateTable(db, TEST_TABLE, zmqServer) // create ZMQ client side zmqAddress := "tcp://127.0.0.1:1234" - client := MixedDbClient { - applDB : swsscommon.NewDBConnector(APPL_DB_NAME, SWSS_TIMEOUT, false), - tableMap : map[string]swsscommon.ProducerStateTable{}, - zmqClient : swsscommon.NewZmqClient(zmqAddress), + client := MixedDbClient{ + applDB: swsscommon.NewDBConnector(APPL_DB_NAME, SWSS_TIMEOUT, false), + tableMap: map[string]swsscommon.ProducerStateTable{}, + zmqClient: swsscommon.NewZmqClient(zmqAddress), } - data := map[string]string{} + data := map[string]string{} var TEST_KEY string = "TestKey" client.DbSetTable(TEST_TABLE, TEST_KEY, data) if !ReceiveFromZmq(consumer) { @@ -407,10 +408,10 @@ func TestZmqReconnect(t *testing.T) { } // recreate ZMQ server to trigger re-connect - swsscommon.DeleteZmqConsumerStateTable(consumer) + swsscommon.DeleteZmqConsumerStateTable(consumer) swsscommon.DeleteZmqServer(zmqServer) zmqServer = swsscommon.NewZmqServer("tcp://*:1234") - consumer = swsscommon.NewZmqConsumerStateTable(db, TEST_TABLE, zmqServer) + consumer = swsscommon.NewZmqConsumerStateTable(db, TEST_TABLE, zmqServer) // send data again, client will reconnect client.DbSetTable(TEST_TABLE, TEST_KEY, data) @@ -428,16 +429,16 @@ func TestRetryHelper(t *testing.T) { zmqClient := swsscommon.NewZmqClient(zmqAddress) returnError := true exeCount := 0 - RetryHelper( + RetryHelper( zmqClient, - func () (err error) { + func() (err error) { exeCount++ if returnError { returnError = false return fmt.Errorf("connection_reset") } return nil - }) + }) if exeCount == 1 { t.Errorf("RetryHelper does not retry") diff --git a/sonic_data_client/db_client.go b/sonic_data_client/db_client.go index ddca58b8..e5274924 100644 --- a/sonic_data_client/db_client.go +++ b/sonic_data_client/db_client.go @@ -5,19 +5,19 @@ import ( "bytes" "encoding/json" "fmt" + log "github.com/golang/glog" "net" "reflect" "strconv" "strings" "sync" "time" - log "github.com/golang/glog" - spb "github.com/sonic-net/sonic-gnmi/proto" - sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" "github.com/Workiva/go-datastructures/queue" "github.com/go-redis/redis" gnmipb "github.com/openconfig/gnmi/proto/gnmi" + spb "github.com/sonic-net/sonic-gnmi/proto" + sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" ) const ( @@ -28,7 +28,6 @@ const ( // Client defines a set of methods which every client must implement. // This package provides one implmentation for now: the DbClient -// type Client interface { // StreamRun will start watching service on data source // and enqueue data change to the priority queue. @@ -54,7 +53,7 @@ type Client interface { // callbacks on send failed FailedSend() - // callback on sent + // callback on sent SentOne(*Value) } @@ -459,7 +458,7 @@ func GetRedisClientsForDb(target string) (redis_client_map map[string]*redis.Cli // This function get target present in GNMI Request and // returns: 1. DbName (string) 2. Is DbName valid (bool) -// 3. DbNamespace (string) 4. Is DbNamespace present in Target (bool) +// 3. DbNamespace (string) 4. Is DbNamespace present in Target (bool) func IsTargetDb(target string) (string, bool, string, bool) { targetname := strings.Split(target, "/") dbName := targetname[0] @@ -610,7 +609,6 @@ func populateDbtablePath(prefix, path *gnmipb.Path, pathG2S *map[*gnmipb.Path][] } } - fullPath := path if prefix != nil { fullPath = gnmiFullPath(prefix, path) diff --git a/sonic_data_client/dummy_client_test.go b/sonic_data_client/dummy_client_test.go new file mode 100644 index 00000000..7157d181 --- /dev/null +++ b/sonic_data_client/dummy_client_test.go @@ -0,0 +1,50 @@ +package client + +import ( + "sync" + "testing" + "time" + + "github.com/Workiva/go-datastructures/queue" + gnmipb "github.com/openconfig/gnmi/proto/gnmi" + spb "github.com/sonic-net/sonic-gnmi/proto" +) + +func TestDummyEventClient(t *testing.T) { + evtc := &EventClient{} + evtc.last_latencies[0] = 1 + evtc.last_latencies[1] = 2 + evtc.last_latency_index = 9 + evtc.last_latency_full = true + evtc.counters = make(map[string]uint64) + evtc.counters["COUNTERS_EVENTS:latency_in_ms"] = 0 + compute_latency(evtc) + + // Prepare necessary arguments for each function + var wg sync.WaitGroup + var q *queue.PriorityQueue // Assuming queue.PriorityQueue is a valid type + once := make(chan struct{}) + poll := make(chan struct{}) + var subscribe *gnmipb.SubscriptionList // Assuming gnmipb.SubscriptionList is a valid type + var deletePaths []*gnmipb.Path // Assuming gnmipb.Path is a valid type + var replaceUpdates, updateUpdates []*gnmipb.Update // Assuming gnmipb.Update is a valid type + + evtc.Get(&wg) + evtc.OnceRun(q, once, &wg, subscribe) + evtc.PollRun(q, poll, &wg, subscribe) + evtc.Close() + evtc.Set(deletePaths, replaceUpdates, updateUpdates) + evtc.Capabilities() + evtc.last_latencies[0] = 1 + evtc.last_latencies[1] = 2 + evtc.last_latency_index = 9 + evtc.last_latency_full = true + evtc.SentOne(&Value{ + &spb.Value{ + Timestamp: time.Now().UnixNano(), + }, + }) + evtc.FailedSend() + evtc.subs_handle = C_init_subs(true) + +} diff --git a/sonic_data_client/events_client.go b/sonic_data_client/events_client.go index e8fa2648..5086c297 100644 --- a/sonic_data_client/events_client.go +++ b/sonic_data_client/events_client.go @@ -1,7 +1,7 @@ package client /* -#cgo CFLAGS: -g -Wall -I../../sonic-swss-common/common -Wformat -Werror=format-security -fPIE +#cgo CFLAGS: -g -Wall -I../../sonic-swss-common/common -Wformat -Werror=format-security -fPIE #cgo LDFLAGS: -L/usr/lib -lpthread -lboost_thread -lboost_system -lzmq -lboost_serialization -luuid -lswsscommon #include #include @@ -11,41 +11,41 @@ package client import "C" import ( - "strconv" - "encoding/json" - "fmt" - "reflect" - "strings" - "sync" - "time" - "unsafe" - - "github.com/go-redis/redis" - - spb "github.com/sonic-net/sonic-gnmi/proto" - sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" - "github.com/Workiva/go-datastructures/queue" - log "github.com/golang/glog" - gnmipb "github.com/openconfig/gnmi/proto/gnmi" + "encoding/json" + "fmt" + "reflect" + "strconv" + "strings" + "sync" + "time" + "unsafe" + + "github.com/go-redis/redis" + + "github.com/Workiva/go-datastructures/queue" + log "github.com/golang/glog" + gnmipb "github.com/openconfig/gnmi/proto/gnmi" + spb "github.com/sonic-net/sonic-gnmi/proto" + sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" ) -const SUBSCRIBER_TIMEOUT = (2 * 1000) // 2 seconds +const SUBSCRIBER_TIMEOUT = (2 * 1000) // 2 seconds const EVENT_BUFFSZ = 4096 -const LATENCY_LIST_SIZE = 10 // Size of list of latencies. -const PQ_DEF_SIZE = 10240 // Def size for pending events in PQ. -const PQ_MIN_SIZE = 1024 // Min size for pending events in PQ. -const PQ_MAX_SIZE = 102400 // Max size for pending events in PQ. +const LATENCY_LIST_SIZE = 10 // Size of list of latencies. +const PQ_DEF_SIZE = 10240 // Def size for pending events in PQ. +const PQ_MIN_SIZE = 1024 // Min size for pending events in PQ. +const PQ_MAX_SIZE = 102400 // Max size for pending events in PQ. -const HEARTBEAT_MAX = 600 // 10 mins +const HEARTBEAT_MAX = 600 // 10 mins // STATS counters const MISSED = "COUNTERS_EVENTS:missed_internal" const DROPPED = "COUNTERS_EVENTS:missed_by_slow_receiver" const LATENCY = "COUNTERS_EVENTS:latency_in_ms" -var STATS_CUMULATIVE_KEYS = [...]string {MISSED, DROPPED} -var STATS_ABSOLUTE_KEYS = [...]string {LATENCY} +var STATS_CUMULATIVE_KEYS = [...]string{MISSED, DROPPED} +var STATS_ABSOLUTE_KEYS = [...]string{LATENCY} const STATS_FIELD_NAME = "value" @@ -59,432 +59,423 @@ const PARAM_QSIZE = "qsize" const PARAM_USE_CACHE = "usecache" type EventClient struct { + prefix *gnmipb.Path + path *gnmipb.Path - prefix *gnmipb.Path - path *gnmipb.Path + q *queue.PriorityQueue + pq_max int + channel chan struct{} - q *queue.PriorityQueue - pq_max int - channel chan struct{} + wg *sync.WaitGroup // wait for all sub go routines to finish - wg *sync.WaitGroup // wait for all sub go routines to finish + subs_handle unsafe.Pointer - subs_handle unsafe.Pointer + stopped int + stopMutex sync.RWMutex - stopped int - stopMutex sync.RWMutex + // Stats counter + counters map[string]uint64 + countersMutex sync.RWMutex - // Stats counter - counters map[string]uint64 - countersMutex sync.RWMutex + last_latencies [LATENCY_LIST_SIZE]uint64 + last_latency_index int + last_latency_full bool - last_latencies [LATENCY_LIST_SIZE]uint64 - last_latency_index int - last_latency_full bool - - last_errors uint64 + last_errors uint64 } func Set_heartbeat(val int) { - s := fmt.Sprintf("{\"HEARTBEAT_INTERVAL\":%d}", val) - rc := C.event_set_global_options(C.CString(s)); - if rc != 0 { - log.V(4).Infof("Failed to set heartbeat val=%d rc=%d", val, rc) - } + s := fmt.Sprintf("{\"HEARTBEAT_INTERVAL\":%d}", val) + rc := C.event_set_global_options(C.CString(s)) + if rc != 0 { + log.V(4).Infof("Failed to set heartbeat val=%d rc=%d", val, rc) + } } func C_init_subs(use_cache bool) unsafe.Pointer { - return C.events_init_subscriber_wrap(C.bool(use_cache), C.int(SUBSCRIBER_TIMEOUT)) + return C.events_init_subscriber_wrap(C.bool(use_cache), C.int(SUBSCRIBER_TIMEOUT)) } func NewEventClient(paths []*gnmipb.Path, prefix *gnmipb.Path, logLevel int) (Client, error) { - var evtc EventClient - use_cache := true - evtc.prefix = prefix - evtc.pq_max = PQ_DEF_SIZE - log.V(4).Infof("Events priority Q max set default = %v", evtc.pq_max) - - for _, path := range paths { - // Only one path is expected. Take the last if many - evtc.path = path - } - - for _, e := range evtc.path.GetElem() { - keys := e.GetKey() - for k, v := range keys { - if (k == PARAM_HEARTBEAT) { - if val, err := strconv.Atoi(v); err == nil { - if (val > HEARTBEAT_MAX) { - log.V(4).Infof("heartbeat req %v > max %v; default to max", val, HEARTBEAT_MAX) - val = HEARTBEAT_MAX - } - log.V(7).Infof("evtc.heartbeat_interval is set to %d", val) - Set_heartbeat(val) - } - } else if (k == PARAM_QSIZE) { - if val, err := strconv.Atoi(v); err == nil { - qval := val - if (val < PQ_MIN_SIZE) { - val = PQ_MIN_SIZE - } else if (val > PQ_MAX_SIZE) { - val = PQ_MAX_SIZE - } - if val != qval { - log.V(4).Infof("Events priority Q request %v updated to nearest limit %v", - qval, val) - } - evtc.pq_max = val - log.V(7).Infof("Events priority Q max set by qsize param = %v", evtc.pq_max) - } - } else if (k == PARAM_USE_CACHE) { - if strings.ToLower(v) == "false" { - use_cache = false - log.V(7).Infof("Cache use is turned off") - } - } - } - } - - C.swssSetLogPriority(C.int(logLevel)) - - /* Init subscriber with cache use and defined time out */ - evtc.subs_handle = C_init_subs(use_cache) - evtc.stopped = 0 - - /* Init list & counters */ - evtc.counters = make(map[string]uint64) - - for _, key := range STATS_CUMULATIVE_KEYS { - evtc.counters[key] = 0 - } - - for _, key := range STATS_ABSOLUTE_KEYS { - evtc.counters[key] = 0 - } - - for i := 0; i < len(evtc.last_latencies); i++ { - evtc.last_latencies[i] = 0 - } - evtc.last_latency_index = 0 - evtc.last_errors = 0 - evtc.last_latency_full = false - - log.V(7).Infof("NewEventClient constructed. logLevel=%d", logLevel) - - return &evtc, nil + var evtc EventClient + use_cache := true + evtc.prefix = prefix + evtc.pq_max = PQ_DEF_SIZE + log.V(4).Infof("Events priority Q max set default = %v", evtc.pq_max) + + for _, path := range paths { + // Only one path is expected. Take the last if many + evtc.path = path + } + + for _, e := range evtc.path.GetElem() { + keys := e.GetKey() + for k, v := range keys { + if k == PARAM_HEARTBEAT { + if val, err := strconv.Atoi(v); err == nil { + if val > HEARTBEAT_MAX { + log.V(4).Infof("heartbeat req %v > max %v; default to max", val, HEARTBEAT_MAX) + val = HEARTBEAT_MAX + } + log.V(7).Infof("evtc.heartbeat_interval is set to %d", val) + Set_heartbeat(val) + } + } else if k == PARAM_QSIZE { + if val, err := strconv.Atoi(v); err == nil { + qval := val + if val < PQ_MIN_SIZE { + val = PQ_MIN_SIZE + } else if val > PQ_MAX_SIZE { + val = PQ_MAX_SIZE + } + if val != qval { + log.V(4).Infof("Events priority Q request %v updated to nearest limit %v", + qval, val) + } + evtc.pq_max = val + log.V(7).Infof("Events priority Q max set by qsize param = %v", evtc.pq_max) + } + } else if k == PARAM_USE_CACHE { + if strings.ToLower(v) == "false" { + use_cache = false + log.V(7).Infof("Cache use is turned off") + } + } + } + } + + C.swssSetLogPriority(C.int(logLevel)) + + /* Init subscriber with cache use and defined time out */ + evtc.subs_handle = C_init_subs(use_cache) + evtc.stopped = 0 + + /* Init list & counters */ + evtc.counters = make(map[string]uint64) + + for _, key := range STATS_CUMULATIVE_KEYS { + evtc.counters[key] = 0 + } + + for _, key := range STATS_ABSOLUTE_KEYS { + evtc.counters[key] = 0 + } + + for i := 0; i < len(evtc.last_latencies); i++ { + evtc.last_latencies[i] = 0 + } + evtc.last_latency_index = 0 + evtc.last_errors = 0 + evtc.last_latency_full = false + + log.V(7).Infof("NewEventClient constructed. logLevel=%d", logLevel) + + return &evtc, nil } - func compute_latency(evtc *EventClient) { - if evtc.last_latency_full { - var total uint64 = 0 - - for _, v := range evtc.last_latencies { - if v > 0 { - total += v - } - } - evtc.countersMutex.RLock() - evtc.counters[LATENCY] = (uint64) (total/LATENCY_LIST_SIZE/1000/1000) - evtc.countersMutex.RUnlock() - } + if evtc.last_latency_full { + var total uint64 = 0 + + for _, v := range evtc.last_latencies { + if v > 0 { + total += v + } + } + evtc.countersMutex.RLock() + evtc.counters[LATENCY] = (uint64)(total / LATENCY_LIST_SIZE / 1000 / 1000) + evtc.countersMutex.RUnlock() + } } func update_stats(evtc *EventClient) { - defer evtc.wg.Done() - - /* Wait for any update */ - db_counters := make(map[string]uint64) - var wr_counters *map[string]uint64 = nil - var rclient *redis.Client - - /* - * This loop pauses until at least one non zero counter. - * This helps add some initial pause before accessing DB - * for existing values. - */ - - for !evtc.isStopped() { - var val uint64 - - compute_latency(evtc) - - evtc.countersMutex.Lock() - for _, val = range evtc.counters { - if val != 0 { - break - } - } - evtc.countersMutex.Unlock() - - if val != 0 { - break - } - time.Sleep(time.Second) - } - - - /* Populate counters from DB for cumulative counters. */ - if !evtc.isStopped() { - ns, _ := sdcfg.GetDbDefaultNamespace() - addr, err := sdcfg.GetDbTcpAddr("COUNTERS_DB", ns) - if err != nil { - log.Errorf("Address error: %v", err) - return - } - dbId, err := sdcfg.GetDbId("COUNTERS_DB", ns) - if err != nil { - log.Errorf("DB error: %v", err) - return - } - - rclient = redis.NewClient(&redis.Options{ - Network: "tcp", - Addr: addr, - Password: "", // no password set, - DB: dbId, - DialTimeout:0, - }) - - - // Init current values for cumulative keys and clear for absolute - for _, key := range STATS_CUMULATIVE_KEYS { - fv, err := rclient.HGetAll(key).Result() - if err != nil { - number, errC := strconv.ParseUint(fv[STATS_FIELD_NAME], 10, 64) - if errC == nil { - db_counters[key] = number - } - } - } - for _, key := range STATS_ABSOLUTE_KEYS { - db_counters[key] = 0 - } - } - - /* Main running loop that updates DB */ - for !evtc.isStopped() { - tmp_counters := make(map[string]uint64) - - // compute latency - compute_latency(evtc) - - evtc.countersMutex.Lock() - current_counters := evtc.counters - evtc.countersMutex.Unlock() - - for key, val := range current_counters { - tmp_counters[key] = val + db_counters[key] - } - - tmp_counters[DROPPED] += evtc.last_errors - - if (wr_counters == nil) || !reflect.DeepEqual(tmp_counters, *wr_counters) { - for key, val := range tmp_counters { - sval := strconv.FormatUint(val, 10) - ret, err := rclient.HSet(key, STATS_FIELD_NAME, sval).Result() - if !ret { - log.V(3).Infof("EventClient failed to update COUNTERS key:%s val:%v err:%v", - key, sval, err) - } - } - wr_counters = &tmp_counters - } - time.Sleep(time.Second) - } + defer evtc.wg.Done() + + /* Wait for any update */ + db_counters := make(map[string]uint64) + var wr_counters *map[string]uint64 = nil + var rclient *redis.Client + + /* + * This loop pauses until at least one non zero counter. + * This helps add some initial pause before accessing DB + * for existing values. + */ + + for !evtc.isStopped() { + var val uint64 + + compute_latency(evtc) + + evtc.countersMutex.Lock() + for _, val = range evtc.counters { + if val != 0 { + break + } + } + evtc.countersMutex.Unlock() + + if val != 0 { + break + } + time.Sleep(time.Second) + } + + /* Populate counters from DB for cumulative counters. */ + if !evtc.isStopped() { + ns, _ := sdcfg.GetDbDefaultNamespace() + addr, err := sdcfg.GetDbTcpAddr("COUNTERS_DB", ns) + if err != nil { + log.Errorf("Address error: %v", err) + return + } + dbId, err := sdcfg.GetDbId("COUNTERS_DB", ns) + if err != nil { + log.Errorf("DB error: %v", err) + return + } + + rclient = redis.NewClient(&redis.Options{ + Network: "tcp", + Addr: addr, + Password: "", // no password set, + DB: dbId, + DialTimeout: 0, + }) + + // Init current values for cumulative keys and clear for absolute + for _, key := range STATS_CUMULATIVE_KEYS { + fv, err := rclient.HGetAll(key).Result() + if err != nil { + number, errC := strconv.ParseUint(fv[STATS_FIELD_NAME], 10, 64) + if errC == nil { + db_counters[key] = number + } + } + } + for _, key := range STATS_ABSOLUTE_KEYS { + db_counters[key] = 0 + } + } + + /* Main running loop that updates DB */ + for !evtc.isStopped() { + tmp_counters := make(map[string]uint64) + + // compute latency + compute_latency(evtc) + + evtc.countersMutex.Lock() + current_counters := evtc.counters + evtc.countersMutex.Unlock() + + for key, val := range current_counters { + tmp_counters[key] = val + db_counters[key] + } + + tmp_counters[DROPPED] += evtc.last_errors + + if (wr_counters == nil) || !reflect.DeepEqual(tmp_counters, *wr_counters) { + for key, val := range tmp_counters { + sval := strconv.FormatUint(val, 10) + ret, err := rclient.HSet(key, STATS_FIELD_NAME, sval).Result() + if !ret { + log.V(3).Infof("EventClient failed to update COUNTERS key:%s val:%v err:%v", + key, sval, err) + } + } + wr_counters = &tmp_counters + } + time.Sleep(time.Second) + } } - // String returns the target the client is querying. func (evtc *EventClient) String() string { - return fmt.Sprintf("EventClient Prefix %v", evtc.prefix.GetTarget()) + return fmt.Sprintf("EventClient Prefix %v", evtc.prefix.GetTarget()) } var evt_ptr *C.event_receive_op_C_t type Evt_rcvd struct { - Event_str string - Missed_cnt uint32 - Publish_epoch_ms int64 + Event_str string + Missed_cnt uint32 + Publish_epoch_ms int64 } func C_recv_evt(h unsafe.Pointer) (int, Evt_rcvd) { - var evt Evt_rcvd + var evt Evt_rcvd - rc := (int)(C.event_receive_wrap(h, evt_ptr)) - evt.Event_str = C.GoString((*C.char)(evt_ptr.event_str)) - evt.Missed_cnt = (uint32)(evt_ptr.missed_cnt) - evt.Publish_epoch_ms = (int64)(evt_ptr.publish_epoch_ms) + rc := (int)(C.event_receive_wrap(h, evt_ptr)) + evt.Event_str = C.GoString((*C.char)(evt_ptr.event_str)) + evt.Missed_cnt = (uint32)(evt_ptr.missed_cnt) + evt.Publish_epoch_ms = (int64)(evt_ptr.publish_epoch_ms) - return rc, evt + return rc, evt } func C_deinit_subs(h unsafe.Pointer) { - C.events_deinit_subscriber_wrap(h) + C.events_deinit_subscriber_wrap(h) } func get_events(evtc *EventClient) { - defer evtc.wg.Done() - - str_ptr := C.malloc(C.sizeof_char * C.size_t(EVENT_BUFFSZ)) - defer C.free(unsafe.Pointer(str_ptr)) - - evt_ptr = (*C.event_receive_op_C_t)(C.malloc(C.size_t(unsafe.Sizeof(C.event_receive_op_C_t{})))) - defer C.free(unsafe.Pointer(evt_ptr)) - - evt_ptr.event_str = (*C.char)(str_ptr) - evt_ptr.event_sz = C.uint32_t(EVENT_BUFFSZ) - - for { - - rc, evt := C_recv_evt(evtc.subs_handle) - - if rc == 0 { - evtc.countersMutex.Lock() - current_missed_cnt := evtc.counters[MISSED] - evtc.countersMutex.Unlock() - - evtc.countersMutex.RLock() - evtc.counters[MISSED] = current_missed_cnt + (uint64)(evt.Missed_cnt) - evtc.countersMutex.RUnlock() - - if !strings.HasPrefix(evt.Event_str, TEST_EVENT) { - qlen := evtc.q.Len() - - if (qlen < evtc.pq_max) { - var fvp map[string]interface{} - json.Unmarshal([]byte(evt.Event_str), &fvp) - - jv, err := json.Marshal(fvp) - - if err == nil { - evtTv := &gnmipb.TypedValue { - Value: &gnmipb.TypedValue_JsonIetfVal { - JsonIetfVal: jv, - }} - if err := send_event(evtc, evtTv, evt.Publish_epoch_ms); err != nil { - break - } - } else { - log.V(1).Infof("Invalid event string: %v", evt.Event_str) - } - } else { - evtc.countersMutex.Lock() - dropped_cnt := evtc.counters[DROPPED] - evtc.countersMutex.Unlock() - - evtc.countersMutex.RLock() - evtc.counters[DROPPED] = dropped_cnt + 1 - evtc.countersMutex.RUnlock() - } - } - } - if evtc.isStopped() { - break - } - // TODO: Record missed count in stats table. - // intVar, err := strconv.Atoi(C.GoString((*C.char)(c_mptr))) - } - log.V(1).Infof("%v stop channel closed or send_event err, exiting get_events routine", evtc) - C_deinit_subs(evtc.subs_handle) - evtc.subs_handle = nil - // set evtc.stopped for case where send_event error and channel was not stopped - evtc.stopMutex.RLock() - evtc.stopped = 1 - evtc.stopMutex.RUnlock() + defer evtc.wg.Done() + + str_ptr := C.malloc(C.sizeof_char * C.size_t(EVENT_BUFFSZ)) + defer C.free(unsafe.Pointer(str_ptr)) + + evt_ptr = (*C.event_receive_op_C_t)(C.malloc(C.size_t(unsafe.Sizeof(C.event_receive_op_C_t{})))) + defer C.free(unsafe.Pointer(evt_ptr)) + + evt_ptr.event_str = (*C.char)(str_ptr) + evt_ptr.event_sz = C.uint32_t(EVENT_BUFFSZ) + + for { + + rc, evt := C_recv_evt(evtc.subs_handle) + + if rc == 0 { + evtc.countersMutex.Lock() + current_missed_cnt := evtc.counters[MISSED] + evtc.countersMutex.Unlock() + + evtc.countersMutex.RLock() + evtc.counters[MISSED] = current_missed_cnt + (uint64)(evt.Missed_cnt) + evtc.countersMutex.RUnlock() + + if !strings.HasPrefix(evt.Event_str, TEST_EVENT) { + qlen := evtc.q.Len() + + if qlen < evtc.pq_max { + var fvp map[string]interface{} + json.Unmarshal([]byte(evt.Event_str), &fvp) + + jv, err := json.Marshal(fvp) + + if err == nil { + evtTv := &gnmipb.TypedValue{ + Value: &gnmipb.TypedValue_JsonIetfVal{ + JsonIetfVal: jv, + }} + if err := send_event(evtc, evtTv, evt.Publish_epoch_ms); err != nil { + break + } + } else { + log.V(1).Infof("Invalid event string: %v", evt.Event_str) + } + } else { + evtc.countersMutex.Lock() + dropped_cnt := evtc.counters[DROPPED] + evtc.countersMutex.Unlock() + + evtc.countersMutex.RLock() + evtc.counters[DROPPED] = dropped_cnt + 1 + evtc.countersMutex.RUnlock() + } + } + } + if evtc.isStopped() { + break + } + // TODO: Record missed count in stats table. + // intVar, err := strconv.Atoi(C.GoString((*C.char)(c_mptr))) + } + log.V(1).Infof("%v stop channel closed or send_event err, exiting get_events routine", evtc) + C_deinit_subs(evtc.subs_handle) + evtc.subs_handle = nil + // set evtc.stopped for case where send_event error and channel was not stopped + evtc.stopMutex.RLock() + evtc.stopped = 1 + evtc.stopMutex.RUnlock() } func send_event(evtc *EventClient, tv *gnmipb.TypedValue, - timestamp int64) error { - spbv := &spb.Value{ - Prefix: evtc.prefix, - Path: evtc.path, - Timestamp: timestamp, - Val: tv, - } - - if err := evtc.q.Put(Value{spbv}); err != nil { - log.V(3).Infof("Queue error: %v", err) - return err - } - return nil + timestamp int64) error { + spbv := &spb.Value{ + Prefix: evtc.prefix, + Path: evtc.path, + Timestamp: timestamp, + Val: tv, + } + + if err := evtc.q.Put(Value{spbv}); err != nil { + log.V(3).Infof("Queue error: %v", err) + return err + } + return nil } func (evtc *EventClient) StreamRun(q *queue.PriorityQueue, stop chan struct{}, wg *sync.WaitGroup, subscribe *gnmipb.SubscriptionList) { - evtc.wg = wg - defer evtc.wg.Done() - - evtc.q = q - evtc.channel = stop - - go get_events(evtc) - evtc.wg.Add(1) - go update_stats(evtc) - evtc.wg.Add(1) - - for !evtc.isStopped() { - select { - case <-evtc.channel: - evtc.stopMutex.RLock() - evtc.stopped = 1 - evtc.stopMutex.RUnlock() - log.V(3).Infof("Channel closed by client") - return - } - } + evtc.wg = wg + defer evtc.wg.Done() + + evtc.q = q + evtc.channel = stop + + go get_events(evtc) + evtc.wg.Add(1) + go update_stats(evtc) + evtc.wg.Add(1) + + for !evtc.isStopped() { + select { + case <-evtc.channel: + evtc.stopMutex.RLock() + evtc.stopped = 1 + evtc.stopMutex.RUnlock() + log.V(3).Infof("Channel closed by client") + return + } + } } func (evtc *EventClient) isStopped() bool { - evtc.stopMutex.Lock() - val := evtc.stopped - evtc.stopMutex.Unlock() - return val == 1 + evtc.stopMutex.Lock() + val := evtc.stopped + evtc.stopMutex.Unlock() + return val == 1 } - func (evtc *EventClient) Get(wg *sync.WaitGroup) ([]*spb.Value, error) { - return nil, nil + return nil, nil } func (evtc *EventClient) OnceRun(q *queue.PriorityQueue, once chan struct{}, wg *sync.WaitGroup, subscribe *gnmipb.SubscriptionList) { - return + return } func (evtc *EventClient) PollRun(q *queue.PriorityQueue, poll chan struct{}, wg *sync.WaitGroup, subscribe *gnmipb.SubscriptionList) { - return + return } - func (evtc *EventClient) Close() error { - return nil + return nil } -func (evtc *EventClient) Set(delete []*gnmipb.Path, replace []*gnmipb.Update, update []*gnmipb.Update) error { - return nil +func (evtc *EventClient) Set(delete []*gnmipb.Path, replace []*gnmipb.Update, update []*gnmipb.Update) error { + return nil } func (evtc *EventClient) Capabilities() []gnmipb.ModelData { - return nil + return nil } func (c *EventClient) SentOne(val *Value) { - var udiff uint64 + var udiff uint64 - diff := time.Now().UnixNano() - val.GetTimestamp() - udiff = (uint64)(diff) + diff := time.Now().UnixNano() - val.GetTimestamp() + udiff = (uint64)(diff) - c.last_latencies[c.last_latency_index] = udiff - c.last_latency_index += 1 - if c.last_latency_index >= len(c.last_latencies) { - c.last_latency_index = 0 - c.last_latency_full = true - } + c.last_latencies[c.last_latency_index] = udiff + c.last_latency_index += 1 + if c.last_latency_index >= len(c.last_latencies) { + c.last_latency_index = 0 + c.last_latency_full = true + } } func (c *EventClient) FailedSend() { - c.last_errors += 1 + c.last_errors += 1 } - // cgo LDFLAGS: -L/sonic/target/files/bullseye -lxswsscommon -lpthread -lboost_thread -lboost_system -lzmq -lboost_serialization -luuid -lxxeventxx -Wl,-rpath,/sonic/target/files/bullseye - diff --git a/sonic_data_client/json_client.go b/sonic_data_client/json_client.go index d54f5dee..d5b8f653 100644 --- a/sonic_data_client/json_client.go +++ b/sonic_data_client/json_client.go @@ -1,11 +1,11 @@ package client import ( - "os" + "encoding/json" "fmt" - "strconv" "io/ioutil" - "encoding/json" + "os" + "strconv" log "github.com/golang/glog" ) @@ -85,9 +85,9 @@ func NewJsonClient(fileName string) (*JsonClient, error) { return nil, err } defer jsonFile.Close() - + jsonData, err := ioutil.ReadAll(jsonFile) - if err!= nil { + if err != nil { return nil, err } res, err := parseJson([]byte(jsonData)) @@ -293,7 +293,7 @@ func (c *JsonClient) Add(path []string, value string) error { log.V(2).Infof("Invalid db table Path %v", path) return fmt.Errorf("Invalid db table Path %v", path) } - + return nil } @@ -382,4 +382,4 @@ func (c *JsonClient) Remove(path []string) error { } return nil -} \ No newline at end of file +} diff --git a/sonic_data_client/mixed_db_client.go b/sonic_data_client/mixed_db_client.go index 0247cfd3..d379e56b 100644 --- a/sonic_data_client/mixed_db_client.go +++ b/sonic_data_client/mixed_db_client.go @@ -8,6 +8,16 @@ import ( "bytes" "encoding/json" "fmt" + "github.com/Workiva/go-datastructures/queue" + log "github.com/golang/glog" + gnmipb "github.com/openconfig/gnmi/proto/gnmi" + "github.com/sonic-net/sonic-gnmi/common_utils" + spb "github.com/sonic-net/sonic-gnmi/proto" + sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" + ssc "github.com/sonic-net/sonic-gnmi/sonic_service_client" + "github.com/sonic-net/sonic-gnmi/swsscommon" + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" "io/ioutil" "os" "reflect" @@ -16,16 +26,6 @@ import ( "sync" "time" "unsafe" - "google.golang.org/grpc/codes" - "google.golang.org/grpc/status" - log "github.com/golang/glog" - "github.com/Workiva/go-datastructures/queue" - "github.com/sonic-net/sonic-gnmi/common_utils" - "github.com/sonic-net/sonic-gnmi/swsscommon" - sdcfg "github.com/sonic-net/sonic-gnmi/sonic_db_config" - spb "github.com/sonic-net/sonic-gnmi/proto" - ssc "github.com/sonic-net/sonic-gnmi/sonic_service_client" - gnmipb "github.com/openconfig/gnmi/proto/gnmi" ) const REDIS_SOCK string = "/var/run/redis/redis.sock" @@ -39,8 +39,8 @@ const RETRY_DELAY_FACTOR uint = 2 const CHECK_POINT_PATH string = "/etc/sonic" const ( - opAdd = iota - opRemove + opAdd = iota + opRemove ) var ( @@ -54,19 +54,19 @@ var ( ) type MixedDbClient struct { - prefix *gnmipb.Path - paths []*gnmipb.Path - pathG2S map[*gnmipb.Path][]tablePath - encoding gnmipb.Encoding - q *queue.PriorityQueue - channel chan struct{} - target string - origin string - workPath string - jClient *JsonClient - applDB swsscommon.DBConnector + prefix *gnmipb.Path + paths []*gnmipb.Path + pathG2S map[*gnmipb.Path][]tablePath + encoding gnmipb.Encoding + q *queue.PriorityQueue + channel chan struct{} + target string + origin string + workPath string + jClient *JsonClient + applDB swsscommon.DBConnector zmqClient swsscommon.ZmqClient - tableMap map[string]swsscommon.ProducerStateTable + tableMap map[string]swsscommon.ProducerStateTable synced sync.WaitGroup // Control when to send gNMI sync_response w *sync.WaitGroup // wait for all sub go routines to finish @@ -75,12 +75,12 @@ type MixedDbClient struct { var mixedDbClientMap = map[string]MixedDbClient{} -func getMixedDbClient(zmqAddress string) (MixedDbClient) { +func getMixedDbClient(zmqAddress string) MixedDbClient { client, ok := mixedDbClientMap[zmqAddress] if !ok { - client = MixedDbClient { - applDB : swsscommon.NewDBConnector(APPL_DB_NAME, SWSS_TIMEOUT, false), - tableMap : map[string]swsscommon.ProducerStateTable{}, + client = MixedDbClient{ + applDB: swsscommon.NewDBConnector(APPL_DB_NAME, SWSS_TIMEOUT, false), + tableMap: map[string]swsscommon.ProducerStateTable{}, } // enable ZMQ by zmqAddress parameter @@ -89,7 +89,7 @@ func getMixedDbClient(zmqAddress string) (MixedDbClient) { } else { client.zmqClient = nil } - + mixedDbClientMap[zmqAddress] = client } @@ -128,7 +128,7 @@ func ParseTarget(target string, paths []*gnmipb.Path) (string, error) { return target, nil } -func (c *MixedDbClient) GetTable(table string) (swsscommon.ProducerStateTable) { +func (c *MixedDbClient) GetTable(table string) swsscommon.ProducerStateTable { pt, ok := c.tableMap[table] if !ok { if strings.HasPrefix(table, DASH_TABLE_PREFIX) && c.zmqClient != nil { @@ -146,9 +146,9 @@ func (c *MixedDbClient) GetTable(table string) (swsscommon.ProducerStateTable) { } func CatchException(err *error) { - if r := recover(); r != nil { - *err = fmt.Errorf("%v", r) - } + if r := recover(); r != nil { + *err = fmt.Errorf("%v", r) + } } func ProducerStateTableSetWrapper(pt swsscommon.ProducerStateTable, key string, value swsscommon.FieldValuePairs) (err error) { @@ -174,7 +174,7 @@ func RetryHelper(zmqClient swsscommon.ZmqClient, action ActionNeedRetry) { for { err := action() if err != nil { - if (err.Error() == ConnectionResetErr && retry <= MAX_RETRY_COUNT) { + if err.Error() == ConnectionResetErr && retry <= MAX_RETRY_COUNT { log.V(6).Infof("RetryHelper: connection reset, reconnect and retry later") time.Sleep(retry_delay) @@ -202,20 +202,20 @@ func (c *MixedDbClient) DbSetTable(table string, key string, values map[string]s pt := c.GetTable(table) RetryHelper( - c.zmqClient, - func () error { - return ProducerStateTableSetWrapper(pt, key, vec) - }) + c.zmqClient, + func() error { + return ProducerStateTableSetWrapper(pt, key, vec) + }) return nil } func (c *MixedDbClient) DbDelTable(table string, key string) error { pt := c.GetTable(table) RetryHelper( - c.zmqClient, - func () error { - return ProducerStateTableDeleteWrapper(pt, key) - }) + c.zmqClient, + func() error { + return ProducerStateTableDeleteWrapper(pt, key) + }) return nil } @@ -467,7 +467,7 @@ func (c *MixedDbClient) makeJSON_redis(msi *map[string]interface{}, key *string, fp[k] = slice } else { fp[f] = v - } + } } if key == nil { @@ -534,7 +534,7 @@ func (c *MixedDbClient) tableData2Msi(tblPath *tablePath, useKey bool, op *strin return err } - if (tblPath.tableName == "") { + if tblPath.tableName == "" { // Split dbkey string into two parts // First part is table name and second part is key in table keys := strings.SplitN(dbkey, tblPath.delimitor, 2) @@ -681,16 +681,16 @@ func (c *MixedDbClient) tableData2TypedValue(tblPaths []tablePath, op *string) ( } func ConvertDbEntry(inputData map[string]interface{}) map[string]string { - outputData := map[string]string{} - for key, value := range inputData { - switch value.(type) { + outputData := map[string]string{} + for key, value := range inputData { + switch value.(type) { case string: outputData[key] = value.(string) case []interface{}: list := value.([]interface{}) key_redis := key + "@" slice := []string{} - for _, item := range(list) { + for _, item := range list { if str, check := item.(string); check { slice = append(slice, str) } else { @@ -699,9 +699,9 @@ func ConvertDbEntry(inputData map[string]interface{}) map[string]string { } str_val := strings.Join(slice, ",") outputData[key_redis] = str_val - } - } - return outputData + } + } + return outputData } func (c *MixedDbClient) handleTableData(tblPaths []tablePath) error { @@ -745,7 +745,7 @@ func (c *MixedDbClient) handleTableData(tblPaths []tablePath) error { } for _, dbkey := range dbkeys { - tableKey := strings.TrimPrefix(dbkey, tblPath.tableName + tblPath.delimitor) + tableKey := strings.TrimPrefix(dbkey, tblPath.tableName+tblPath.delimitor) err = c.DbDelTable(tblPath.tableName, tableKey) if err != nil { log.V(2).Infof("swsscommon delete failed for %v, dbkey %s", tblPath, dbkey) @@ -877,8 +877,7 @@ func RunPyCode(text string) error { return nil } -var PyCodeForYang string = -` +var PyCodeForYang string = ` import sonic_yang import json @@ -1061,7 +1060,7 @@ func (c *MixedDbClient) SetDB(delete []*gnmipb.Path, replace []*gnmipb.Update, u if err != nil { return err } - + for _, tblPaths := range deleteMap { err = c.handleTableData(tblPaths) if err != nil { @@ -1109,7 +1108,7 @@ func (c *MixedDbClient) SetConfigDB(delete []*gnmipb.Path, replace []*gnmipb.Upd deleteLen := len(delete) replaceLen := len(replace) updateLen := len(update) - if (deleteLen == 1 && replaceLen == 0 && updateLen == 1) { + if deleteLen == 1 && replaceLen == 0 && updateLen == 1 { deletePath := c.gnmiFullPath(c.prefix, delete[0]) updatePath := c.gnmiFullPath(c.prefix, update[0].GetPath()) if (len(deletePath.GetElem()) == 0) && (len(updatePath.GetElem()) == 0) { @@ -1157,8 +1156,8 @@ func (c *MixedDbClient) GetCheckPoint() ([]*spb.Value, error) { } val := gnmipb.TypedValue{ - Value: &gnmipb.TypedValue_JsonIetfVal{JsonIetfVal: jv}, - } + Value: &gnmipb.TypedValue_JsonIetfVal{JsonIetfVal: jv}, + } values = append(values, &spb.Value{ Prefix: c.prefix, Path: path, @@ -1234,4 +1233,3 @@ func (c *MixedDbClient) SentOne(val *Value) { func (c *MixedDbClient) FailedSend() { } - diff --git a/sonic_data_client/non_db_client.go b/sonic_data_client/non_db_client.go index 54324065..aa5170e4 100644 --- a/sonic_data_client/non_db_client.go +++ b/sonic_data_client/non_db_client.go @@ -9,11 +9,11 @@ import ( "gopkg.in/yaml.v2" - spb "github.com/sonic-net/sonic-gnmi/proto" "github.com/Workiva/go-datastructures/queue" linuxproc "github.com/c9s/goprocinfo/linux" log "github.com/golang/glog" gnmipb "github.com/openconfig/gnmi/proto/gnmi" + spb "github.com/sonic-net/sonic-gnmi/proto" ) // Non db client is to Handle @@ -137,7 +137,7 @@ func getCpuUtilPercents(cur, last *linuxproc.CPUStat) uint64 { idleTicks := cur.Idle - last.Idle totalTicks := curTotal - lastTotal if totalTicks == 0 { // No change in CPU Utilization - return 0 + return 0 } return 100 * (totalTicks - idleTicks) / totalTicks } @@ -588,7 +588,7 @@ func (c *NonDbClient) Close() error { return nil } -func (c *NonDbClient) Set(delete []*gnmipb.Path, replace []*gnmipb.Update, update []*gnmipb.Update) error { +func (c *NonDbClient) Set(delete []*gnmipb.Path, replace []*gnmipb.Update, update []*gnmipb.Update) error { return nil } func (c *NonDbClient) Capabilities() []gnmipb.ModelData { @@ -599,4 +599,3 @@ func (c *NonDbClient) SentOne(val *Value) { func (c *NonDbClient) FailedSend() { } - diff --git a/sonic_data_client/virtual_db.go b/sonic_data_client/virtual_db.go index c3ded154..f390e47e 100644 --- a/sonic_data_client/virtual_db.go +++ b/sonic_data_client/virtual_db.go @@ -342,9 +342,13 @@ func v2rEthPortStats(paths []string) ([]tablePath, error) { // Supported cases: // <1> port name having suffix of "*" with specific field; -// Ex. [COUNTER_DB COUNTERS Ethernet* SAI_PORT_STAT_PFC_0_RX_PKTS] +// +// Ex. [COUNTER_DB COUNTERS Ethernet* SAI_PORT_STAT_PFC_0_RX_PKTS] +// // <2> exact port name with specific field. -// Ex. [COUNTER_DB COUNTERS Ethernet68 SAI_PORT_STAT_PFC_0_RX_PKTS] +// +// Ex. [COUNTER_DB COUNTERS Ethernet68 SAI_PORT_STAT_PFC_0_RX_PKTS] +// // case of "*" field could be covered in v2rEthPortStats() func v2rEthPortFieldStats(paths []string) ([]tablePath, error) { var tblPaths []tablePath diff --git a/sonic_db_config/db_config.go b/sonic_db_config/db_config.go index ad7c2f3c..dcea3238 100644 --- a/sonic_db_config/db_config.go +++ b/sonic_db_config/db_config.go @@ -1,12 +1,12 @@ -//Package dbconfig provides a generic functions for parsing sonic database config file in system -//package main +// Package dbconfig provides a generic functions for parsing sonic database config file in system +// package main package dbconfig import ( - "os" "fmt" - "strconv" "github.com/sonic-net/sonic-gnmi/swsscommon" + "os" + "strconv" ) const ( diff --git a/sonic_db_config/db_config_test.go b/sonic_db_config/db_config_test.go index da85b866..2b6656d5 100644 --- a/sonic_db_config/db_config_test.go +++ b/sonic_db_config/db_config_test.go @@ -2,9 +2,9 @@ package dbconfig import ( "fmt" - "testing" - "github.com/sonic-net/sonic-gnmi/test_utils" "github.com/agiledragon/gomonkey/v2" + "github.com/sonic-net/sonic-gnmi/test_utils" + "testing" ) func TestGetDb(t *testing.T) { @@ -184,4 +184,3 @@ func TestGetDbMultiNs(t *testing.T) { } }) } - diff --git a/sonic_service_client/dbus_client.go b/sonic_service_client/dbus_client.go index d3069e8f..515a947d 100644 --- a/sonic_service_client/dbus_client.go +++ b/sonic_service_client/dbus_client.go @@ -1,12 +1,12 @@ package host_service import ( - "time" "fmt" - "reflect" - log "github.com/golang/glog" "github.com/godbus/dbus/v5" + log "github.com/golang/glog" "github.com/sonic-net/sonic-gnmi/common_utils" + "reflect" + "time" ) type Service interface { @@ -14,7 +14,7 @@ type Service interface { ConfigSave(fileName string) error ApplyPatchYang(fileName string) error ApplyPatchDb(fileName string) error - CreateCheckPoint(cpName string) error + CreateCheckPoint(cpName string) error DeleteCheckPoint(cpName string) error } @@ -22,7 +22,7 @@ type DbusClient struct { busNamePrefix string busPathPrefix string intNamePrefix string - channel chan struct{} + channel chan struct{} } func NewDbusClient() (Service, error) { diff --git a/sonic_service_client/dbus_client_test.go b/sonic_service_client/dbus_client_test.go index aced4123..9dc231ca 100644 --- a/sonic_service_client/dbus_client_test.go +++ b/sonic_service_client/dbus_client_test.go @@ -1,409 +1,409 @@ -package host_service - -import ( - "testing" - "reflect" - - "github.com/agiledragon/gomonkey/v2" - "github.com/godbus/dbus/v5" -) - -func TestSystemBusNegative(t *testing.T) { - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ConfigReload("abc") - if err == nil { - t.Errorf("SystemBus should fail") - } -} - -func TestConfigReload(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.config.reload" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(0) - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ConfigReload("abc") - if err != nil { - t.Errorf("ConfigReload should pass: %v", err) - } -} - -func TestConfigReloadNegative(t *testing.T) { - err_msg := "This is the mock error message" - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.config.reload" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(1) - ret.Body[1] = err_msg - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ConfigReload("abc") - if err == nil { - t.Errorf("ConfigReload should fail") - } - if err.Error() != err_msg { - t.Errorf("Wrong error: %v", err) - } -} - -func TestConfigReloadTimeout(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.config.reload" { - t.Errorf("Wrong method: %v", method) - } - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ConfigReload("abc") - if err == nil { - t.Errorf("ConfigReload should timeout: %v", err) - } -} - -func TestConfigSave(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.config.save" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(0) - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ConfigSave("abc") - if err != nil { - t.Errorf("ConfigSave should pass: %v", err) - } -} - -func TestConfigSaveNegative(t *testing.T) { - err_msg := "This is the mock error message" - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.config.save" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(1) - ret.Body[1] = err_msg - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ConfigSave("abc") - if err == nil { - t.Errorf("ConfigSave should fail") - } - if err.Error() != err_msg { - t.Errorf("Wrong error: %v", err) - } -} - -func TestApplyPatchYang(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.apply_patch_yang" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(0) - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ApplyPatchYang("abc") - if err != nil { - t.Errorf("ApplyPatchYang should pass: %v", err) - } -} - -func TestApplyPatchYangNegative(t *testing.T) { - err_msg := "This is the mock error message" - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.apply_patch_yang" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(1) - ret.Body[1] = err_msg - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ApplyPatchYang("abc") - if err == nil { - t.Errorf("ApplyPatchYang should fail") - } - if err.Error() != err_msg { - t.Errorf("Wrong error: %v", err) - } -} - -func TestApplyPatchDb(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.apply_patch_db" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(0) - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ApplyPatchDb("abc") - if err != nil { - t.Errorf("ApplyPatchDb should pass: %v", err) - } -} - -func TestApplyPatchDbNegative(t *testing.T) { - err_msg := "This is the mock error message" - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.apply_patch_db" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(1) - ret.Body[1] = err_msg - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.ApplyPatchDb("abc") - if err == nil { - t.Errorf("ApplyPatchDb should fail") - } - if err.Error() != err_msg { - t.Errorf("Wrong error: %v", err) - } -} - -func TestCreateCheckPoint(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.create_checkpoint" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(0) - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.CreateCheckPoint("abc") - if err != nil { - t.Errorf("CreateCheckPoint should pass: %v", err) - } -} - -func TestCreateCheckPointNegative(t *testing.T) { - err_msg := "This is the mock error message" - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.create_checkpoint" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(1) - ret.Body[1] = err_msg - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.CreateCheckPoint("abc") - if err == nil { - t.Errorf("CreateCheckPoint should fail") - } - if err.Error() != err_msg { - t.Errorf("Wrong error: %v", err) - } -} - -func TestDeleteCheckPoint(t *testing.T) { - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.delete_checkpoint" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(0) - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.DeleteCheckPoint("abc") - if err != nil { - t.Errorf("DeleteCheckPoint should pass: %v", err) - } -} - -func TestDeleteCheckPointNegative(t *testing.T) { - err_msg := "This is the mock error message" - mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { - return &dbus.Conn{}, nil - }) - defer mock1.Reset() - mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { - if method != "org.SONiC.HostService.gcu.delete_checkpoint" { - t.Errorf("Wrong method: %v", method) - } - ret := &dbus.Call{} - ret.Err = nil - ret.Body = make([]interface{}, 2) - ret.Body[0] = int32(1) - ret.Body[1] = err_msg - ch <- ret - return &dbus.Call{} - }) - defer mock2.Reset() - - client, err := NewDbusClient() - if err != nil { - t.Errorf("NewDbusClient failed: %v", err) - } - err = client.DeleteCheckPoint("abc") - if err == nil { - t.Errorf("DeleteCheckPoint should fail") - } - if err.Error() != err_msg { - t.Errorf("Wrong error: %v", err) - } -} +package host_service + +import ( + "reflect" + "testing" + + "github.com/agiledragon/gomonkey/v2" + "github.com/godbus/dbus/v5" +) + +func TestSystemBusNegative(t *testing.T) { + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ConfigReload("abc") + if err == nil { + t.Errorf("SystemBus should fail") + } +} + +func TestConfigReload(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.config.reload" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(0) + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ConfigReload("abc") + if err != nil { + t.Errorf("ConfigReload should pass: %v", err) + } +} + +func TestConfigReloadNegative(t *testing.T) { + err_msg := "This is the mock error message" + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.config.reload" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(1) + ret.Body[1] = err_msg + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ConfigReload("abc") + if err == nil { + t.Errorf("ConfigReload should fail") + } + if err.Error() != err_msg { + t.Errorf("Wrong error: %v", err) + } +} + +func TestConfigReloadTimeout(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.config.reload" { + t.Errorf("Wrong method: %v", method) + } + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ConfigReload("abc") + if err == nil { + t.Errorf("ConfigReload should timeout: %v", err) + } +} + +func TestConfigSave(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.config.save" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(0) + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ConfigSave("abc") + if err != nil { + t.Errorf("ConfigSave should pass: %v", err) + } +} + +func TestConfigSaveNegative(t *testing.T) { + err_msg := "This is the mock error message" + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.config.save" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(1) + ret.Body[1] = err_msg + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ConfigSave("abc") + if err == nil { + t.Errorf("ConfigSave should fail") + } + if err.Error() != err_msg { + t.Errorf("Wrong error: %v", err) + } +} + +func TestApplyPatchYang(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.apply_patch_yang" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(0) + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ApplyPatchYang("abc") + if err != nil { + t.Errorf("ApplyPatchYang should pass: %v", err) + } +} + +func TestApplyPatchYangNegative(t *testing.T) { + err_msg := "This is the mock error message" + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.apply_patch_yang" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(1) + ret.Body[1] = err_msg + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ApplyPatchYang("abc") + if err == nil { + t.Errorf("ApplyPatchYang should fail") + } + if err.Error() != err_msg { + t.Errorf("Wrong error: %v", err) + } +} + +func TestApplyPatchDb(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.apply_patch_db" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(0) + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ApplyPatchDb("abc") + if err != nil { + t.Errorf("ApplyPatchDb should pass: %v", err) + } +} + +func TestApplyPatchDbNegative(t *testing.T) { + err_msg := "This is the mock error message" + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.apply_patch_db" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(1) + ret.Body[1] = err_msg + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.ApplyPatchDb("abc") + if err == nil { + t.Errorf("ApplyPatchDb should fail") + } + if err.Error() != err_msg { + t.Errorf("Wrong error: %v", err) + } +} + +func TestCreateCheckPoint(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.create_checkpoint" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(0) + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.CreateCheckPoint("abc") + if err != nil { + t.Errorf("CreateCheckPoint should pass: %v", err) + } +} + +func TestCreateCheckPointNegative(t *testing.T) { + err_msg := "This is the mock error message" + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.create_checkpoint" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(1) + ret.Body[1] = err_msg + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.CreateCheckPoint("abc") + if err == nil { + t.Errorf("CreateCheckPoint should fail") + } + if err.Error() != err_msg { + t.Errorf("Wrong error: %v", err) + } +} + +func TestDeleteCheckPoint(t *testing.T) { + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.delete_checkpoint" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(0) + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.DeleteCheckPoint("abc") + if err != nil { + t.Errorf("DeleteCheckPoint should pass: %v", err) + } +} + +func TestDeleteCheckPointNegative(t *testing.T) { + err_msg := "This is the mock error message" + mock1 := gomonkey.ApplyFunc(dbus.SystemBus, func() (conn *dbus.Conn, err error) { + return &dbus.Conn{}, nil + }) + defer mock1.Reset() + mock2 := gomonkey.ApplyMethod(reflect.TypeOf(&dbus.Object{}), "Go", func(obj *dbus.Object, method string, flags dbus.Flags, ch chan *dbus.Call, args ...interface{}) *dbus.Call { + if method != "org.SONiC.HostService.gcu.delete_checkpoint" { + t.Errorf("Wrong method: %v", method) + } + ret := &dbus.Call{} + ret.Err = nil + ret.Body = make([]interface{}, 2) + ret.Body[0] = int32(1) + ret.Body[1] = err_msg + ch <- ret + return &dbus.Call{} + }) + defer mock2.Reset() + + client, err := NewDbusClient() + if err != nil { + t.Errorf("NewDbusClient failed: %v", err) + } + err = client.DeleteCheckPoint("abc") + if err == nil { + t.Errorf("DeleteCheckPoint should fail") + } + if err.Error() != err_msg { + t.Errorf("Wrong error: %v", err) + } +} diff --git a/telemetry/telemetry.go b/telemetry/telemetry.go index 6cc128fe..674595a0 100644 --- a/telemetry/telemetry.go +++ b/telemetry/telemetry.go @@ -1,9 +1,9 @@ package main import ( + "crypto/md5" "crypto/tls" "crypto/x509" - "crypto/md5" "flag" "io/ioutil" "strconv" @@ -20,22 +20,22 @@ import ( var ( userAuth = gnmi.AuthTypes{"password": false, "cert": false, "jwt": false} - port = flag.Int("port", -1, "port to listen on") + port = flag.Int("port", -1, "port to listen on") // Certificate files. - caCert = flag.String("ca_crt", "", "CA certificate for client certificate validation. Optional.") - serverCert = flag.String("server_crt", "", "TLS server certificate") - serverKey = flag.String("server_key", "", "TLS server private key") - zmqAddress = flag.String("zmq_address", "", "Orchagent ZMQ address, when not set or empty string telemetry server will switch to Redis based communication channel.") - insecure = flag.Bool("insecure", false, "Skip providing TLS cert and key, for testing only!") - noTLS = flag.Bool("noTLS", false, "disable TLS, for testing only!") - allowNoClientCert = flag.Bool("allow_no_client_auth", false, "When set, telemetry server will request but not require a client certificate.") - jwtRefInt = flag.Uint64("jwt_refresh_int", 900, "Seconds before JWT expiry the token can be refreshed.") - jwtValInt = flag.Uint64("jwt_valid_int", 3600, "Seconds that JWT token is valid for.") - gnmi_translib_write = flag.Bool("gnmi_translib_write", gnmi.ENABLE_TRANSLIB_WRITE, "Enable gNMI translib write for management framework") - gnmi_native_write = flag.Bool("gnmi_native_write", gnmi.ENABLE_NATIVE_WRITE, "Enable gNMI native write") - threshold = flag.Int("threshold", 100, "max number of client connections") + caCert = flag.String("ca_crt", "", "CA certificate for client certificate validation. Optional.") + serverCert = flag.String("server_crt", "", "TLS server certificate") + serverKey = flag.String("server_key", "", "TLS server private key") + zmqAddress = flag.String("zmq_address", "", "Orchagent ZMQ address, when not set or empty string telemetry server will switch to Redis based communication channel.") + insecure = flag.Bool("insecure", false, "Skip providing TLS cert and key, for testing only!") + noTLS = flag.Bool("noTLS", false, "disable TLS, for testing only!") + allowNoClientCert = flag.Bool("allow_no_client_auth", false, "When set, telemetry server will request but not require a client certificate.") + jwtRefInt = flag.Uint64("jwt_refresh_int", 900, "Seconds before JWT expiry the token can be refreshed.") + jwtValInt = flag.Uint64("jwt_valid_int", 3600, "Seconds that JWT token is valid for.") + gnmi_translib_write = flag.Bool("gnmi_translib_write", gnmi.ENABLE_TRANSLIB_WRITE, "Enable gNMI translib write for management framework") + gnmi_native_write = flag.Bool("gnmi_native_write", gnmi.ENABLE_NATIVE_WRITE, "Enable gNMI native write") + threshold = flag.Int("threshold", 100, "max number of client connections") withMasterArbitration = flag.Bool("with-master-arbitration", false, "Enables master arbitration policy.") - idle_conn_duration = flag.Int("idle_conn_duration", 5, "Seconds before server closes idle connections") + idle_conn_duration = flag.Int("idle_conn_duration", 5, "Seconds before server closes idle connections") ) func main() { @@ -46,13 +46,13 @@ func main() { if *gnmi_translib_write { //In read/write mode we want to enable auth by default. defUserAuth = gnmi.AuthTypes{"password": true, "cert": false, "jwt": true} - }else { + } else { defUserAuth = gnmi.AuthTypes{"jwt": false, "password": false, "cert": false} } if isFlagPassed("client_auth") { log.V(1).Infof("client_auth provided") - }else { + } else { log.V(1).Infof("client_auth not provided, using defaults.") userAuth = defUserAuth } @@ -75,8 +75,8 @@ func main() { return } - gnmi.JwtRefreshInt = time.Duration(*jwtRefInt*uint64(time.Second)) - gnmi.JwtValidInt = time.Duration(*jwtValInt*uint64(time.Second)) + gnmi.JwtRefreshInt = time.Duration(*jwtRefInt * uint64(time.Second)) + gnmi.JwtValidInt = time.Duration(*jwtValInt * uint64(time.Second)) cfg := &gnmi.Config{} cfg.Port = int64(*port) @@ -120,59 +120,59 @@ func main() { } tlsCfg := &tls.Config{ - ClientAuth: tls.RequireAndVerifyClientCert, - Certificates: []tls.Certificate{certificate}, - MinVersion: tls.VersionTLS12, - CurvePreferences: []tls.CurveID{tls.CurveP521, tls.CurveP384, tls.CurveP256}, - PreferServerCipherSuites: true, - CipherSuites: []uint16{ - tls.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, - tls.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, - tls.TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305, - tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305, - tls.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, - tls.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256, - }, - } - - if *allowNoClientCert { - // RequestClientCert will ask client for a certificate but won't - // require it to proceed. If certificate is provided, it will be - // verified. - tlsCfg.ClientAuth = tls.RequestClientCert - } - - if *caCert != "" { - ca, err := ioutil.ReadFile(*caCert) - if err != nil { - log.Exitf("could not read CA certificate: %s", err) + ClientAuth: tls.RequireAndVerifyClientCert, + Certificates: []tls.Certificate{certificate}, + MinVersion: tls.VersionTLS12, + CurvePreferences: []tls.CurveID{tls.CurveP521, tls.CurveP384, tls.CurveP256}, + PreferServerCipherSuites: true, + CipherSuites: []uint16{ + tls.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384, + tls.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, + tls.TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305, + tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305, + tls.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, + tls.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256, + }, } - certPool := x509.NewCertPool() - if ok := certPool.AppendCertsFromPEM(ca); !ok { - log.Exit("failed to append CA certificate") + + if *allowNoClientCert { + // RequestClientCert will ask client for a certificate but won't + // require it to proceed. If certificate is provided, it will be + // verified. + tlsCfg.ClientAuth = tls.RequestClientCert } - tlsCfg.ClientCAs = certPool - } else { - if userAuth.Enabled("cert") { - userAuth.Unset("cert") - log.Warning("client_auth mode cert requires ca_crt option. Disabling cert mode authentication.") + + if *caCert != "" { + ca, err := ioutil.ReadFile(*caCert) + if err != nil { + log.Exitf("could not read CA certificate: %s", err) + } + certPool := x509.NewCertPool() + if ok := certPool.AppendCertsFromPEM(ca); !ok { + log.Exit("failed to append CA certificate") + } + tlsCfg.ClientCAs = certPool + } else { + if userAuth.Enabled("cert") { + userAuth.Unset("cert") + log.Warning("client_auth mode cert requires ca_crt option. Disabling cert mode authentication.") + } } - } - keep_alive_params := keepalive.ServerParameters{ - MaxConnectionIdle: time.Duration(cfg.IdleConnDuration) * time.Second, // duration in which idle connection will be closed, default is inf - } + keep_alive_params := keepalive.ServerParameters{ + MaxConnectionIdle: time.Duration(cfg.IdleConnDuration) * time.Second, // duration in which idle connection will be closed, default is inf + } - opts = []grpc.ServerOption{grpc.Creds(credentials.NewTLS(tlsCfg))} + opts = []grpc.ServerOption{grpc.Creds(credentials.NewTLS(tlsCfg))} - if cfg.IdleConnDuration > 0 { // non inf case - opts = append(opts, grpc.KeepaliveParams(keep_alive_params)) - } + if cfg.IdleConnDuration > 0 { // non inf case + opts = append(opts, grpc.KeepaliveParams(keep_alive_params)) + } - cfg.UserAuth = userAuth + cfg.UserAuth = userAuth - gnmi.GenerateJwtSecretKey() -} + gnmi.GenerateJwtSecretKey() + } s, err := gnmi.NewServer(cfg, opts) if err != nil { @@ -183,7 +183,7 @@ func main() { if *withMasterArbitration { s.ReqFromMaster = gnmi.ReqFromMasterEnabledMA } - + log.V(1).Infof("Auth Modes: ", userAuth) log.V(1).Infof("Starting RPC server on address: %s", s.Address()) s.Serve() // blocks until close diff --git a/transl_utils/transl_utils.go b/transl_utils/transl_utils.go index 2160fdbf..095dc6ad 100644 --- a/transl_utils/transl_utils.go +++ b/transl_utils/transl_utils.go @@ -18,33 +18,33 @@ import ( ) var ( - Writer *syslog.Writer + Writer *syslog.Writer ) func __log_audit_msg(ctx context.Context, reqType string, uriPath string, err error) { - var err1 error - username := "invalid" - statusMsg := "failure" - errMsg := "None" - if (err == nil) { - statusMsg = "success" - } else { - errMsg = err.Error() - } - - if Writer == nil { - Writer, err1 = syslog.Dial("", "", (syslog.LOG_LOCAL4), "") - if (err1 != nil) { - log.V(2).Infof("Could not open connection to syslog with error =%v", err1.Error()) - return - } - } - - common_utils.GetUsername(ctx, &username) - - auditMsg := fmt.Sprintf("User \"%s\" request \"%s %s\" status - %s error - %s", - username, reqType, uriPath, statusMsg, errMsg) - Writer.Info(auditMsg) + var err1 error + username := "invalid" + statusMsg := "failure" + errMsg := "None" + if err == nil { + statusMsg = "success" + } else { + errMsg = err.Error() + } + + if Writer == nil { + Writer, err1 = syslog.Dial("", "", (syslog.LOG_LOCAL4), "") + if err1 != nil { + log.V(2).Infof("Could not open connection to syslog with error =%v", err1.Error()) + return + } + } + + common_utils.GetUsername(ctx, &username) + + auditMsg := fmt.Sprintf("User \"%s\" request \"%s %s\" status - %s error - %s", + username, reqType, uriPath, statusMsg, errMsg) + Writer.Info(auditMsg) } func GnmiTranslFullPath(prefix, path *gnmipb.Path) *gnmipb.Path { @@ -102,7 +102,7 @@ func TranslProcessGet(uriPath string, op *string, ctx context.Context) (*gnmipb. var data []byte rc, _ := common_utils.GetContext(ctx) - req := translib.GetRequest{Path:uriPath, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} + req := translib.GetRequest{Path: uriPath, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} if rc.BundleVersion != nil { nver, err := translib.NewVersion(*rc.BundleVersion) if err != nil { @@ -127,11 +127,10 @@ func TranslProcessGet(uriPath string, op *string, ctx context.Context) (*gnmipb. json.Compact(dst, data) jv = dst.Bytes() - /* Fill the values into GNMI data structures . */ return &gnmipb.TypedValue{ Value: &gnmipb.TypedValue_JsonIetfVal{ - JsonIetfVal: jv, + JsonIetfVal: jv, }}, nil } @@ -144,7 +143,7 @@ func TranslProcessDelete(prefix, delPath *gnmipb.Path, ctx context.Context) erro } rc, _ := common_utils.GetContext(ctx) - req := translib.SetRequest{Path:uri, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} + req := translib.SetRequest{Path: uri, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} if rc.BundleVersion != nil { nver, err := translib.NewVersion(*rc.BundleVersion) if err != nil { @@ -157,7 +156,7 @@ func TranslProcessDelete(prefix, delPath *gnmipb.Path, ctx context.Context) erro req.AuthEnabled = true } resp, err := translib.Delete(req) - if err != nil{ + if err != nil { log.V(2).Infof("DELETE operation failed with error =%v, %v", resp.ErrSrc, err.Error()) return err } @@ -174,7 +173,7 @@ func TranslProcessReplace(prefix *gnmipb.Path, entry *gnmipb.Update, ctx context payload := entry.GetVal().GetJsonIetfVal() rc, _ := common_utils.GetContext(ctx) - req := translib.SetRequest{Path:uri, Payload:payload, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} + req := translib.SetRequest{Path: uri, Payload: payload, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} if rc.BundleVersion != nil { nver, err := translib.NewVersion(*rc.BundleVersion) if err != nil { @@ -188,12 +187,11 @@ func TranslProcessReplace(prefix *gnmipb.Path, entry *gnmipb.Update, ctx context } resp, err1 := translib.Replace(req) - if err1 != nil{ + if err1 != nil { log.V(2).Infof("REPLACE operation failed with error =%v, %v", resp.ErrSrc, err1.Error()) return err1 } - return nil } @@ -206,7 +204,7 @@ func TranslProcessUpdate(prefix *gnmipb.Path, entry *gnmipb.Update, ctx context. payload := entry.GetVal().GetJsonIetfVal() rc, _ := common_utils.GetContext(ctx) - req := translib.SetRequest{Path:uri, Payload:payload, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} + req := translib.SetRequest{Path: uri, Payload: payload, User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}} if rc.BundleVersion != nil { nver, err := translib.NewVersion(*rc.BundleVersion) if err != nil { @@ -219,7 +217,7 @@ func TranslProcessUpdate(prefix *gnmipb.Path, entry *gnmipb.Update, ctx context. req.AuthEnabled = true } resp, err := translib.Update(req) - if err != nil{ + if err != nil { switch err.(type) { case tlerr.NotFoundError: //If Update fails, it may be due to object not existing in this case use Replace to create and update the object. @@ -229,7 +227,7 @@ func TranslProcessUpdate(prefix *gnmipb.Path, entry *gnmipb.Update, ctx context. return err } } - if err != nil{ + if err != nil { log.V(2).Infof("UPDATE operation failed with error =%v, %v", resp.ErrSrc, err.Error()) return err } @@ -240,9 +238,9 @@ func TranslProcessBulk(delete []*gnmipb.Path, replace []*gnmipb.Update, update [ var br translib.BulkRequest var uri string - var deleteUri []string - var replaceUri []string - var updateUri []string + var deleteUri []string + var replaceUri []string + var updateUri []string rc, ctx := common_utils.GetContext(ctx) log.V(2).Info("TranslProcessBulk Called") @@ -255,7 +253,7 @@ func TranslProcessBulk(delete []*gnmipb.Path, replace []*gnmipb.Update, update [ return err } } - for _,d := range delete { + for _, d := range delete { if uri, err = ConvertToURI(prefix, d); err != nil { return err } @@ -270,17 +268,17 @@ func TranslProcessBulk(delete []*gnmipb.Path, replace []*gnmipb.Update, update [ req.AuthEnabled = true } br.DeleteRequest = append(br.DeleteRequest, req) - deleteUri = append(deleteUri, uri) + deleteUri = append(deleteUri, uri) } - for _,r := range replace { + for _, r := range replace { if uri, err = ConvertToURI(prefix, r.GetPath()); err != nil { return err } payload := r.GetVal().GetJsonIetfVal() req := translib.SetRequest{ - Path: uri, + Path: uri, Payload: payload, - User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}, + User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}, } if rc.BundleVersion != nil { req.ClientVersion = nver @@ -289,17 +287,17 @@ func TranslProcessBulk(delete []*gnmipb.Path, replace []*gnmipb.Update, update [ req.AuthEnabled = true } br.ReplaceRequest = append(br.ReplaceRequest, req) - replaceUri = append(replaceUri, uri) + replaceUri = append(replaceUri, uri) } - for _,u := range update { + for _, u := range update { if uri, err = ConvertToURI(prefix, u.GetPath()); err != nil { return err } payload := u.GetVal().GetJsonIetfVal() req := translib.SetRequest{ - Path: uri, + Path: uri, Payload: payload, - User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}, + User: translib.UserRoles{Name: rc.Auth.User, Roles: rc.Auth.Roles}, } if rc.BundleVersion != nil { req.ClientVersion = nver @@ -308,43 +306,43 @@ func TranslProcessBulk(delete []*gnmipb.Path, replace []*gnmipb.Update, update [ req.AuthEnabled = true } br.UpdateRequest = append(br.UpdateRequest, req) - updateUri = append(updateUri, uri) - } - - resp,err := translib.Bulk(br) - - i := 0 - for _,d := range resp.DeleteResponse { - __log_audit_msg(ctx, "DELETE", deleteUri[i], d.Err) - i++ - } - i = 0 - for _,r := range resp.ReplaceResponse { - __log_audit_msg(ctx, "REPLACE", replaceUri[i], r.Err) - i++ - } - i = 0 - for _,u := range resp.UpdateResponse { - __log_audit_msg(ctx, "UPDATE", updateUri[i], u.Err) - i++ - } + updateUri = append(updateUri, uri) + } + + resp, err := translib.Bulk(br) + + i := 0 + for _, d := range resp.DeleteResponse { + __log_audit_msg(ctx, "DELETE", deleteUri[i], d.Err) + i++ + } + i = 0 + for _, r := range resp.ReplaceResponse { + __log_audit_msg(ctx, "REPLACE", replaceUri[i], r.Err) + i++ + } + i = 0 + for _, u := range resp.UpdateResponse { + __log_audit_msg(ctx, "UPDATE", updateUri[i], u.Err) + i++ + } var errors []string - if err != nil{ + if err != nil { log.V(2).Info("BULK SET operation failed with error(s):") - for _,d := range resp.DeleteResponse { + for _, d := range resp.DeleteResponse { if d.Err != nil { log.V(2).Infof("%s=%v", d.Err.Error(), d.ErrSrc) errors = append(errors, d.Err.Error()) } } - for _,r := range resp.ReplaceResponse { + for _, r := range resp.ReplaceResponse { if r.Err != nil { log.V(2).Infof("%s=%v", r.Err.Error(), r.ErrSrc) errors = append(errors, r.Err.Error()) } } - for _,u := range resp.UpdateResponse { + for _, u := range resp.UpdateResponse { if u.Err != nil { log.V(2).Infof("%s=%v", u.Err.Error(), u.ErrSrc) errors = append(errors, u.Err.Error()) @@ -375,9 +373,9 @@ func TranslProcessAction(uri string, payload []byte, ctx context.Context) ([]byt req.Payload = payload resp, err := translib.Action(req) - __log_audit_msg(ctx, "ACTION", uri, err) + __log_audit_msg(ctx, "ACTION", uri, err) - if err != nil{ + if err != nil { log.V(2).Infof("Action operation failed with error =%v, %v", resp.ErrSrc, err.Error()) return nil, err } @@ -389,21 +387,20 @@ func GetModels() []gnmipb.ModelData { gnmiModels := make([]gnmipb.ModelData, 0, 1) supportedModels, _ := translib.GetModels() - for _,model := range supportedModels { + for _, model := range supportedModels { gnmiModels = append(gnmiModels, gnmipb.ModelData{ - Name: model.Name, + Name: model.Name, Organization: model.Org, - Version: model.Ver, - + Version: model.Ver, }) } return gnmiModels } func isTranslibSuccess(err error) bool { - if err != nil && err.Error() != "Success" { - return false - } + if err != nil && err.Error() != "Success" { + return false + } - return true + return true }