blob: a3136f01d469605f048d03dbce47169f8bb4eb0c [file] [log] [blame]
//
// 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 controllers
import (
"e2mgr/logger"
"e2mgr/models"
"e2mgr/providers/httpmsghandlerprovider"
"e2mgr/rNibWriter"
"e2mgr/services"
"e2mgr/sessions"
"e2mgr/utils"
"encoding/json"
"errors"
"gerrit.o-ran-sc.org/r/ric-plt/nodeb-rnib.git/common"
"gerrit.o-ran-sc.org/r/ric-plt/nodeb-rnib.git/reader"
"github.com/go-ozzo/ozzo-validation"
"github.com/go-ozzo/ozzo-validation/is"
"github.com/golang/protobuf/jsonpb"
"github.com/gorilla/mux"
"net/http"
"net/http/httputil"
"strings"
"sync"
"time"
)
const (
parseErrorCode int = 401
validationErrorCode int = 402
notFoundErrorCode int = 404
internalErrorCode int = 501
requiredInputErrorMessage = "Mandatory fields are missing"
validationFailedMessage = "Validation failed"
parseErrorMessage = "Parse failure"
notFoundErrorMessage = "Resource not found"
internalErrorMessage = "Internal Server Error. Please try again later"
sendMessageErrorMessage = "Failed to send message. For more information please check logs"
)
var E2Sessions = make(sessions.E2Sessions)
var messageChannel chan *models.E2RequestMessage
var errorChannel chan error
type INodebController interface {
HandleRequest(writer http.ResponseWriter, request *http.Request)
GetNodebIdList (writer http.ResponseWriter, request *http.Request)
GetNodeb(writer http.ResponseWriter, request *http.Request)
HandleHealthCheckRequest(writer http.ResponseWriter, request *http.Request)
}
type NodebController struct {
rmrService *services.RmrService
Logger *logger.Logger
rnibReaderProvider func() reader.RNibReader
rnibWriterProvider func() rNibWriter.RNibWriter
}
func NewNodebController(logger *logger.Logger, rmrService *services.RmrService, rnibReaderProvider func() reader.RNibReader,
rnibWriterProvider func() rNibWriter.RNibWriter) *NodebController {
messageChannel = make(chan *models.E2RequestMessage)
errorChannel = make(chan error)
return &NodebController{
rmrService: rmrService,
Logger: logger,
rnibReaderProvider: rnibReaderProvider,
rnibWriterProvider: rnibWriterProvider,
}
}
func prettifyRequest(request *http.Request) string {
dump, _ := httputil.DumpRequest(request, true)
requestPrettyPrint := strings.Replace(string(dump), "\r\n", " ", -1)
return strings.Replace(requestPrettyPrint, "\n", "", -1)
}
func (rc NodebController) HandleRequest(writer http.ResponseWriter, request *http.Request) {
startTime := time.Now()
rc.Logger.Infof("[Client -> E2 Manager] #nodeb_controller.HandleRequest - request: %v", prettifyRequest(request))
vars := mux.Vars(request)
messageTypeParam := vars["messageType"]
requestHandlerProvider := httpmsghandlerprovider.NewRequestHandlerProvider(rc.rnibWriterProvider)
handler, err := requestHandlerProvider.GetHandler(rc.Logger, messageTypeParam)
if err != nil {
handleErrorResponse(rc.Logger, http.StatusNotFound, notFoundErrorCode, notFoundErrorMessage, writer, startTime)
return
}
requestDetails, err := parseJson(rc.Logger, request)
if err != nil {
handleErrorResponse(rc.Logger, http.StatusBadRequest, parseErrorCode, parseErrorMessage, writer, startTime)
return
}
rc.Logger.Infof("#nodeb_controller.HandleRequest - request: %+v", requestDetails)
if err := validateRequestDetails(rc.Logger, requestDetails); err != nil {
handleErrorResponse(rc.Logger, http.StatusBadRequest, validationErrorCode, requiredInputErrorMessage, writer, startTime)
return
}
err = handler.PreHandle(rc.Logger, &requestDetails)
if err != nil {
handleErrorResponse(rc.Logger, http.StatusInternalServerError, internalErrorCode, err.Error(), writer, startTime)
return
}
rc.Logger.Infof("[E2 Manager -> Client] #nodeb_controller.HandleRequest - http status: 200")
writer.WriteHeader(http.StatusOK)
var wg sync.WaitGroup
go handler.CreateMessage(rc.Logger, &requestDetails, messageChannel, E2Sessions, startTime, wg)
go rc.rmrService.SendMessage(handler.GetMessageType(), messageChannel, errorChannel, wg)
wg.Wait()
err = <-errorChannel
if err != nil {
handleErrorResponse(rc.Logger, http.StatusInternalServerError, internalErrorCode, sendMessageErrorMessage, writer, startTime)
return
}
printHandlingRequestElapsedTimeInMs(rc.Logger, startTime)
}
func (rc NodebController) GetNodebIdList (writer http.ResponseWriter, request *http.Request) {
startTime := time.Now()
rnibReaderService := services.NewRnibReaderService(rc.rnibReaderProvider)
nodebIdList, rnibError := rnibReaderService.GetNodebIdList()
if rnibError != nil {
rc.Logger.Errorf("%v", rnibError);
httpStatusCode, errorCode, errorMessage := rnibErrorToHttpError(rnibError)
handleErrorResponse(rc.Logger, httpStatusCode, errorCode, errorMessage, writer, startTime)
return;
}
pmList := utils.ConvertNodebIdListToProtoMessageList(*nodebIdList)
result, err := utils.MarshalProtoMessageListToJsonArray(pmList)
if err != nil {
rc.Logger.Errorf("%v", err);
handleErrorResponse(rc.Logger, http.StatusInternalServerError, internalErrorCode, internalErrorMessage, writer, startTime)
return;
}
writer.Header().Set("Content-Type", "application/json")
rc.Logger.Infof("[E2 Manager -> Client] #nodeb_controller.GetNodebIdList - response: %s", result)
writer.Write([]byte(result))
}
func (rc NodebController) GetNodeb(writer http.ResponseWriter, request *http.Request) {
startTime := time.Now()
vars := mux.Vars(request)
ranName := vars["ranName"]
// WAS: respondingNode, rnibError := reader.GetRNibReader().GetNodeb(ranName)
rnibReaderService := services.NewRnibReaderService(rc.rnibReaderProvider);
respondingNode, rnibError := rnibReaderService.GetNodeb(ranName)
if rnibError != nil {
rc.Logger.Errorf("%v", rnibError)
httpStatusCode, errorCode, errorMessage := rnibErrorToHttpError(rnibError)
handleErrorResponse(rc.Logger, httpStatusCode, errorCode, errorMessage, writer, startTime)
return
}
m := jsonpb.Marshaler{}
result, err := m.MarshalToString(respondingNode)
if err != nil {
rc.Logger.Errorf("%v", err)
handleErrorResponse(rc.Logger, http.StatusInternalServerError, internalErrorCode, internalErrorMessage, writer, startTime)
return
}
writer.Header().Set("Content-Type", "application/json")
rc.Logger.Infof("[E2 Manager -> Client] #nodeb_controller.GetNodeb - response: %s", result)
writer.Write([]byte(result))
}
func (rc NodebController) HandleHealthCheckRequest(writer http.ResponseWriter, request *http.Request) {
//fmt.Println("[X-APP -> Client] #HandleHealthCheckRequest - http status: 200")
writer.WriteHeader(http.StatusOK)
}
func parseJson(logger *logger.Logger, request *http.Request) (models.RequestDetails, error) {
var requestDetails models.RequestDetails
if err := json.NewDecoder(request.Body).Decode(&requestDetails); err != nil {
logger.Errorf("#nodeb_controller.parseJson - cannot deserialize incoming request. request: %v, error: %v", request, err)
return requestDetails, err
}
return requestDetails, nil
}
func validateRequestDetails(logger *logger.Logger, requestDetails models.RequestDetails) error {
if requestDetails.RanPort == 0 {
logger.Errorf("#nodeb_controller.validateRequestDetails - validation failure: port cannot be zero")
return errors.New("port: cannot be blank")
}
err := validation.ValidateStruct(&requestDetails,
validation.Field(&requestDetails.RanIp, validation.Required, is.IP),
validation.Field(&requestDetails.RanName, validation.Required),
)
if err != nil {
logger.Errorf("#nodeb_controller.validateRequestDetails - validation failure, error: %v", err)
}
return err
}
func handleErrorResponse(logger *logger.Logger, httpStatus int, errorCode int, errorMessage string, writer http.ResponseWriter, startTime time.Time) {
errorResponseDetails := models.ErrorResponse{errorCode, errorMessage}
errorResponse, _ := json.Marshal(errorResponseDetails)
printHandlingRequestElapsedTimeInMs(logger, startTime)
logger.Infof("[E2 Manager -> Client] #nodeb_controller.handleErrorResponse - http status: %d, error response: %+v", httpStatus, errorResponseDetails)
writer.Header().Set("Content-Type", "application/json")
writer.WriteHeader(httpStatus)
_, err := writer.Write(errorResponse)
if err != nil {
logger.Errorf("#nodeb_controller.handleErrorResponse - Cannot send response. writer:%v", writer)
}
}
func printHandlingRequestElapsedTimeInMs(logger *logger.Logger, startTime time.Time) {
logger.Infof("Summary: #nodeb_controller.printElapsedTimeInMs - Elapsed time for handling request from client to E2 termination: %f ms",
float64(time.Since(startTime))/float64(time.Millisecond))
}
func rnibErrorToHttpError(rnibError common.IRNibError) (int, int, string) {
switch rnibError.GetCode() {
case common.RESOURCE_NOT_FOUND:
return http.StatusNotFound, notFoundErrorCode, notFoundErrorMessage
case common.INTERNAL_ERROR:
return http.StatusInternalServerError, internalErrorCode, internalErrorMessage
case common.VALIDATION_ERROR:
return http.StatusBadRequest, validationErrorCode, validationFailedMessage
default:
return http.StatusInternalServerError, internalErrorCode, internalErrorMessage
}
}