mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-25 22:18:07 +00:00 
			
		
		
		
	Testing revealed that the cache added no benefit but that it could consume excessive memory. Two new index related functions were created: ast_sounds_get_index_for_file() and ast_media_index_update_for_file() which restrict index updating to specific sound files. The original ast_sounds_get_index() and ast_media_index_update() calls are still available but since they no longer cache the results internally, developers should re-use an index they may already have instead of calling ast_sounds_get_index() repeatedly. If information for only a single file is needed, ast_sounds_get_index_for_file() should be called instead of ast_sounds_get_index(). The media_index directory scan code was elimininated in favor of using the existing ast_file_read_dirs() function. Since there's no more cache, ast_sounds_index_init now only registers the sounds cli commands instead of generating the initial index and subscribing to stasis format register/unregister messages. "sounds" is no longer a valid target for the "module reload" command. Both the sounds cli commands and the sounds ari resources were refactored to only call ast_sounds_get_index() once per invocation and to use ast_sounds_get_index_for_file() when a specific sound file is requested. Change-Id: I1cef327ba1b0648d85d218b70ce469ad07f4aa8d
		
			
				
	
	
		
			236 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			236 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 2012 - 2013, Digium, Inc.
 | |
|  *
 | |
|  * David M. Lee, II <dlee@digium.com>
 | |
|  *
 | |
|  * See http://www.asterisk.org for more information about
 | |
|  * the Asterisk project. Please do not directly contact
 | |
|  * any of the maintainers of this project for assistance;
 | |
|  * the project provides a web site, mailing lists and IRC
 | |
|  * channels for your use.
 | |
|  *
 | |
|  * This program is free software, distributed under the terms of
 | |
|  * the GNU General Public License Version 2. See the LICENSE file
 | |
|  * at the top of the source tree.
 | |
|  */
 | |
| 
 | |
| /*! \file
 | |
|  *
 | |
|  * \brief /api-docs/sounds.{format} implementation- Sound resources
 | |
|  *
 | |
|  * \author David M. Lee, II <dlee@digium.com>
 | |
|  */
 | |
| 
 | |
| #include "asterisk.h"
 | |
| 
 | |
| #include "resource_sounds.h"
 | |
| #include "asterisk/media_index.h"
 | |
| #include "asterisk/sounds_index.h"
 | |
| #include "asterisk/format.h"
 | |
| #include "asterisk/format_cap.h"
 | |
| #include "asterisk/json.h"
 | |
| 
 | |
| /*! \brief arguments that are necessary for adding format/lang pairs */
 | |
| struct lang_format_info {
 | |
| 	struct ast_json *format_list;	/*!< The embedded array to which format/lang pairs should be added */
 | |
| 	const char *filename;		/*!< Name of the file for which to add format/lang pairs */
 | |
| 	const char *format_filter;	/*!< Format filter provided in the request */
 | |
| };
 | |
| 
 | |
| /*! \brief Add format/lang pairs to the array embedded in the sound object */
 | |
| static int add_format_information_cb(void *obj, void *arg, void *data, int flags)
 | |
| {
 | |
| 	char *language = obj;
 | |
| 	struct lang_format_info *args = arg;
 | |
| 	int idx;
 | |
| 	RAII_VAR(struct ast_format_cap *, cap, NULL, ao2_cleanup);
 | |
| 	struct ast_media_index *sounds_index = data;
 | |
| 
 | |
| 	if (!sounds_index) {
 | |
| 		return CMP_STOP;
 | |
| 	}
 | |
| 
 | |
| 	cap = ast_media_get_format_cap(sounds_index, args->filename, language);
 | |
| 	if (!cap) {
 | |
| 		return CMP_STOP;
 | |
| 	}
 | |
| 
 | |
| 	for (idx = 0; idx < ast_format_cap_count(cap); idx++) {
 | |
| 		struct ast_format *format = ast_format_cap_get_format(cap, idx);
 | |
| 		struct ast_json *lang_format_pair;
 | |
| 
 | |
| 		if (!ast_strlen_zero(args->format_filter)
 | |
| 			&& strcmp(args->format_filter, ast_format_get_name(format))) {
 | |
| 			ao2_ref(format, -1);
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		lang_format_pair = ast_json_pack("{s: s, s: s}",
 | |
| 			"language", language,
 | |
| 			"format", ast_format_get_name(format));
 | |
| 		if (!lang_format_pair) {
 | |
| 			ao2_ref(format, -1);
 | |
| 			return CMP_STOP;
 | |
| 		}
 | |
| 
 | |
| 		ast_json_array_append(args->format_list, lang_format_pair);
 | |
| 		ao2_ref(format, -1);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*! \brief Filter out all languages not matching the specified language */
 | |
| static int filter_langs_cb(void *obj, void *arg, int flags)
 | |
| {
 | |
| 	char *lang_filter = arg;
 | |
| 	char *lang = obj;
 | |
| 	if (strcmp(lang, lang_filter)) {
 | |
| 		return CMP_MATCH;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*! \brief Generate a Sound structure as documented in sounds.json for the specified filename */
 | |
| static struct ast_json *create_sound_blob(const char *filename,
 | |
| 	struct ast_ari_sounds_list_args *args, struct ast_media_index *sounds_index)
 | |
| {
 | |
| 	RAII_VAR(struct ast_json *, sound, NULL, ast_json_unref);
 | |
| 	RAII_VAR(struct ao2_container *, languages, NULL, ao2_cleanup);
 | |
| 	const char *description;
 | |
| 	struct ast_json *format_lang_list;
 | |
| 	struct lang_format_info info;
 | |
| 
 | |
| 	if (!sounds_index) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	description = ast_media_get_description(sounds_index, filename, "en");
 | |
| 	if (ast_strlen_zero(description)) {
 | |
| 		sound = ast_json_pack("{s: s, s: []}",
 | |
| 			"id", filename,
 | |
| 			"formats");
 | |
| 	} else {
 | |
| 		sound = ast_json_pack("{s: s, s: s, s: []}",
 | |
| 			"id", filename,
 | |
| 			"text", description,
 | |
| 			"formats");
 | |
| 	}
 | |
| 	if (!sound) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	format_lang_list = ast_json_object_get(sound, "formats");
 | |
| 	if (!format_lang_list) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	languages = ast_media_get_variants(sounds_index, filename);
 | |
| 	if (!languages || !ao2_container_count(languages)) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	/* filter requested languages */
 | |
| 	if (args && !ast_strlen_zero(args->lang)) {
 | |
| 		char *lang_filter = ast_strdupa(args->lang);
 | |
| 		ao2_callback(languages, OBJ_NODATA | OBJ_MULTIPLE | OBJ_UNLINK, filter_langs_cb, lang_filter);
 | |
| 		if (!languages || !ao2_container_count(languages)) {
 | |
| 			return NULL;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	info.filename = filename;
 | |
| 	info.format_list = format_lang_list;
 | |
| 	info.format_filter = NULL;
 | |
| 	if (args) {
 | |
| 		info.format_filter = args->format;
 | |
| 	}
 | |
| 	ao2_callback_data(languages, OBJ_NODATA, add_format_information_cb, &info, sounds_index);
 | |
| 
 | |
| 	/* no format/lang pairs for this sound so nothing to return */
 | |
| 	if (!ast_json_array_size(format_lang_list)) {
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return ast_json_ref(sound);
 | |
| }
 | |
| 
 | |
| struct sounds_cb_data {
 | |
| 	struct ast_ari_sounds_list_args *args;
 | |
| 	struct ast_media_index *index;
 | |
| };
 | |
| 
 | |
| /*! \brief Generate a Sound structure and append it to the output blob */
 | |
| static int append_sound_cb(void *obj, void *arg, void *data, int flags)
 | |
| {
 | |
| 	struct ast_json *sounds_array = arg;
 | |
| 	char *filename = obj;
 | |
| 	struct sounds_cb_data *cb_data = data;
 | |
| 	struct ast_json *sound_blob = create_sound_blob(filename, cb_data->args, cb_data->index);
 | |
| 	if (!sound_blob) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	ast_json_array_append(sounds_array, sound_blob);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void ast_ari_sounds_list(struct ast_variable *headers,
 | |
| 	struct ast_ari_sounds_list_args *args,
 | |
| 	struct ast_ari_response *response)
 | |
| {
 | |
| 	RAII_VAR(struct ao2_container *, sound_files, NULL, ao2_cleanup);
 | |
| 	struct ast_json *sounds_blob;
 | |
| 	RAII_VAR(struct ast_media_index *, sounds_index, ast_sounds_get_index(), ao2_cleanup);
 | |
| 	struct sounds_cb_data cb_data = {
 | |
| 		.args = args,
 | |
| 		.index = sounds_index,
 | |
| 	};
 | |
| 
 | |
| 	if (!sounds_index) {
 | |
| 		ast_ari_response_error(response, 500, "Internal Error", "Sounds index not available");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	sound_files = ast_media_get_media(sounds_index);
 | |
| 	if (!sound_files) {
 | |
| 		ast_ari_response_error(response, 500, "Internal Error", "Allocation Error");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	sounds_blob = ast_json_array_create();
 | |
| 	if (!sounds_blob) {
 | |
| 		ast_ari_response_error(response, 500, "Internal Error", "Allocation Error");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	ao2_callback_data(sound_files, OBJ_NODATA, append_sound_cb, sounds_blob, &cb_data);
 | |
| 
 | |
| 	if (!ast_json_array_size(sounds_blob)) {
 | |
| 		ast_ari_response_error(response, 404, "Not Found", "No sounds found that matched the query");
 | |
| 		ast_json_unref(sounds_blob);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	ast_ari_response_ok(response, sounds_blob);
 | |
| }
 | |
| 
 | |
| void ast_ari_sounds_get(struct ast_variable *headers,
 | |
| 	struct ast_ari_sounds_get_args *args,
 | |
| 	struct ast_ari_response *response)
 | |
| {
 | |
| 	struct ast_json *sound_blob;
 | |
| 	struct ast_media_index *sounds_index = ast_sounds_get_index_for_file(args->sound_id);
 | |
| 
 | |
| 	sound_blob = create_sound_blob(args->sound_id, NULL, sounds_index);
 | |
| 	ao2_cleanup(sounds_index);
 | |
| 	if (!sound_blob) {
 | |
| 		ast_ari_response_error(response, 404, "Not Found", "Sound not found");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	ast_ari_response_ok(response, sound_blob);
 | |
| }
 |