2 * Asterisk -- An open source telephony toolkit.
4 * Copyright (C) 2013, Digium, Inc.
6 * David M. Lee, II <dlee@digium.com>
8 * See http://www.asterisk.org for more information about
9 * the Asterisk project. Please do not directly contact
10 * any of the maintainers of this project for assistance;
11 * the project provides a web site, mailing lists and IRC
12 * channels for your use.
14 * This program is free software, distributed under the terms of
15 * the GNU General Public License Version 2. See the LICENSE file
16 * at the top of the source tree.
21 * \brief Config framework stuffz for ARI.
22 * \author David M. Lee, II <dlee@digium.com>
27 ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
29 #include "asterisk/config_options.h"
32 /*! \brief Locking container for safe configuration access. */
33 static AO2_GLOBAL_OBJ_STATIC(confs);
35 /*! \brief Mapping of the ARI conf struct's globals to the
36 * general context in the config file. */
37 static struct aco_type general_option = {
40 .item_offset = offsetof(struct ast_ari_conf, general),
41 .category = "^general$",
42 .category_match = ACO_WHITELIST,
45 static struct aco_type *general_options[] = ACO_TYPES(&general_option);
47 /*! \brief Encoding format handler converts from boolean to enum. */
48 static int encoding_format_handler(const struct aco_option *opt,
49 struct ast_variable *var, void *obj)
51 struct ast_ari_conf_general *general = obj;
53 if (!strcasecmp(var->name, "pretty")) {
54 general->format = ast_true(var->value) ?
55 AST_JSON_PRETTY : AST_JSON_COMPACT;
63 /*! \brief Parses the ast_ari_password_format enum from a config file */
64 static int password_format_handler(const struct aco_option *opt,
65 struct ast_variable *var, void *obj)
67 struct ast_ari_conf_user *user = obj;
69 if (strcasecmp(var->value, "plain") == 0) {
70 user->password_format = ARI_PASSWORD_FORMAT_PLAIN;
71 } else if (strcasecmp(var->value, "crypt") == 0) {
72 user->password_format = ARI_PASSWORD_FORMAT_CRYPT;
80 /*! \brief Destructor for \ref ast_ari_conf_user */
81 static void user_dtor(void *obj)
83 struct ast_ari_conf_user *user = obj;
84 ast_debug(3, "Disposing of user %s\n", user->username);
85 ast_free(user->username);
88 /*! \brief Allocate an \ref ast_ari_conf_user for config parsing */
89 static void *user_alloc(const char *cat)
91 RAII_VAR(struct ast_ari_conf_user *, user, NULL, ao2_cleanup);
97 ast_debug(3, "Allocating user %s\n", cat);
99 user = ao2_alloc_options(sizeof(*user), user_dtor,
100 AO2_ALLOC_OPT_LOCK_NOLOCK);
105 user->username = ast_strdup(cat);
106 if (!user->username) {
114 /*! \brief Sorting function for use with red/black tree */
115 static int user_sort_cmp(const void *obj_left, const void *obj_right, int flags)
117 const struct ast_ari_conf_user *user_left = obj_left;
119 if (flags & OBJ_PARTIAL_KEY) {
120 const char *key_right = obj_right;
121 return strncasecmp(user_left->username, key_right,
123 } else if (flags & OBJ_KEY) {
124 const char *key_right = obj_right;
125 return strcasecmp(user_left->username, key_right);
127 const struct ast_ari_conf_user *user_right = obj_right;
128 const char *key_right = user_right->username;
129 return strcasecmp(user_left->username, key_right);
133 /*! \brief \ref aco_type item_find function */
134 static void *user_find(struct ao2_container *tmp_container, const char *cat)
140 return ao2_find(tmp_container, cat, OBJ_KEY);
143 static struct aco_type user_option = {
146 .category_match = ACO_BLACKLIST,
147 .category = "^general$",
148 .matchfield = "type",
149 .matchvalue = "user",
150 .item_alloc = user_alloc,
151 .item_find = user_find,
152 .item_offset = offsetof(struct ast_ari_conf, users),
155 static struct aco_type *user[] = ACO_TYPES(&user_option);
157 /*! \brief \ref ast_ari_conf destructor. */
158 static void conf_destructor(void *obj)
160 struct ast_ari_conf *cfg = obj;
162 ast_string_field_free_memory(cfg->general);
164 ao2_cleanup(cfg->general);
165 ao2_cleanup(cfg->users);
168 /*! \brief Allocate an \ref ast_ari_conf for config parsing */
169 static void *conf_alloc(void)
171 RAII_VAR(struct ast_ari_conf *, cfg, NULL, ao2_cleanup);
173 cfg = ao2_alloc_options(sizeof(*cfg), conf_destructor,
174 AO2_ALLOC_OPT_LOCK_NOLOCK);
179 cfg->general = ao2_alloc_options(sizeof(*cfg->general), NULL,
180 AO2_ALLOC_OPT_LOCK_NOLOCK);
184 aco_set_defaults(&general_option, "general", cfg->general);
186 if (ast_string_field_init(cfg->general, 64)) {
190 cfg->users = ao2_container_alloc_rbtree(AO2_ALLOC_OPT_LOCK_NOLOCK,
191 AO2_CONTAINER_ALLOC_OPT_DUPS_REPLACE, user_sort_cmp, NULL);
197 #define CONF_FILENAME "ari.conf"
199 /*! \brief The conf file that's processed for the module. */
200 static struct aco_file conf_file = {
201 /*! The config file name. */
202 .filename = CONF_FILENAME,
203 /*! The mapping object types to be processed. */
204 .types = ACO_TYPES(&general_option, &user_option),
207 CONFIG_INFO_STANDARD(cfg_info, confs, conf_alloc,
208 .files = ACO_FILES(&conf_file));
210 struct ast_ari_conf *ast_ari_config_get(void)
212 struct ast_ari_conf *res = ao2_global_obj_ref(confs);
215 "Error obtaining config from " CONF_FILENAME "\n");
220 struct ast_ari_conf_user *ast_ari_config_validate_user(const char *username,
221 const char *password)
223 RAII_VAR(struct ast_ari_conf *, conf, NULL, ao2_cleanup);
224 RAII_VAR(struct ast_ari_conf_user *, user, NULL, ao2_cleanup);
227 conf = ast_ari_config_get();
232 user = ao2_find(conf->users, username, OBJ_KEY);
237 if (ast_strlen_zero(user->password)) {
239 "User '%s' missing password; authentication failed\n",
244 switch (user->password_format) {
245 case ARI_PASSWORD_FORMAT_PLAIN:
246 is_valid = strcmp(password, user->password) == 0;
248 case ARI_PASSWORD_FORMAT_CRYPT:
249 is_valid = ast_crypt_validate(password, user->password);
261 /*! \brief Callback to validate a user object */
262 static int validate_user_cb(void *obj, void *arg, int flags)
264 struct ast_ari_conf_user *user = obj;
266 if (ast_strlen_zero(user->password)) {
267 ast_log(LOG_WARNING, "User '%s' missing password\n",
274 /*! \brief Load (or reload) configuration. */
275 static int process_config(int reload)
277 RAII_VAR(struct ast_ari_conf *, conf, NULL, ao2_cleanup);
279 switch (aco_process_config(&cfg_info, reload)) {
280 case ACO_PROCESS_ERROR:
283 case ACO_PROCESS_UNCHANGED:
287 conf = ast_ari_config_get();
289 ast_assert(0); /* We just configured; it should be there */
293 if (ao2_container_count(conf->users) == 0) {
294 ast_log(LOG_ERROR, "No configured users for ARI\n");
297 ao2_callback(conf->users, OBJ_NODATA, validate_user_cb, NULL);
302 int ast_ari_config_init(void)
304 if (aco_info_init(&cfg_info)) {
305 aco_info_destroy(&cfg_info);
309 aco_option_register(&cfg_info, "enabled", ACO_EXACT, general_options,
310 "yes", OPT_BOOL_T, 1,
311 FLDSET(struct ast_ari_conf_general, enabled));
312 aco_option_register_custom(&cfg_info, "pretty", ACO_EXACT,
313 general_options, "no", encoding_format_handler, 0);
314 aco_option_register(&cfg_info, "auth_realm", ACO_EXACT, general_options,
315 "Asterisk REST Interface", OPT_CHAR_ARRAY_T, 0,
316 FLDSET(struct ast_ari_conf_general, auth_realm),
318 aco_option_register(&cfg_info, "allowed_origins", ACO_EXACT, general_options,
319 "", OPT_STRINGFIELD_T, 0,
320 STRFLDSET(struct ast_ari_conf_general, allowed_origins));
322 aco_option_register(&cfg_info, "type", ACO_EXACT, user, NULL,
324 aco_option_register(&cfg_info, "read_only", ACO_EXACT, user,
326 FLDSET(struct ast_ari_conf_user, read_only));
327 aco_option_register(&cfg_info, "password", ACO_EXACT, user,
328 "", OPT_CHAR_ARRAY_T, 0,
329 FLDSET(struct ast_ari_conf_user, password), ARI_PASSWORD_LEN);
330 aco_option_register_custom(&cfg_info, "password_format", ACO_EXACT,
331 user, "plain", password_format_handler, 0);
333 return process_config(0);
336 int ast_ari_config_reload(void)
338 return process_config(1);
341 void ast_ari_config_destroy(void)
343 aco_info_destroy(&cfg_info);
344 ao2_global_obj_release(confs);