blob: 22d536b6c19693fad571e27a0b7c9d3b606fb10d [file] [log] [blame]
/*
* Copyright 2020 AT&T Intellectual Property
* Copyright 2020 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.
*/
//
// Created by adi ENZEL on 2/16/20.
//
#include "HttpServer.h"
#include <algorithm>
#include <random>
#include "../sctpClient/sctpClient.h"
#include "../T1/E2Builder.h"
#include "../base64.h"
using namespace std;
using namespace Pistache;
#define RECEIVE_SCTP_BUFFER_SIZE 8192
namespace Generic {
void handleReady(const Rest::Request&, Http::ResponseWriter response) {
response.send(Http::Code::Ok, "1");
}
}
HttpServer::HttpServer(Address addr)
: httpBaseSocket(0), httpEndpoint(std::make_shared<Http::Endpoint>(addr)) { }
void HttpServer::init(size_t thr) {
if ((httpBaseSocket = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
fprintf(stderr, "Socket() error. %s\n", strerror(errno));
exit(-1);
}
auto optval = 1;
if (setsockopt(httpBaseSocket, SOL_SOCKET, SO_REUSEPORT, &optval, sizeof optval) != 0) {
fprintf(stderr, "setsockopt SO_REUSEPORT Error, %s %s, %d\n", strerror(errno), __func__, __LINE__);
close(httpBaseSocket);
exit(-1);
}
optval = 1;
if (setsockopt(httpBaseSocket, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof optval) != 0) {
fprintf(stderr, "setsockopt SO_REUSEADDR Error, %s %s, %d\n", strerror(errno), __func__, __LINE__);
close(httpBaseSocket);
exit(-1);
}
struct sockaddr_in address{};
address.sin_family = AF_INET;
if(inet_pton(AF_INET, "127.0.0.1", &address.sin_addr)<=0)
{
fprintf(stderr,"Invalid address/Address not supported. %s", strerror(errno));
exit(-1);
}
address.sin_port = htons(9098);
if (connect(httpBaseSocket, (SA *)(&address), sizeof(address)) < 0) {
fprintf(stderr, "connect() error. %s\n", strerror(errno));
exit(-1);
}
auto opts = Http::Endpoint::options().threads(thr);
httpEndpoint->init(opts);
setupRoutes();
}
void HttpServer::start() {
std::random_device device{};
std::mt19937 generator(device());
std::uniform_int_distribution<long> distribution(1, (long) 1e12);
transactionCounter = distribution(generator);
httpEndpoint->setHandler(router.handler());
httpEndpoint->serve();
}
void HttpServer::setupRoutes() {
using namespace Rest;
Routes::Get(router, "/setup/:ricaddress/:ricPort/:mcc/:mnc", Routes::bind(&HttpServer::sendSetupReq, this));
//Routes::Post(router, "/ricIndication/:ricid/:subscriptionId/:mcc/:mnc", Routes::bind(&HttpServer::sendSetupReq, this));
Routes::Get(router, "/ready", Routes::bind(&Generic::handleReady));
}
void HttpServer::sendSetupReq(const Rest::Request& request, Http::ResponseWriter response) {
auto mcc = request.param(":mcc").as<int>();
auto mnc = request.param(":mnc").as<int>();
auto ricAdress = request.param(":ricaddress").as<std::string>();
auto ricPort = request.param(":ricPort").as<int>();
//TODO build setup to send to address
E2AP_PDU_t pdu;
buildSetupRequest(&pdu,mcc, mnc);
// encode PDU to PER
auto buffer_size = RECEIVE_SCTP_BUFFER_SIZE;
unsigned char buffer[RECEIVE_SCTP_BUFFER_SIZE] = {};
// encode to xml
asn_enc_rval_t er;
er = asn_encode_to_buffer(0, ATS_ALIGNED_BASIC_PER, &asn_DEF_E2AP_PDU, &pdu, buffer, buffer_size);
if (er.encoded == -1) {
cerr << "encoding of : " << asn_DEF_E2AP_PDU.name << " failed, "<< strerror(errno) << endl;
response.send(Http::Code::Internal_Server_Error, "strerror(errno)");
return;
} else if (er.encoded > (ssize_t)buffer_size) {
cerr << "Buffer of size : " << buffer_size << " is to small for : " << asn_DEF_E2AP_PDU.name << endl;
response.send(Http::Code::Internal_Server_Error, "Buffer of size is too small");
return;
}
long len = er.encoded * 4 / 3 + 128;
auto *base64Buff = (unsigned char *)calloc(1,len + 1024);
char tx[32];
snprintf((char *) tx, sizeof tx, "%15ld", transactionCounter++);
auto sentLen = snprintf((char *)base64Buff, 1024, "%d|%s|%s|%d|", setupRequest_gnb, tx, ricAdress.c_str(), ricPort);
base64::encode(buffer, er.encoded, &base64Buff[sentLen], len);
sentLen += len;
len = send(httpBaseSocket, base64Buff, sentLen, 0);
if (len < 0) {
cerr << "failed sending setupRequest_gnb to Other thread. Error : " << strerror(errno) << endl;
response.send(Http::Code::Internal_Server_Error, "Failed send buffer");
free(base64Buff);
return;
}
char tx1[128];
snprintf((char *) tx1, sizeof tx1, "{\"id\": %s}", tx);
response.send(Http::Code::Ok, tx1);
free(base64Buff);
}