view extensions/dict_json/dict_json.cc @ 1333:cbe1da7a32f1

Add dict_json extension. This extension allows loading diameter dictionares from a JSON file. Example files are provided by contrib/dict_json.
author Thomas Klausner <tk@giga.or.at>
date Tue, 09 Apr 2019 15:11:32 +0200
parents
children
line wrap: on
line source

/**********************************************************************************************************
 * Software License Agreement (BSD License)                                                               *
 * Author: Thomas Klausner <tk@giga.or.at>                                                                *
 *                                                                                                        *
 * Copyright (c) 2016, 2017, 2019 Thomas Klausner                                                         *
 * All rights reserved.                                                                                   *
 *                                                                                                        *
 * Written under contract by nfotex IT GmbH, http://nfotex.com/                                           *
 *                                                                                                        *
 * Redistribution and use of this software in source and binary forms, with or without modification, are  *
 * permitted provided that the following conditions are met:                                              *
 *                                                                                                        *
 * * Redistributions of source code must retain the above                                                 *
 *   copyright notice, this list of conditions and the                                                    *
 *   following disclaimer.                                                                                *
 *                                                                                                        *
 * * Redistributions in binary form must reproduce the above                                              *
 *   copyright notice, this list of conditions and the                                                    *
 *   following disclaimer in the documentation and/or other                                               *
 *   materials provided with the distribution.                                                            *
 *                                                                                                        *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED *
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A *
 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR *
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT     *
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS    *
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR *
 * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF   *
 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.                                                             *
 **********************************************************************************************************/

/*
 * Extend a dictionary using data from a JSON file.
 *
 */
#include <freeDiameter/extension.h>
#include <json/json.h>
#include <json/SchemaValidator.h>
#include <sys/stat.h>

extern const char *dict_json_dict_schema;

static int
make_type(const char *typestring, struct dict_object **type)
{
        *type = NULL;

        if (strcmp("Unsigned32", typestring) == 0)
                return AVP_TYPE_UNSIGNED32;
        else if (strcmp("Enumerated", typestring) == 0 || strcmp("Integer32", typestring) == 0)
                return AVP_TYPE_INTEGER32;
        else if (strcmp("OctetString", typestring) == 0)
                return AVP_TYPE_OCTETSTRING;
        else if (strcmp("Grouped", typestring) == 0)
                return AVP_TYPE_GROUPED;
        else if (strcmp("Integer64", typestring) == 0)
                return AVP_TYPE_INTEGER64;
        else if (strcmp("Unsigned64", typestring) == 0)
                return AVP_TYPE_UNSIGNED64;
        else if (strcmp("Float32", typestring) == 0)
                return AVP_TYPE_FLOAT32;
        else if (strcmp("Float64", typestring) == 0)
                return AVP_TYPE_FLOAT64;
        else {
                if (fd_dict_search(fd_g_config->cnf_dict, DICT_TYPE, TYPE_BY_NAME, typestring, type, ENOENT) != 0) {
                        LOG_E("Unknown type '%s'", typestring);
                        return -1;
                }
                return AVP_TYPE_OCTETSTRING;
        }

        return -1;
}


static uint8_t
make_avp_flags(const char *flagstring)
{
        uint8_t flags = 0;

        if (strchr(flagstring, 'M') != NULL)
                flags |= AVP_FLAG_MANDATORY;
        if (strchr(flagstring, 'V') != NULL)
                flags |= AVP_FLAG_VENDOR;

        return flags;
}

static uint8_t
make_command_flags(const char *flagstring)
{
        uint8_t flags = 0;

        if (strchr(flagstring, 'E') != NULL)
                flags |= CMD_FLAG_ERROR;
        if (strchr(flagstring, 'P') != NULL)
                flags |= CMD_FLAG_PROXIABLE;
        if (strchr(flagstring, 'R') != NULL)
                flags |= CMD_FLAG_REQUEST;

        return flags;
}


static bool
add_applications(const Json::Value &config)
{
        Json::Value applications = config["Applications"];
        if (applications == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<applications.size(); i++) {
                int ret;
                struct dict_application_data application_data;
                application_data.application_id = applications[i]["Code"].asUInt();
                application_data.application_name = (char *)(void *)applications[i]["Name"].asCString();
                if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_APPLICATION, &application_data, NULL, NULL)) != 0) {
                        LOG_E("error adding Application '%s' to dictionary: %s", applications[i]["Name"].asCString(), strerror(ret));
                        return false;
                }
                LOG_D("Added Application '%s' to dictionary", applications[i]["Name"].asCString());
        }

        return true;
}

static bool
add_enumtype(const char *enumtypename, enum dict_avp_basetype basetype, struct dict_object **enumtype)
{
        struct dict_type_data data;
        int ret;

        memset(&data, 0, sizeof(data));
        data.type_base = basetype;
        data.type_name = (char *)(void *)enumtypename;
        data.type_interpret = NULL;
        data.type_encode = NULL;
        data.type_dump = NULL;
        if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_TYPE, &data, NULL, enumtype)) != 0) {
                /* TODO: allow ret == EEXIST? */
                LOG_E("error defining type '%s': %s", enumtypename, strerror(ret));
                return false;
        }
        LOG_D("Added enumerated type '%s' to dictionary", enumtypename);

        return true;
}

static unsigned int
hexchar(char input) {
        if (input >= '0' and input <= '9')
                return input - '0';
        if (input >= 'a' and input <= 'f')
                return input - 'a' + 10;
        if (input >= 'A' and input <= 'F')
                return input - 'a' + 10;

        return 0;
}

static char
hexdecode(const char *input) {
        return (char)(hexchar(input[0])*16 + hexchar(input[1]));
}

static char *
unpack_enumval_os(const Json::Value &enumvalue, unsigned int &len)
{
        if (!enumvalue.isString()) {
                len = 0;
                return NULL;
        }
        len = enumvalue.asString().size();
        if (enumvalue.asString()[0] == '<' && enumvalue.asString()[len-1] == '>') {
                char *retstr;
                len = len/2 - 1;
                if ((retstr = (char *)malloc(len)) == NULL) {
                        return retstr;
                }
                for (size_t i=0; i<len; i++) {
                        retstr[i] = hexdecode(enumvalue.asCString()+2*i+1);
                }
                return retstr;
        }

        return strdup(enumvalue.asCString());
}

static bool
add_enumvalue(const Json::Value &enumvalue, enum dict_avp_basetype basetype, struct dict_object *enumtype, const char *avpname)
{
        struct dict_enumval_data data;
        int ret;
        bool free_os = false;

        memset(&data, 0, sizeof(data));
        data.enum_name = (char *)(void *)enumvalue["Name"].asCString();
        switch (basetype) {
        case AVP_TYPE_INTEGER32:
                if (enumvalue["Code"].isInt()) {
                        data.enum_value.i32 = enumvalue["Code"].asInt();
                } else {
                        LOG_E("unsupported EnumValue '%s' for AVP '%s', base type Integer32", enumvalue["Name"].asCString(), avpname);
                        return false;
                }
                break;
        case AVP_TYPE_UNSIGNED32:
                if (enumvalue["Code"].isInt()) {
                        data.enum_value.u32 = enumvalue["Code"].asInt();
                } else {
                        LOG_E("unsupported EnumValue '%s' for AVP '%s', base type Unsigned32", enumvalue["Name"].asCString(), avpname);
                        return false;
                }
                break;
        case AVP_TYPE_OCTETSTRING:
                if (enumvalue["Code"].isString()) {
                        unsigned int len;
                        data.enum_value.os.data = (unsigned char *)unpack_enumval_os(enumvalue["Code"], len);
                        data.enum_value.os.len = len;
                        free_os = true;
                } else {
                        LOG_E("unsupported EnumValue '%s' for AVP '%s', base type OctetString", enumvalue["Name"].asCString(), avpname);
                        return false;
                }
                break;
        /* TODO: add other cases when we find that they occur */
        default:
                LOG_E("unsupported EnumValue type '%s' for AVP '%s', type %d", enumvalue["Name"].asCString(), avpname, (int)basetype);
                return false;
        }

        if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_ENUMVAL, &data, enumtype, NULL)) != 0) {
                LOG_E("error adding EnumValue '%s' for AVP '%s' to dictionary: %s", enumvalue["Name"].asCString(), avpname, strerror(errno));
                return false;
        }
        LOG_D("Added enumerated value '%s' to dictionary", enumvalue["Name"].asCString());

        if (free_os) {
                free(data.enum_value.os.data);
        }

        return true;
}

static bool
add_avp(const Json::Value &avp)
{
        int bt;
        struct dict_avp_data data;
        struct dict_object *type = NULL;
        struct dict_avp_request dar;
        struct dict_object *existing_avp_obj;
        int fds_ret;

        if (!avp["Code"] || !avp["Name"] || !avp["Type"]) {
                LOG_E("invalid AVP, need Code, Name, and Type");
                return false;
        }

        memset(&data, 0, sizeof(data));

        data.avp_code = avp["Code"].asUInt();
        data.avp_vendor = 0;
        data.avp_name = (char *)(void *)avp["Name"].asCString();
        data.avp_flag_mask = 0;
        if (avp["Flags"]["Must"] != Json::Value::null) {
                data.avp_flag_mask = data.avp_flag_val = make_avp_flags(avp["Flags"]["Must"].asCString());
        }
        if (avp["Flags"]["MustNot"] != Json::Value::null) {
                data.avp_flag_mask |= make_avp_flags(avp["Flags"]["MustNot"].asCString());
        }
        if (avp["Vendor"] != Json::Value::null) {
                data.avp_vendor = avp["Vendor"].asUInt();
                if ((data.avp_flag_mask & AVP_FLAG_VENDOR) == 0) {
                        LOG_D("Vendor flag not set for AVP '%s'", data.avp_name);
                        // error out?
                }
                data.avp_flag_mask |= AVP_FLAG_VENDOR;
                data.avp_flag_val |= AVP_FLAG_VENDOR;
        }
        bt = make_type(avp["Type"].asCString(), &type);
        if (bt == -1) {
                LOG_E("Unknown AVP type %s", avp["Type"].asCString());
                return false;
        }
        data.avp_basetype = (enum dict_avp_basetype)bt;

        if (avp["EnumValues"] != Json::Value::null || avp["Type"] == "Enumerated") {
                if (avp["Type"] != "Enumerated") {
                        LOG_D("AVP '%s': EnumValues defined for non-Enumerated type (freeDiameter extension); type is '%s'", avp["Name"].asCString(), avp["Type"].asCString());
                }
                Json::Value enumvalues = avp["EnumValues"];
                std::string enumtypename = "Enumerated(" + avp["Name"].asString() + ")";
                if (!add_enumtype(enumtypename.c_str(), data.avp_basetype, &type)) {
                        LOG_E("Unknown AVP Enum: %s", enumtypename.c_str());
                        return false;
                }
                for (Json::ArrayIndex i=0; i<enumvalues.size(); i++) {
                        if (!add_enumvalue(enumvalues[i], data.avp_basetype, type, avp["Name"].asCString())) {
                                LOG_E("Unknown AVP Enum Value: %s", avp["Name"].asCString());
                                return false;
                        }
                }
        }

        dar.avp_vendor = data.avp_vendor;
        // dar.avp_code = data.avp_code;
        dar.avp_name = data.avp_name;

        fds_ret = fd_dict_search(fd_g_config->cnf_dict, DICT_AVP, AVP_BY_NAME_AND_VENDOR, &dar, &existing_avp_obj, ENOENT);
        if (fds_ret == 0) {
                struct dict_avp_data existing_avp;
                if (fd_dict_getval(existing_avp_obj, &existing_avp) < 0) {
                        LOG_E("Cannot get information for AVP '%s' from dictionary", data.avp_name);
                        return false;
                }
                if (data.avp_code != existing_avp.avp_code
//                    || data.avp_vendor != existing_avp.avp_vendor
//                    || strcmp(data.avp_name, existing_avp.avp_name) != 0
//                    || data.avp_flag_mask != existing_avp.avp_flag_mask
                    || data.avp_vendor != existing_avp.avp_vendor
                    || data.avp_basetype != existing_avp.avp_basetype) {
                        LOG_E("AVP with name '%s' but different properties already exists in dictionary (existing vs. new: Code: %d/%d, Flags: %x/%x, Vendor Id: %d/%d, Basetype: %d/%d)", data.avp_name, data.avp_code, existing_avp.avp_code, data.avp_flag_mask, existing_avp.avp_flag_mask, data.avp_vendor, existing_avp.avp_vendor, data.avp_basetype, existing_avp.avp_basetype);
                        return false;
                }
        } else if (fds_ret == ENOENT) {
                switch(fd_dict_new(fd_g_config->cnf_dict, DICT_AVP, &data, type, NULL)) {
                case 0:
                        LOG_D("Added AVP '%s' to dictionary", data.avp_name);
                        break;
                case EINVAL:
                        LOG_E("error adding AVP '%s' to dictionary: invalid data", data.avp_name);
                        return false;
                case EEXIST:
                        LOG_E("error adding AVP '%s' to dictionary: duplicate entry", data.avp_name);
                        return false;
                default:
                        LOG_E("error adding AVP '%s' to dictionary: %s", data.avp_name, strerror(errno));
                        return false;
                }
        } else {
                LOG_E("error looking up AVP '%s' in dictionary: %s", data.avp_name, strerror(errno));
                return false;
        }
            
        return true;
}

static bool
add_avps(const Json::Value &config)
{
        Json::Value avps = config["AVPs"];
        if (avps == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<avps.size(); i++) {
                if (!add_avp(avps[i])) {
                        LOG_E("error adding AVP to dictionary");
                        return false;
                }
        }

        return true;
}

static bool
look_up_application(const Json::Value &application, struct dict_object **application_object)
{
        fd_dict_search(fd_g_config->cnf_dict, DICT_APPLICATION, APPLICATION_BY_NAME, (void *)application.asCString(), application_object, 0);
        if (*application_object == NULL) {
                LOG_E("Application '%s' not found", application.asCString());
                return false;
        }
        return true;
}

static bool
add_commands(const Json::Value &config)
{
        Json::Value commands = config["Commands"];
        if (commands == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<commands.size(); i++) {
                int ret;
                struct dict_cmd_data command_data;
                command_data.cmd_code = commands[i]["Code"].asUInt();
                command_data.cmd_name = (char *)(void *)commands[i]["Name"].asCString();
                command_data.cmd_flag_mask = 0;
                command_data.cmd_flag_val = 0;
                if (commands[i]["Flags"]["Must"] != Json::Value::null) {
                        command_data.cmd_flag_mask = command_data.cmd_flag_val = make_command_flags(commands[i]["Flags"]["Must"].asCString());
                }
                if (commands[i]["Flags"]["MustNot"] != Json::Value::null) {
                        command_data.cmd_flag_mask |= make_command_flags(commands[i]["Flags"]["MustNot"].asCString());
                }
                if (commands[i]["Application"] != Json::Value::null) {
                        struct dict_object *application_object;
                        if (!look_up_application(commands[i]["Application"], &application_object)) {
                                LOG_E("Application not found in dictionary");
                                return false;
                        }
                        if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_COMMAND, &command_data, application_object, NULL)) != 0) {
                                LOG_E("error adding Command '%s' to dictionary: %s", commands[i]["Name"].asCString(), strerror(ret));
                                return false;
                        }
                } else {
                        if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_COMMAND, &command_data, NULL, NULL)) != 0) {
                                LOG_E("error adding Command '%s' to dictionary: %s", commands[i]["Name"].asCString(), strerror(ret));
                                return false;
                        }
                }
                LOG_D("Added Command '%s' to dictionary", commands[i]["Name"].asCString());
        }

        return true;
}

static bool
look_up_avp(const Json::Value &avp, const Json::Value &vendor, struct dict_object **avp_object)
{
        if (vendor != Json::Value::null) {
                struct dict_avp_request_ex dare;
                memset(&dare, 0, sizeof(dare));
                if (vendor.isInt()) {
                        LOG_D("Looking for AVP '%s', Vendor %d", avp.asCString(), vendor.asInt());
                        dare.avp_vendor.vendor_id = vendor.asUInt();
                } else if (vendor.isString()) {
                        LOG_D("Looking for AVP '%s', Vendor '%s'", avp.asCString(), vendor.asCString());
                        dare.avp_vendor.vendor_name = vendor.asCString();
                } else {
                        LOG_E("error finding AVP '%s', invalid value for Vendor", avp.asCString());
                        return false;
                }
                dare.avp_data.avp_name = avp.asCString();
                fd_dict_search(fd_g_config->cnf_dict, DICT_AVP, AVP_BY_STRUCT, &dare, avp_object, 0);
        } else {
                fd_dict_search(fd_g_config->cnf_dict, DICT_AVP, AVP_BY_NAME, (void *)avp.asCString(), avp_object, 0);
        }
        if (*avp_object == NULL) {
                LOG_E("AVP '%s' not found", avp.asCString());
                return false;
        }
        return true;
}

static bool
add_rule_entry(const Json::Value &rule, struct dict_object *parent)
{
        struct dict_rule_data data;
        int ret;

        memset(&data, 0, sizeof(data));
        data.rule_min = -1;
        if (rule["Min"] != Json::Value::null) {
                if (rule["Min"].isInt())
                        data.rule_min = rule["Min"].asInt();
                else if (rule["Min"].isString() && strcmp(rule["Min"].asCString(), "unbounded") == 0)
                        data.rule_min = -1;
                else {
                        LOG_E("error adding rule with AVP '%s', invalid value for Min", rule["AVP"].asCString());
                }
                data.rule_min = rule["Min"].asInt();
        }
        if (data.rule_min > 0) {
                data.rule_position = RULE_REQUIRED;
        } else {
                data.rule_position = RULE_OPTIONAL;
        }
        data.rule_max = -1;
        if (rule["Max"] != Json::Value::null) {
                if (rule["Max"].isInt())
                        data.rule_max = rule["Max"].asInt();
                else if (rule["Max"].isString() && strcmp(rule["Max"].asCString(), "unbounded") == 0)
                        data.rule_max = -1;
                else {
                        LOG_E("error adding rule with AVP '%s', invalid value for Max", rule["AVP"].asCString());
                        return false;
                }
        }
        if (rule["First"] == true) {
                data.rule_position = RULE_FIXED_HEAD;
                data.rule_order = 1;
        }

        LOG_D("Looking up AVP '%s' for rule", rule["AVP"].asCString());
        if (!look_up_avp(rule["AVP"], rule["Vendor"], &data.rule_avp)) {
                LOG_E("Adding rule: AVP '%s' not found", rule["AVP"].asCString());
                return false;
        }
        LOG_D("Found AVP '%s' for rule", rule["AVP"].asCString());
        if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_RULE, &data, parent, NULL)) < 0) {
                LOG_E("error adding rule with AVP '%s': %s", rule["AVP"].asCString(), strerror(ret));
                return false;
        }
        LOG_D("Added AVP '%s' to rule", rule["AVP"].asCString());
        return true;
}


static bool
add_command_rule(const Json::Value &rule)
{
        int ret;
        struct dict_object *command;

        if ((ret=fd_dict_search(fd_g_config->cnf_dict, DICT_COMMAND, CMD_BY_NAME, rule["Command"].asCString(), &command, ENOENT)) < 0) {
                LOG_E("Command '%s' not found in dictionary: %s", rule["Command"].asCString(), strerror(ret));
                return false;
        }

        Json::Value content = rule["Content"];
        if (content == Json::Value::null) {
                LOG_E("No rules for Command '%s'", rule["Command"].asCString());
                return false;
        }
        for (Json::ArrayIndex i=0; i<content.size(); i++) {
                if (!add_rule_entry(content[i], command)) {
                        LOG_E("error adding command rule to dictionary");
                        return false;
                }
        }
        LOG_D("Added rules for Command '%s'", rule["Command"].asCString());

        return true;
}

static bool
add_avp_rule(const Json::Value &rule)
{
        struct dict_object *avp;
        struct dict_avp_data user_name_data;

        if (!look_up_avp(rule["AVP"], rule["Vendor"], &avp)) {
                LOG_E("AVP '%s' not found in dictionary", rule["AVP"].asCString());
                return false;
        }

        if (fd_dict_getval(avp, &user_name_data) < 0) {
                LOG_E("Cannot get information for AVP '%s' from dictionary", rule["AVP"].asCString());
                return false;
        }
        if (user_name_data.avp_basetype != AVP_TYPE_GROUPED) {
                LOG_E("Invalid type for AVP '%s': cannot define rule for non-Grouped AVP", rule["AVP"].asCString());
                return false;
        }

        Json::Value content = rule["Content"];
        if (content == Json::Value::null) {
                LOG_E("No rules for AVP '%s'", rule["AVP"].asCString());
                return false;
        }
        for (Json::ArrayIndex i=0; i<content.size(); i++) {
                if (!add_rule_entry(content[i], avp)) {
                        LOG_E("error adding AVP rule to dictionary");
                        return false;
                }
        }
        LOG_D("Added rules for AVP '%s'", rule["AVP"].asCString());

        return true;
}

static bool
add_command_rules(const Json::Value &config)
{
        Json::Value commandrules = config["CommandRules"];
        if (commandrules == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<commandrules.size(); i++) {
                if (!add_command_rule(commandrules[i])) {
                        LOG_E("error adding command rules to dictionary");
                        return false;
                }
        }

        return true;
}

static bool
add_avp_rules(const Json::Value &config)
{
        Json::Value avprules = config["AVPRules"];
        if (avprules == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<avprules.size(); i++) {
                if (!add_avp_rule(avprules[i])) {
                        LOG_E("error adding AVP rules to dictionary");
                        return false;
                }
        }

        return true;
}

struct base_types_map {
        std::string name;
        enum dict_avp_basetype value;
};
struct base_types_map base_types[] = {
        { "OctetString", AVP_TYPE_OCTETSTRING },
        { "Integer32", AVP_TYPE_INTEGER32 },
        { "Integer64", AVP_TYPE_INTEGER64 },
        { "Unsigned32", AVP_TYPE_UNSIGNED32 },
        { "Enumerated", AVP_TYPE_INTEGER32 },
        { "Unsigned64", AVP_TYPE_UNSIGNED64 },
        { "Float32", AVP_TYPE_FLOAT32 },
        { "Float64", AVP_TYPE_FLOAT64 }
};

static int
basic_type(std::string name) {
        for (unsigned int i=0; i<sizeof(base_types)/sizeof(base_types[0]); i++) {
                if (name == base_types[i].name)
                        return base_types[i].value;
        }
        return -1;
}

static bool
add_types(const Json::Value &config)
{
        Json::Value types = config["Types"];
        if (types == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<types.size(); i++) {
                int ret;
                struct dict_type_data type_data;
                struct dict_object *base_type;
                struct dict_type_data base_data;

                ret = fd_dict_search(fd_g_config->cnf_dict, DICT_TYPE, TYPE_BY_NAME, types[i]["Base"].asCString(), &base_type, ENOENT);
                switch (ret) {
                case 0:
                        if (fd_dict_getval(base_type, &base_data) < 0) {
                                LOG_E("Error looking up base type '%s' for type '%s'", types[i]["Base"].asCString(), types[i]["Name"].asCString());
                                return false;
                        }
                        type_data.type_base = base_data.type_base;
                        break;

                case ENOENT:
                        /* not an extended type, perhaps a basic one? */
                        ret = basic_type(types[i]["Base"].asString());
                        if (ret != -1) {
                                type_data.type_base = (enum dict_avp_basetype)ret;
                                break;
                        }
                        /* fallthrough */

                default:
                        /* not found, or error */
                        LOG_E("Base type '%s' for type '%s' not found", types[i]["Base"].asCString(), types[i]["Name"].asCString());
                        return false;
                }

                type_data.type_name = (char *)(void *)types[i]["Name"].asCString();
                if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_TYPE, &type_data, NULL, NULL)) != 0) {
                        LOG_E("error adding Type '%s' to dictionary: %s", types[i]["Name"].asCString(), strerror(ret));
                        return false;
                }
                LOG_D("Added Type '%s' to dictionary", types[i]["Name"].asCString());
        }

        return true;
}

static bool
add_vendors(const Json::Value &config)
{
        Json::Value vendors = config["Vendors"];
        if (vendors == Json::Value::null)
                return true;
        for (Json::ArrayIndex i=0; i<vendors.size(); i++) {
                int ret;
                struct dict_vendor_data vendor_data;
                vendor_data.vendor_id = vendors[i]["Code"].asUInt();
                vendor_data.vendor_name = (char *)(void *)vendors[i]["Name"].asCString();
                if ((ret=fd_dict_new(fd_g_config->cnf_dict, DICT_VENDOR, &vendor_data, NULL, NULL)) != 0) {
                        LOG_E("error adding Vendor '%s' to dictionary: %s", vendors[i]["Name"].asCString(), strerror(ret));
                        return false;
                }
                LOG_D("Added Vendor '%s' to dictionary", vendors[i]["Name"].asCString());
        }

        return true;
}

static bool
parse_json_from_file(const char *conffile, Json::Value &jv)
{
        struct stat sb;
        char *buf;
        FILE *fp;
        Json::Reader reader;
        static Json::SchemaValidator *validator = NULL;

        if (conffile == NULL || stat(conffile, &sb) < 0 || !S_ISREG(sb.st_mode)) {
                LOG_E("invalid or missing configuration: %s", conffile ?: "(null)");
                return false;
        }
        if ((buf=(char *)malloc(static_cast<size_t>(sb.st_size + 1))) == NULL) {
                LOG_E("malloc failure of %zu bytes", static_cast<size_t>(sb.st_size));
                return false;
        }
        if ((fp=fopen(conffile, "r")) == NULL) {
                LOG_E("error opening file '%s': %s", conffile, strerror(errno));
                return false;
        }
        if (fread(buf, static_cast<size_t>(sb.st_size), 1, fp) != 1) {
                LOG_E("error reading from file '%s': %s", conffile, strerror(errno));
                return false;
        }
        (void)fclose(fp);

        buf[sb.st_size] = '\0';

        if (!reader.parse (buf, jv)) {
                LOG_E("error parsing JSON data from file '%s': %s", conffile, reader.getFormattedErrorMessages().c_str());
                return false;
        }
        free(buf);

        if (validator == NULL) {
                try {
                        validator = new Json::SchemaValidator(std::string(dict_json_dict_schema));
                } catch (Json::SchemaValidator::Exception &e) {
                        LOG_E("error creating JSON schema validator: %s", e.type_message().c_str());
                        return false;
                }
        }

        if (!validator->validate(jv)) {
                LOG_E("error validating config file %s:", conffile);
                const std::vector<Json::SchemaValidator::Error> &errors = validator->errors();
                for (std::vector<Json::SchemaValidator::Error>::const_iterator it = errors.begin(); it != errors.end(); ++it) {
                        LOG_E("  %s:%s", it->path.c_str(), it->message.c_str());
                }
                return false;
        }

#if 0
        Json::StyledWriter styledWriter;
        std::cout << styledWriter.write(jv);
#endif

        return true;
}

static bool
read_dictionary(const char *filename)
{
        Json::Value config = Json::Value::null;
        if (!parse_json_from_file (filename, config)) {
                LOG_E("error parsing JSON file");
                return false;
        }

        if (!add_vendors(config)) {
                LOG_E("error adding vendors");
                return false;
        }
        if (!add_types(config)) {
                LOG_E("error adding types");
                return false;
        }
        if (!add_avps(config)) {
                LOG_E("error adding AVPs");
                return false;
        }
        if (!add_applications(config)) {
                LOG_E("error adding applications");
                return false;
        }
        if (!add_commands(config)) {
                LOG_E("error adding commands");
                return false;
        }
        if (!add_command_rules(config)) {
                LOG_E("error adding command rules");
                return false;
        }
        if (!add_avp_rules(config)) {
                LOG_E("error adding AVP rules");
                return false;
        }

        return true;
}

static int
dict_json_entry(char * conffile)
{
        Json::Value main_config = Json::Value::null;
        char *filename, *filename_base, *p;

        TRACE_ENTRY("%p", conffile);

        filename_base = strdup(conffile);
        if (filename_base == NULL) {
                LOG_E("error initialising JSON dictionary extension: %s", strerror(errno));
                return 1;
        }
        filename = filename_base;
        while ((p=strsep(&filename, ";")) != NULL) {
                LOG_D("parsing dictionary '%s'", p);
                if (!read_dictionary(p)) {
                        LOG_E("error reading JSON dictionary '%s'", p);
                        return EINVAL;
                }
                LOG_N("loaded JSON dictionary '%s'", p);
                if (filename == NULL)
                        break;
        }

        free(filename_base);
        LOG_N("Extension 'Dictionary definitions from JSON dictionaries' initialized");
        return 0;
}

extern "C" {
        EXTENSION_ENTRY("dict_json", dict_json_entry);
}
"Welcome to our mercurial repository"