open5gs/src/pcrf/pcrf_context.c

569 lines
24 KiB
C
Raw Normal View History

2017-08-17 05:15:08 +00:00
#define TRACE_MODULE _pcrf_context
#include "core_debug.h"
2017-08-25 10:40:49 +00:00
#include "core_pkbuf.h"
2017-08-17 05:15:08 +00:00
#include "core_lib.h"
#include <mongoc.h>
2017-11-29 00:51:34 +00:00
#include <yaml.h>
2018-01-04 11:38:22 +00:00
#include "app/yaml_helper.h"
2017-08-17 05:15:08 +00:00
2018-01-04 11:38:22 +00:00
#include "fd/fd_lib.h"
2017-08-17 05:15:08 +00:00
2018-01-04 11:38:22 +00:00
#include "app/context.h"
2017-08-17 05:15:08 +00:00
#include "pcrf_context.h"
static pcrf_context_t self;
static fd_config_t g_fd_conf;
2017-08-17 05:15:08 +00:00
static int context_initialized = 0;
pcrf_context_t* pcrf_self()
{
return &self;
}
status_t pcrf_context_init(void)
{
d_assert(context_initialized == 0, return CORE_ERROR,
"PCRF context already has been context_initialized");
/* Initial FreeDiameter Config */
memset(&g_fd_conf, 0, sizeof(fd_config_t));
2017-08-17 05:15:08 +00:00
/* Initialize PCRF context */
memset(&self, 0, sizeof(pcrf_context_t));
self.fd_config = &g_fd_conf;
2017-08-17 05:15:08 +00:00
if (mutex_create(&self.db_lock, MUTEX_DEFAULT) != CORE_OK)
{
d_error("Mutex creation failed");
return CORE_ERROR;
}
context_initialized = 1;
return CORE_OK;
}
status_t pcrf_context_final(void)
{
d_assert(context_initialized == 1, return CORE_ERROR,
"PCRF context already has been finalized");
mutex_delete(self.db_lock);
context_initialized = 0;
return CORE_OK;
}
static status_t pcrf_context_prepare()
{
return CORE_OK;
}
static status_t pcrf_context_validation()
{
if (self.fd_conf_path == NULL)
{
2017-11-29 05:44:14 +00:00
d_error("No pcrf.freeDiameter in '%s'",
2017-08-17 05:15:08 +00:00
context_self()->config.path);
return CORE_ERROR;
}
return CORE_OK;
}
status_t pcrf_context_parse_config()
{
status_t rv;
config_t *config = &context_self()->config;
2017-11-29 00:51:34 +00:00
yaml_document_t *document = NULL;
yaml_iter_t root_iter;
2017-08-17 05:15:08 +00:00
2017-11-29 00:51:34 +00:00
d_assert(config, return CORE_ERROR,);
document = config->document;
d_assert(document, return CORE_ERROR,);
2017-08-17 05:15:08 +00:00
rv = pcrf_context_prepare();
if (rv != CORE_OK) return rv;
2017-11-29 00:51:34 +00:00
yaml_iter_init(&root_iter, document);
while(yaml_iter_next(&root_iter))
2017-08-17 05:15:08 +00:00
{
2017-11-29 00:51:34 +00:00
const char *root_key = yaml_iter_key(&root_iter);
d_assert(root_key, return CORE_ERROR,);
if (!strcmp(root_key, "pcrf"))
2017-08-17 05:15:08 +00:00
{
2017-11-29 00:51:34 +00:00
yaml_iter_t pcrf_iter;
yaml_iter_recurse(&root_iter, &pcrf_iter);
while(yaml_iter_next(&pcrf_iter))
2017-08-17 05:15:08 +00:00
{
2017-11-29 00:51:34 +00:00
const char *pcrf_key = yaml_iter_key(&pcrf_iter);
d_assert(pcrf_key, return CORE_ERROR,);
if (!strcmp(pcrf_key, "freeDiameter"))
2017-08-17 05:15:08 +00:00
{
2017-11-29 00:51:34 +00:00
self.fd_conf_path = yaml_iter_value(&pcrf_iter);
2017-08-17 05:15:08 +00:00
}
2017-11-29 00:51:34 +00:00
else
d_warn("unknown key `%s`", pcrf_key);
2017-08-17 05:15:08 +00:00
}
}
}
rv = pcrf_context_validation();
if (rv != CORE_OK) return rv;
return CORE_OK;
}
status_t pcrf_context_setup_trace_module()
{
2017-11-29 05:44:14 +00:00
int diameter = context_self()->logger.trace.diameter;
2017-11-27 06:45:33 +00:00
int others = context_self()->logger.trace.others;
2017-08-17 05:15:08 +00:00
2017-11-29 05:44:14 +00:00
if (diameter)
2017-08-17 05:15:08 +00:00
{
2017-11-29 05:44:14 +00:00
if (diameter <= 1) fd_g_debug_lvl = FD_LOG_ERROR;
else if (diameter <= 3) fd_g_debug_lvl = FD_LOG_NOTICE;
else if (diameter <= 5) fd_g_debug_lvl = FD_LOG_DEBUG;
2017-08-17 05:15:08 +00:00
else fd_g_debug_lvl = FD_LOG_ANNOYING;
2017-08-22 15:51:57 +00:00
extern int _pcrf_fd_path;
2017-11-29 05:44:14 +00:00
d_trace_level(&_pcrf_fd_path, diameter);
2017-08-17 05:15:08 +00:00
extern int _fd_init;
2017-11-29 05:44:14 +00:00
d_trace_level(&_fd_init, diameter);
2017-08-17 05:15:08 +00:00
extern int _fd_logger;
2017-11-29 05:44:14 +00:00
d_trace_level(&_fd_logger, diameter);
2017-08-17 05:15:08 +00:00
}
if (others)
{
extern int _mutex;
d_trace_level(&_mutex, others);
extern int _pkbuf;
d_trace_level(&_pkbuf, others);
extern int _context;
d_trace_level(&_context, others);
extern int _pcrf_context;
d_trace_level(&_pcrf_context, others);
}
return CORE_OK;
}
status_t pcrf_db_init()
{
if (context_self()->db_client && context_self()->db_name)
{
self.subscriberCollection = mongoc_client_get_collection(
context_self()->db_client,
context_self()->db_name, "subscribers");
d_assert(self.subscriberCollection, return CORE_ERROR,
"Couldn't find Subscriber Collection in '%s'",
2017-08-24 08:05:10 +00:00
context_self()->db_name);
2017-08-17 05:15:08 +00:00
}
return CORE_OK;
}
status_t pcrf_db_final()
{
if (self.subscriberCollection)
{
mongoc_collection_destroy(self.subscriberCollection);
}
return CORE_OK;
}
2017-08-24 08:05:10 +00:00
status_t pcrf_db_pdn_data(
2017-08-25 10:40:49 +00:00
c_int8_t *imsi_bcd, c_int8_t *apn, gx_cca_message_t *cca_message)
2017-08-24 08:05:10 +00:00
{
status_t rv = CORE_OK;
mongoc_cursor_t *cursor = NULL;
bson_t *query = NULL;
bson_t *opts = NULL;
bson_error_t error;
const bson_t *document;
bson_iter_t iter;
bson_iter_t child1_iter, child2_iter, child3_iter;
bson_iter_t child4_iter, child5_iter, child6_iter;
const char *utf8 = NULL;
c_uint32_t length = 0;
d_assert(imsi_bcd, return CORE_ERROR, "Null param");
d_assert(apn, return CORE_ERROR, "Null param");
2017-08-25 10:40:49 +00:00
d_assert(cca_message, return CORE_ERROR, "Null param");
2017-08-24 08:05:10 +00:00
mutex_lock(self.db_lock);
query = BCON_NEW(
"imsi", BCON_UTF8(imsi_bcd),
"pdn.apn", BCON_UTF8(apn));
#if MONGOC_MAJOR_VERSION >= 1 && MONGOC_MINOR_VERSION >= 5
2017-08-24 08:05:10 +00:00
opts = BCON_NEW(
"projection", "{",
"imsi", BCON_INT64(1),
"pdn.$", BCON_INT64(1),
"}"
);
cursor = mongoc_collection_find_with_opts(
self.subscriberCollection, query, opts, NULL);
#else
opts = BCON_NEW(
"imsi", BCON_INT64(1),
"pdn.$", BCON_INT64(1)
);
cursor = mongoc_collection_find(self.subscriberCollection,
MONGOC_QUERY_NONE, 0, 0, 0, query, opts, NULL);
#endif
2017-08-24 08:05:10 +00:00
if (!mongoc_cursor_next(cursor, &document))
{
d_error("Cannot find IMSI(%s)+APN(%s) in DB", imsi_bcd, apn);
rv = CORE_ERROR;
goto out;
}
if (mongoc_cursor_error(cursor, &error))
{
d_error("Cursor Failure: %s", error.message);
rv = CORE_ERROR;
goto out;
}
if (!bson_iter_init(&iter, document))
{
d_error("bson_iter_init failed in this document");
rv = CORE_ERROR;
goto out;
}
while(bson_iter_next(&iter))
{
const char *key = bson_iter_key(&iter);
if (!strcmp(key, "pdn") &&
BSON_ITER_HOLDS_ARRAY(&iter))
{
int pdn_index = 0;
bson_iter_recurse(&iter, &child1_iter);
while(bson_iter_next(&child1_iter))
{
const char *child1_key = bson_iter_key(&child1_iter);
pdn_t *pdn = NULL;
d_assert(child1_key, goto out, "PDN is not ARRAY");
pdn_index = atoi(child1_key);
d_assert(pdn_index == 0, goto out,
"Invalid PDN Index(%d)", pdn_index);
2017-08-25 10:40:49 +00:00
pdn = &cca_message->pdn;
2017-08-24 08:05:10 +00:00
bson_iter_recurse(&child1_iter, &child2_iter);
while(bson_iter_next(&child2_iter))
{
const char *child2_key = bson_iter_key(&child2_iter);
if (!strcmp(child2_key, "apn") &&
BSON_ITER_HOLDS_UTF8(&child2_iter))
{
utf8 = bson_iter_utf8(&child2_iter, &length);
core_cpystrn(pdn->apn, utf8,
c_min(length, MAX_APN_LEN)+1);
}
else if (!strcmp(child2_key, "type") &&
BSON_ITER_HOLDS_INT32(&child2_iter))
{
pdn->pdn_type = bson_iter_int32(&child2_iter);
}
else if (!strcmp(child2_key, "qos") &&
BSON_ITER_HOLDS_DOCUMENT(&child2_iter))
{
bson_iter_recurse(&child2_iter, &child3_iter);
while(bson_iter_next(&child3_iter))
{
const char *child3_key =
bson_iter_key(&child3_iter);
if (!strcmp(child3_key, "qci") &&
BSON_ITER_HOLDS_INT32(&child3_iter))
{
pdn->qos.qci = bson_iter_int32(&child3_iter);
}
else if (!strcmp(child3_key, "arp") &&
BSON_ITER_HOLDS_DOCUMENT(&child3_iter))
{
bson_iter_recurse(&child3_iter, &child4_iter);
while(bson_iter_next(&child4_iter))
{
const char *child4_key =
bson_iter_key(&child4_iter);
if (!strcmp(child4_key, "priority_level") &&
BSON_ITER_HOLDS_INT32(&child4_iter))
{
pdn->qos.arp.priority_level =
bson_iter_int32(&child4_iter);
}
else if (!strcmp(child4_key,
"pre_emption_capability") &&
BSON_ITER_HOLDS_INT32(&child4_iter))
{
pdn->qos.arp.pre_emption_capability =
bson_iter_int32(&child4_iter);
}
else if (!strcmp(child4_key,
"pre_emption_vulnerability") &&
BSON_ITER_HOLDS_INT32(&child4_iter))
{
pdn->qos.arp.pre_emption_vulnerability =
bson_iter_int32(&child4_iter);
}
}
}
}
}
else if (!strcmp(child2_key, "ambr") &&
BSON_ITER_HOLDS_DOCUMENT(&child2_iter))
{
bson_iter_recurse(&child2_iter, &child3_iter);
while(bson_iter_next(&child3_iter))
{
const char *child3_key =
bson_iter_key(&child3_iter);
if (!strcmp(child3_key, "uplink") &&
BSON_ITER_HOLDS_INT64(&child3_iter))
2017-08-24 08:05:10 +00:00
{
pdn->ambr.uplink =
bson_iter_int64(&child3_iter) * 1024;
2017-08-24 08:05:10 +00:00
}
else if (!strcmp(child3_key, "downlink") &&
BSON_ITER_HOLDS_INT64(&child3_iter))
2017-08-24 08:05:10 +00:00
{
pdn->ambr.downlink =
bson_iter_int64(&child3_iter) * 1024;
2017-08-24 08:05:10 +00:00
}
}
}
else if (!strcmp(child2_key, "pcc_rule") &&
BSON_ITER_HOLDS_ARRAY(&child2_iter))
{
int pcc_rule_index = 0;
bson_iter_recurse(&child2_iter, &child3_iter);
while(bson_iter_next(&child3_iter))
{
const char *child3_key =
bson_iter_key(&child3_iter);
pcc_rule_t *pcc_rule = NULL;
d_assert(child3_key, goto out,
"PCC RULE is not ARRAY");
pcc_rule_index = atoi(child3_key);
d_assert(pcc_rule_index < MAX_NUM_OF_PCC_RULE,
goto out,
"Overflow of PCC RULE number(%d>%d)",
pcc_rule_index, MAX_NUM_OF_PCC_RULE);
2017-08-25 10:40:49 +00:00
pcc_rule = &cca_message->pcc_rule[pcc_rule_index];
2017-08-24 08:05:10 +00:00
bson_iter_recurse(&child3_iter, &child4_iter);
while(bson_iter_next(&child4_iter))
{
const char *child4_key =
bson_iter_key(&child4_iter);
if (!strcmp(child4_key, "qos") &&
BSON_ITER_HOLDS_DOCUMENT(&child4_iter))
{
bson_iter_recurse(
&child4_iter, &child5_iter);
while(bson_iter_next(&child5_iter))
{
const char *child5_key =
bson_iter_key(&child5_iter);
if (!strcmp(child5_key, "qci") &&
BSON_ITER_HOLDS_INT32(&child5_iter))
{
pcc_rule->qos.qci =
bson_iter_int32(&child5_iter);
}
else if (!strcmp(child5_key, "arp") &&
BSON_ITER_HOLDS_DOCUMENT(
&child5_iter))
{
bson_iter_recurse(
&child5_iter, &child6_iter);
while(bson_iter_next(&child6_iter))
{
const char *child6_key =
bson_iter_key(&child6_iter);
if (!strcmp(child6_key,
"priority_level") &&
BSON_ITER_HOLDS_INT32(
&child6_iter))
{
pcc_rule->qos.arp.
priority_level =
bson_iter_int32(
&child6_iter);
}
else if (!strcmp(child6_key,
"pre_emption_capability") &&
BSON_ITER_HOLDS_INT32(
&child6_iter))
{
pcc_rule->qos.arp.
pre_emption_capability =
bson_iter_int32(
&child6_iter);
}
else if (!strcmp(child6_key,
"pre_emption_vulnerability") &&
BSON_ITER_HOLDS_INT32(&child6_iter))
{
pcc_rule->qos.arp.
pre_emption_vulnerability =
bson_iter_int32(
&child6_iter);
}
}
}
else if (!strcmp(child5_key, "mbr") &&
BSON_ITER_HOLDS_DOCUMENT(
&child5_iter))
{
bson_iter_recurse(
&child5_iter, &child6_iter);
while(bson_iter_next(&child6_iter))
{
const char *child6_key =
bson_iter_key(&child6_iter);
if (!strcmp(child6_key,
"downlink") &&
BSON_ITER_HOLDS_INT64(
2017-08-24 08:05:10 +00:00
&child6_iter))
{
pcc_rule->qos.mbr.downlink =
bson_iter_int64(
2017-08-24 14:10:36 +00:00
&child6_iter) * 1024;
2017-08-24 08:05:10 +00:00
}
else if (!strcmp(child6_key,
"uplink") &&
BSON_ITER_HOLDS_INT64(
2017-08-24 08:05:10 +00:00
&child6_iter))
{
pcc_rule->qos.mbr.uplink =
bson_iter_int64(
2017-08-24 14:10:36 +00:00
&child6_iter) * 1024;
2017-08-24 08:05:10 +00:00
}
}
}
else if (!strcmp(child5_key, "gbr") &&
BSON_ITER_HOLDS_DOCUMENT(
&child5_iter))
{
bson_iter_recurse(&child5_iter,
&child6_iter);
while(bson_iter_next(&child6_iter))
{
const char *child6_key =
bson_iter_key(&child6_iter);
if (!strcmp(child6_key,
"downlink") &&
BSON_ITER_HOLDS_INT64(
2017-08-24 08:05:10 +00:00
&child6_iter))
{
pcc_rule->qos.gbr.downlink =
bson_iter_int64(
2017-08-24 14:10:36 +00:00
&child6_iter) * 1024;
2017-08-24 08:05:10 +00:00
}
else if (!strcmp(child6_key,
"uplink") &&
BSON_ITER_HOLDS_INT64(
2017-08-24 08:05:10 +00:00
&child6_iter))
{
pcc_rule->qos.gbr.uplink =
bson_iter_int64(
2017-08-24 14:10:36 +00:00
&child6_iter) * 1024;
2017-08-24 08:05:10 +00:00
}
}
}
}
}
else if (!strcmp(child4_key, "flow") &&
BSON_ITER_HOLDS_ARRAY(&child4_iter))
{
int flow_index = 0;
bson_iter_recurse(&child4_iter,
&child5_iter);
while(bson_iter_next(&child5_iter))
{
const char *child5_key =
bson_iter_key(&child5_iter);
flow_t *flow = NULL;
d_assert(child5_key, goto out,
"FLOW is not ARRAY");
flow_index = atoi(child5_key);
d_assert(flow_index < MAX_NUM_OF_FLOW,
goto out,
"Overflow of FLOW number(%d>%d)",
flow_index, MAX_NUM_OF_FLOW);
flow = &pcc_rule->flow[flow_index];
bson_iter_recurse(
&child5_iter, &child6_iter);
while(bson_iter_next(&child6_iter))
{
const char *child6_key =
bson_iter_key(&child6_iter);
if (!strcmp(child6_key, "direction") &&
BSON_ITER_HOLDS_INT32(
&child6_iter))
{
flow->direction =
bson_iter_int32(
&child6_iter);
}
else if (!strcmp(child6_key,
"description") &&
BSON_ITER_HOLDS_UTF8(
&child6_iter))
{
utf8 = bson_iter_utf8(
&child6_iter, &length);
2017-08-25 10:40:49 +00:00
flow->description =
core_malloc(length+1);
2017-08-24 08:05:10 +00:00
core_cpystrn(
2017-08-25 10:40:49 +00:00
(char*)flow->description,
utf8, length+1);
2017-08-24 08:05:10 +00:00
}
}
flow_index++;
}
pcc_rule->num_of_flow = flow_index;
}
}
pcc_rule_index++;
}
2017-08-25 10:40:49 +00:00
cca_message->num_of_pcc_rule = pcc_rule_index;
2017-08-24 08:05:10 +00:00
}
}
}
}
}
out:
if (query) bson_destroy(query);
2017-08-25 10:40:49 +00:00
if (opts) bson_destroy(opts);
2017-08-24 08:05:10 +00:00
if (cursor) mongoc_cursor_destroy(cursor);
mutex_unlock(self.db_lock);
return rv;
}