// // Copyright 2019 AT&T Intellectual Property // Copyright 2019 Nokia // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // package writer import ( "errors" "fmt" rnibcommon "gerrit.o-ran-sc.org/r/ric-plt/nodeb-rnib.git/common" rnibentities "gerrit.o-ran-sc.org/r/ric-plt/nodeb-rnib.git/entities" "gerrit.o-ran-sc.org/r/ric-plt/sdlgo" "github.com/golang/protobuf/proto" ) var writerPool *rnibcommon.Pool type rNibWriterInstance struct { sdl *ISdlInstance namespace string } /* RNibWriter interface allows saving data to the redis BD */ type RNibWriter interface { SaveNodeb(nbIdentity *rnibentities.NbIdentity, nb *rnibentities.NodebInfo) rnibcommon.IRNibError } /* Init initializes the infrastructure required for the RNibWriter instance */ func InitWriter(namespace string, poolSize int) { initWriterPool(poolSize, func() interface{} { var sdlI ISdlInstance = sdlgo.NewSdlInstance(namespace, sdlgo.NewDatabase()) return &rNibWriterInstance{sdl: &sdlI, namespace: namespace} }, func(obj interface{}) { (*obj.(*rNibWriterInstance).sdl).Close() }) } /* InitPool initializes the writer's instances pool */ func initWriterPool(poolSize int, newObj func() interface{}, destroyObj func(interface{})) { writerPool = rnibcommon.NewPool(poolSize, newObj, destroyObj) } /* GetRNibWriter returns RNibWriter instance from the pool */ func GetRNibWriter() RNibWriter { return writerPool.Get().(RNibWriter) } /* SaveNodeb saves nodeB entity data in the redis DB according to the specified data model */ func (w *rNibWriterInstance) SaveNodeb(nbIdentity *rnibentities.NbIdentity, entity *rnibentities.NodebInfo) rnibcommon.IRNibError { isNotEmptyIdentity := isNotEmpty(nbIdentity) if isNotEmptyIdentity && entity.GetNodeType() == rnibentities.Node_UNKNOWN { return rnibcommon.NewValidationError(errors.New(fmt.Sprintf("#rNibWriter.saveNodeB - Unknown responding node type, entity: %v", entity))) } defer writerPool.Put(w) data, err := proto.Marshal(entity) if err != nil { return rnibcommon.NewInternalError(err) } var pairs []interface{} key, rNibErr := rnibcommon.ValidateAndBuildNodeBNameKey(nbIdentity.InventoryName) if rNibErr != nil { return rNibErr } pairs = append(pairs, key, data) if isNotEmptyIdentity { key, rNibErr = rnibcommon.ValidateAndBuildNodeBIdKey(entity.GetNodeType().String(), nbIdentity.GlobalNbId.GetPlmnId(), nbIdentity.GlobalNbId.GetNbId()) if rNibErr != nil { return rNibErr } pairs = append(pairs, key, data) } if entity.GetEnb() != nil { pairs, rNibErr = appendEnbCells(nbIdentity, entity.GetEnb().GetServedCells(), pairs) if rNibErr != nil { return rNibErr } } if entity.GetGnb() != nil { pairs, rNibErr = appendGnbCells(nbIdentity, entity.GetGnb().GetServedNrCells(), pairs) if rNibErr != nil { return rNibErr } } err = (*w.sdl).Set(pairs) if err != nil { return rnibcommon.NewInternalError(err) } if isNotEmptyIdentity { nbIdData, err := proto.Marshal(nbIdentity) if err != nil { return rnibcommon.NewInternalError(err) } err = (*w.sdl).AddMember(entity.GetNodeType().String(), nbIdData) if err != nil { return rnibcommon.NewInternalError(err) } } return nil } /* Close closes writer's pool */ func CloseWriter() { writerPool.Close() } func appendEnbCells(nbIdentity *rnibentities.NbIdentity, cells []*rnibentities.ServedCellInfo, pairs []interface{}) ([]interface{}, rnibcommon.IRNibError) { for _, cell := range cells { cellEntity := rnibentities.Cell{Type: rnibentities.Cell_LTE_CELL, Cell: &rnibentities.Cell_ServedCellInfo{ServedCellInfo: cell}} cellData, err := proto.Marshal(&cellEntity) if err != nil { return pairs, rnibcommon.NewInternalError(err) } key, rNibErr := rnibcommon.ValidateAndBuildCellIdKey(cell.GetCellId()) if rNibErr != nil { return pairs, rNibErr } pairs = append(pairs, key, cellData) key, rNibErr = rnibcommon.ValidateAndBuildCellNamePciKey(nbIdentity.InventoryName, cell.GetPci()) if rNibErr != nil { return pairs, rNibErr } pairs = append(pairs, key, cellData) } return pairs, nil } func appendGnbCells(nbIdentity *rnibentities.NbIdentity, cells []*rnibentities.ServedNRCell, pairs []interface{}) ([]interface{}, rnibcommon.IRNibError) { for _, cell := range cells { cellEntity := rnibentities.Cell{Type: rnibentities.Cell_NR_CELL, Cell: &rnibentities.Cell_ServedNrCell{ServedNrCell: cell}} cellData, err := proto.Marshal(&cellEntity) if err != nil { return pairs, rnibcommon.NewInternalError(err) } key, rNibErr := rnibcommon.ValidateAndBuildNrCellIdKey(cell.GetServedNrCellInformation().GetCellId()) if rNibErr != nil { return pairs, rNibErr } pairs = append(pairs, key, cellData) key, rNibErr = rnibcommon.ValidateAndBuildCellNamePciKey(nbIdentity.InventoryName, cell.GetServedNrCellInformation().GetNrPci()) if rNibErr != nil { return pairs, rNibErr } pairs = append(pairs, key, cellData) } return pairs, nil } func isNotEmpty(nbIdentity *rnibentities.NbIdentity) bool { return nbIdentity.GlobalNbId != nil && nbIdentity.GlobalNbId.PlmnId != "" && nbIdentity.GlobalNbId.NbId != "" }