732 lines
18 KiB
C
732 lines
18 KiB
C
|
/*
|
||
|
* Asterisk -- An open source telephony toolkit.
|
||
|
*
|
||
|
* Copyright (C) 2021, Sangoma Technologies Corporation
|
||
|
*
|
||
|
* Kevin Harwell <kharwell@sangoma.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 Asterisk External Application Speech Engine
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*** MODULEINFO
|
||
|
<support_level>core</support_level>
|
||
|
***/
|
||
|
|
||
|
#include "asterisk.h"
|
||
|
|
||
|
#include "asterisk/astobj2.h"
|
||
|
#include "asterisk/config.h"
|
||
|
#include "asterisk/format.h"
|
||
|
#include "asterisk/format_cap.h"
|
||
|
#include "asterisk/json.h"
|
||
|
#include "asterisk/module.h"
|
||
|
#include "asterisk/speech.h"
|
||
|
#include "asterisk/sorcery.h"
|
||
|
|
||
|
#include "asterisk/res_aeap.h"
|
||
|
#include "asterisk/res_aeap_message.h"
|
||
|
|
||
|
#define SPEECH_AEAP_VERSION "0.1.0"
|
||
|
#define SPEECH_PROTOCOL "speech_to_text"
|
||
|
|
||
|
#define CONNECTION_TIMEOUT 2000
|
||
|
|
||
|
#define log_error(obj, fmt, ...) \
|
||
|
ast_log(LOG_ERROR, "AEAP speech (%p): " fmt "\n", obj, ##__VA_ARGS__)
|
||
|
|
||
|
static struct ast_json *custom_fields_to_params(const struct ast_variable *variables)
|
||
|
{
|
||
|
const struct ast_variable *i;
|
||
|
struct ast_json *obj;
|
||
|
|
||
|
if (!variables) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
obj = ast_json_object_create();
|
||
|
if (!obj) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
for (i = variables; i; i = i->next) {
|
||
|
if (i->name[0] == '@' && i->name[1]) {
|
||
|
ast_json_object_set(obj, i->name + 1, ast_json_string_create(i->value));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return obj;
|
||
|
}
|
||
|
|
||
|
/*!
|
||
|
* \internal
|
||
|
* \brief Create, and send a request to the external application
|
||
|
*
|
||
|
* Create, then sends a request to an Asterisk external application, and then blocks
|
||
|
* until a response is received or a time out occurs. Since this method waits until
|
||
|
* receiving a response the returned result is guaranteed to be pass/fail based upon
|
||
|
* a response handler's result.
|
||
|
*
|
||
|
* \param name The name of the request to send
|
||
|
* \param json The core json request data
|
||
|
* \param data Optional user data to associate with request/response
|
||
|
*
|
||
|
* \returns 0 on success, -1 on error
|
||
|
*/
|
||
|
static int speech_aeap_send_request(struct ast_aeap *aeap, const char *name,
|
||
|
struct ast_json *json, void *obj)
|
||
|
{
|
||
|
/*
|
||
|
* Wait for a response. Also since we're blocking,
|
||
|
* data is expected to be on the stack so no cleanup required.
|
||
|
*/
|
||
|
struct ast_aeap_tsx_params tsx_params = {
|
||
|
.timeout = 1000,
|
||
|
.wait = 1,
|
||
|
.obj = obj,
|
||
|
};
|
||
|
|
||
|
/* "steals" the json ref */
|
||
|
tsx_params.msg = ast_aeap_message_create_request(
|
||
|
ast_aeap_message_type_json, name, NULL, json);
|
||
|
if (!tsx_params.msg) {
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* Send "steals" the json msg ref */
|
||
|
return ast_aeap_send_msg_tsx(aeap, &tsx_params);
|
||
|
}
|
||
|
|
||
|
/*!
|
||
|
* \internal
|
||
|
* \brief Create, and send a "get" request to an external application
|
||
|
*
|
||
|
* Basic structure of the JSON message to send:
|
||
|
*
|
||
|
* { param: [<param>, ...] }
|
||
|
*
|
||
|
* \param speech The speech engine
|
||
|
* \param param The name of the parameter to retrieve
|
||
|
* \param data User data passed to the response handler
|
||
|
*
|
||
|
* \returns 0 on success, -1 on error
|
||
|
*/
|
||
|
static int speech_aeap_get(struct ast_speech *speech, const char *param, void *data)
|
||
|
{
|
||
|
if (!param) {
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* send_request handles json ref */
|
||
|
return speech_aeap_send_request(speech->data,
|
||
|
"get", ast_json_pack("{s:[s]}", "params", param), data);
|
||
|
}
|
||
|
|
||
|
struct speech_param {
|
||
|
const char *name;
|
||
|
const char *value;
|
||
|
};
|
||
|
|
||
|
/*!
|
||
|
* \internal
|
||
|
* \brief Create, and send a "set" request to an external application
|
||
|
*
|
||
|
* Basic structure of the JSON message to send:
|
||
|
*
|
||
|
* { params: { <name> : <value> } }
|
||
|
*
|
||
|
* \param speech The speech engine
|
||
|
* \param name The name of the parameter to set
|
||
|
* \param value The value of the parameter to set
|
||
|
*
|
||
|
* \returns 0 on success, -1 on error
|
||
|
*/
|
||
|
static int speech_aeap_set(struct ast_speech *speech, const char *name, const char *value)
|
||
|
{
|
||
|
if (!name) {
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* send_request handles json ref */
|
||
|
return speech_aeap_send_request(speech->data,
|
||
|
"set", ast_json_pack("{s:{s:s}}", "params", name, value), NULL);
|
||
|
}
|
||
|
|
||
|
static int handle_response_set(struct ast_aeap *aeap, struct ast_aeap_message *message, void *data)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
struct speech_setting {
|
||
|
const char *param;
|
||
|
size_t len;
|
||
|
char *buf;
|
||
|
};
|
||
|
|
||
|
static int handle_setting(struct ast_aeap *aeap, struct ast_json_iter *iter,
|
||
|
struct speech_setting *setting)
|
||
|
{
|
||
|
const char *value;
|
||
|
|
||
|
if (strcmp(ast_json_object_iter_key(iter), setting->param)) {
|
||
|
log_error(aeap, "Unable to 'get' speech setting for '%s'", setting->param);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
value = ast_json_string_get(ast_json_object_iter_value(iter));
|
||
|
if (!value) {
|
||
|
log_error(aeap, "No value for speech setting '%s'", setting->param);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
ast_copy_string(setting->buf, value, setting->len);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int handle_results(struct ast_aeap *aeap, struct ast_json_iter *iter,
|
||
|
struct ast_speech_result **speech_results)
|
||
|
{
|
||
|
struct ast_speech_result *result = NULL;
|
||
|
struct ast_json *json_results;
|
||
|
struct ast_json *json_result;
|
||
|
size_t i;
|
||
|
|
||
|
json_results = ast_json_object_iter_value(iter);
|
||
|
if (!json_results || !speech_results) {
|
||
|
log_error(aeap, "Unable to 'get' speech results");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < ast_json_array_size(json_results); ++i) {
|
||
|
if (!(result = ast_calloc(1, sizeof(*result)))) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
json_result = ast_json_array_get(json_results, i);
|
||
|
|
||
|
result->text = ast_strdup(ast_json_object_string_get(json_result, "text"));
|
||
|
result->score = ast_json_object_integer_get(json_result, "score");
|
||
|
result->grammar = ast_strdup(ast_json_object_string_get(json_result, "grammar"));
|
||
|
result->nbest_num = ast_json_object_integer_get(json_result, "best");
|
||
|
if (*speech_results) {
|
||
|
AST_LIST_NEXT(result, list) = *speech_results;
|
||
|
*speech_results = result;
|
||
|
} else {
|
||
|
*speech_results = result;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*!
|
||
|
* \internal
|
||
|
* \brief Handle a "get" response from an external application
|
||
|
*
|
||
|
* Basic structure of the expected JSON message to received:
|
||
|
*
|
||
|
* {
|
||
|
* response: "get"
|
||
|
* "params" : { <name>: <value> | [ <results> ] }
|
||
|
* }
|
||
|
*
|
||
|
* \param speech The speech engine
|
||
|
* \param param The name of the parameter to retrieve
|
||
|
* \param data User data passed to the response handler
|
||
|
*
|
||
|
* \returns 0 on success, -1 on error
|
||
|
*/
|
||
|
static int handle_response_get(struct ast_aeap *aeap, struct ast_aeap_message *message, void *data)
|
||
|
{
|
||
|
struct ast_json_iter *iter;
|
||
|
|
||
|
iter = ast_json_object_iter(ast_json_object_get(ast_aeap_message_data(message), "params"));
|
||
|
if (!iter) {
|
||
|
log_error(aeap, "no 'get' parameters returned");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
if (!strcmp(ast_json_object_iter_key(iter), "results")) {
|
||
|
return handle_results(aeap, iter, data);
|
||
|
}
|
||
|
|
||
|
return handle_setting(aeap, iter, data);
|
||
|
}
|
||
|
|
||
|
static int handle_response_setup(struct ast_aeap *aeap, struct ast_aeap_message *message, void *data)
|
||
|
{
|
||
|
struct ast_format *format = data;
|
||
|
struct ast_json *json = ast_aeap_message_data(message);
|
||
|
const char *codec_name;
|
||
|
|
||
|
if (!json) {
|
||
|
log_error(aeap, "no 'setup' object returned");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
json = ast_json_object_get(json, "codecs");
|
||
|
if (!json || ast_json_array_size(json) == 0) {
|
||
|
log_error(aeap, "no 'setup' codecs available");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
codec_name = ast_json_object_string_get(ast_json_array_get(json, 0), "name");
|
||
|
if (!codec_name || strcmp(codec_name, ast_format_get_codec_name(format))) {
|
||
|
log_error(aeap, "setup codec '%s' unsupported", ast_format_get_codec_name(format));
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static const struct ast_aeap_message_handler response_handlers[] = {
|
||
|
{ "setup", handle_response_setup },
|
||
|
{ "get", handle_response_get },
|
||
|
{ "set", handle_response_set },
|
||
|
};
|
||
|
|
||
|
static int handle_request_set(struct ast_aeap *aeap, struct ast_aeap_message *message, void *data)
|
||
|
{
|
||
|
struct ast_json_iter *iter;
|
||
|
const char *error_msg = NULL;
|
||
|
|
||
|
iter = ast_json_object_iter(ast_json_object_get(ast_aeap_message_data(message), "params"));
|
||
|
if (!iter) {
|
||
|
error_msg = "no parameter(s) requested";
|
||
|
} else if (!strcmp(ast_json_object_iter_key(iter), "results")) {
|
||
|
struct ast_speech *speech = ast_aeap_user_data_object_by_id(aeap, "speech");
|
||
|
|
||
|
if (!speech) {
|
||
|
error_msg = "no associated speech object";
|
||
|
} else if (handle_results(aeap, iter, &speech->results)) {
|
||
|
error_msg = "unable to handle results";
|
||
|
} else {
|
||
|
ast_speech_change_state(speech, AST_SPEECH_STATE_DONE);
|
||
|
}
|
||
|
} else {
|
||
|
error_msg = "can only set 'results'";
|
||
|
}
|
||
|
|
||
|
if (error_msg) {
|
||
|
log_error(aeap, "set - %s", error_msg);
|
||
|
message = ast_aeap_message_create_error(ast_aeap_message_type_json,
|
||
|
ast_aeap_message_name(message), ast_aeap_message_id(message), error_msg);
|
||
|
} else {
|
||
|
message = ast_aeap_message_create_response(ast_aeap_message_type_json,
|
||
|
ast_aeap_message_name(message), ast_aeap_message_id(message), NULL);
|
||
|
}
|
||
|
|
||
|
ast_aeap_send_msg(aeap, message);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static const struct ast_aeap_message_handler request_handlers[] = {
|
||
|
{ "set", handle_request_set },
|
||
|
};
|
||
|
|
||
|
static struct ast_aeap_params speech_aeap_params = {
|
||
|
.response_handlers = response_handlers,
|
||
|
.response_handlers_size = ARRAY_LEN(response_handlers),
|
||
|
.request_handlers = request_handlers,
|
||
|
.request_handlers_size = ARRAY_LEN(request_handlers),
|
||
|
};
|
||
|
|
||
|
/*!
|
||
|
* \internal
|
||
|
* \brief Create, and connect to an external application and send initial setup
|
||
|
*
|
||
|
* Basic structure of the JSON message to send:
|
||
|
*
|
||
|
* {
|
||
|
* "request": "setup"
|
||
|
* "codecs": [
|
||
|
* {
|
||
|
* "name": <name>,
|
||
|
* "attributes": { <name>: <value>, ..., }
|
||
|
* },
|
||
|
* ...,
|
||
|
* ],
|
||
|
* "params": { <name>: <value>, ..., }
|
||
|
* }
|
||
|
*
|
||
|
* \param speech The speech engine
|
||
|
* \param format The format codec to use
|
||
|
*
|
||
|
* \returns 0 on success, -1 on error
|
||
|
*/
|
||
|
static int speech_aeap_engine_create(struct ast_speech *speech, struct ast_format *format)
|
||
|
{
|
||
|
struct ast_aeap *aeap;
|
||
|
struct ast_variable *vars;
|
||
|
struct ast_json *json;
|
||
|
|
||
|
aeap = ast_aeap_create_and_connect_by_id(
|
||
|
speech->engine->name, &speech_aeap_params, CONNECTION_TIMEOUT);
|
||
|
if (!aeap) {
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
speech->data = aeap;
|
||
|
|
||
|
/* Don't allow unloading of this module while an external application is in use */
|
||
|
ast_module_ref(ast_module_info->self);
|
||
|
|
||
|
vars = ast_aeap_custom_fields_get(speech->engine->name);
|
||
|
|
||
|
/* While the protocol allows sending of codec attributes, for now don't */
|
||
|
json = ast_json_pack("{s:s,s:[{s:s}],s:o*}", "version", SPEECH_AEAP_VERSION, "codecs",
|
||
|
"name", ast_format_get_codec_name(format), "params", custom_fields_to_params(vars));
|
||
|
|
||
|
ast_variables_destroy(vars);
|
||
|
|
||
|
if (ast_aeap_user_data_register(aeap, "speech", speech, NULL)) {
|
||
|
ast_module_unref(ast_module_info->self);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* send_request handles json ref */
|
||
|
if (speech_aeap_send_request(speech->data, "setup", json, format)) {
|
||
|
ast_module_unref(ast_module_info->self);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Add a reference to the engine here, so if it happens to get unregistered
|
||
|
* while executing it won't disappear.
|
||
|
*/
|
||
|
ao2_ref(speech->engine, 1);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_destroy(struct ast_speech *speech)
|
||
|
{
|
||
|
ao2_ref(speech->engine, -1);
|
||
|
ao2_cleanup(speech->data);
|
||
|
|
||
|
ast_module_unref(ast_module_info->self);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_write(struct ast_speech *speech, void *data, int len)
|
||
|
{
|
||
|
return ast_aeap_send_binary(speech->data, data, len);
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_dtmf(struct ast_speech *speech, const char *dtmf)
|
||
|
{
|
||
|
return speech_aeap_set(speech, "dtmf", dtmf);
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_start(struct ast_speech *speech)
|
||
|
{
|
||
|
ast_speech_change_state(speech, AST_SPEECH_STATE_READY);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_change(struct ast_speech *speech, const char *name, const char *value)
|
||
|
{
|
||
|
return speech_aeap_set(speech, name, value);
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_get_setting(struct ast_speech *speech, const char *name,
|
||
|
char *buf, size_t len)
|
||
|
{
|
||
|
struct speech_setting setting = {
|
||
|
.param = name,
|
||
|
.len = len,
|
||
|
.buf = buf,
|
||
|
};
|
||
|
|
||
|
return speech_aeap_get(speech, name, &setting);
|
||
|
}
|
||
|
|
||
|
static int speech_aeap_engine_change_results_type(struct ast_speech *speech,
|
||
|
enum ast_speech_results_type results_type)
|
||
|
{
|
||
|
return speech_aeap_set(speech, "results_type",
|
||
|
ast_speech_results_type_to_string(results_type));
|
||
|
}
|
||
|
|
||
|
static struct ast_speech_result *speech_aeap_engine_get(struct ast_speech *speech)
|
||
|
{
|
||
|
struct ast_speech_result *results = NULL;
|
||
|
|
||
|
if (speech->results) {
|
||
|
return speech->results;
|
||
|
}
|
||
|
|
||
|
if (speech_aeap_get(speech, "results", &results)) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
return results;
|
||
|
}
|
||
|
|
||
|
static void speech_engine_destroy(void *obj)
|
||
|
{
|
||
|
struct ast_speech_engine *engine = obj;
|
||
|
|
||
|
ao2_cleanup(engine->formats);
|
||
|
ast_free(engine->name);
|
||
|
}
|
||
|
|
||
|
static struct ast_speech_engine *speech_engine_alloc(const char *name)
|
||
|
{
|
||
|
struct ast_speech_engine *engine;
|
||
|
|
||
|
engine = ao2_t_alloc_options(sizeof(*engine), speech_engine_destroy,
|
||
|
AO2_ALLOC_OPT_LOCK_NOLOCK, name);
|
||
|
if (!engine) {
|
||
|
ast_log(LOG_ERROR, "AEAP speech: unable create engine '%s'\n", name);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
engine->name = ast_strdup(name);
|
||
|
if (!engine->name) {
|
||
|
ao2_ref(engine, -1);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
engine->create = speech_aeap_engine_create;
|
||
|
engine->destroy = speech_aeap_engine_destroy;
|
||
|
engine->write = speech_aeap_engine_write;
|
||
|
engine->dtmf = speech_aeap_engine_dtmf;
|
||
|
engine->start = speech_aeap_engine_start;
|
||
|
engine->change = speech_aeap_engine_change;
|
||
|
engine->get_setting = speech_aeap_engine_get_setting;
|
||
|
engine->change_results_type = speech_aeap_engine_change_results_type;
|
||
|
engine->get = speech_aeap_engine_get;
|
||
|
|
||
|
engine->formats = ast_format_cap_alloc(AST_FORMAT_CAP_FLAG_DEFAULT);
|
||
|
|
||
|
return engine;
|
||
|
}
|
||
|
|
||
|
static void speech_engine_alloc_and_register(const char *name, const struct ast_format_cap *formats)
|
||
|
{
|
||
|
struct ast_speech_engine *engine;
|
||
|
|
||
|
engine = speech_engine_alloc(name);
|
||
|
if (!engine) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (formats && ast_format_cap_append_from_cap(engine->formats,
|
||
|
formats, AST_MEDIA_TYPE_AUDIO)) {
|
||
|
ast_log(LOG_WARNING, "AEAP speech: Unable to add engine '%s' formats\n", name);
|
||
|
ao2_ref(engine, -1);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (ast_speech_register(engine)) {
|
||
|
ast_log(LOG_WARNING, "AEAP speech: Unable to register engine '%s'\n", name);
|
||
|
ao2_ref(engine, -1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef TEST_FRAMEWORK
|
||
|
|
||
|
static void speech_engine_alloc_and_register2(const char *name, const char *codec_names)
|
||
|
{
|
||
|
struct ast_speech_engine *engine;
|
||
|
|
||
|
engine = speech_engine_alloc(name);
|
||
|
if (!engine) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (codec_names && ast_format_cap_update_by_allow_disallow(engine->formats, codec_names, 1)) {
|
||
|
ast_log(LOG_WARNING, "AEAP speech: Unable to add engine '%s' codecs\n", name);
|
||
|
ao2_ref(engine, -1);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (ast_speech_register(engine)) {
|
||
|
ast_log(LOG_WARNING, "AEAP speech: Unable to register engine '%s'\n", name);
|
||
|
ao2_ref(engine, -1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
static int unload_engine(void *obj, void *arg, int flags)
|
||
|
{
|
||
|
if (ast_aeap_client_config_has_protocol(obj, SPEECH_PROTOCOL)) {
|
||
|
ao2_cleanup(ast_speech_unregister2(ast_sorcery_object_get_id(obj)));
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int load_engine(void *obj, void *arg, int flags)
|
||
|
{
|
||
|
const char *id;
|
||
|
const struct ast_format_cap *formats;
|
||
|
const struct ast_speech_engine *engine;
|
||
|
|
||
|
if (!ast_aeap_client_config_has_protocol(obj, SPEECH_PROTOCOL)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
id = ast_sorcery_object_get_id(obj);
|
||
|
formats = ast_aeap_client_config_codecs(obj);
|
||
|
if (!formats) {
|
||
|
formats = ast_format_cap_alloc(AST_FORMAT_CAP_FLAG_DEFAULT);
|
||
|
if (!formats) {
|
||
|
ast_log(LOG_ERROR, "AEAP speech: unable to allocate default engine format for '%s'\n", id);
|
||
|
return 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
engine = ast_speech_find_engine(id);
|
||
|
if (!engine) {
|
||
|
speech_engine_alloc_and_register(id, formats);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if (ast_format_cap_identical(formats, engine->formats)) {
|
||
|
/* Same name, same formats then nothing changed */
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
ao2_ref(ast_speech_unregister2(engine->name), -1);
|
||
|
speech_engine_alloc_and_register(id, formats);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int matches_engine(void *obj, void *arg, int flags)
|
||
|
{
|
||
|
const struct ast_speech_engine *engine = arg;
|
||
|
|
||
|
return strcmp(ast_sorcery_object_get_id(obj), engine->name) ? 0 : CMP_MATCH;
|
||
|
}
|
||
|
|
||
|
static int should_unregister(const struct ast_speech_engine *engine, void *data)
|
||
|
{
|
||
|
void *obj;
|
||
|
|
||
|
if (engine->create != speech_aeap_engine_create) {
|
||
|
/* Only want to potentially unregister AEAP speech engines */
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#ifdef TEST_FRAMEWORK
|
||
|
if (!strcmp("_aeap_test_speech_", engine->name)) {
|
||
|
/* Don't remove the test engine */
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
obj = ao2_callback(data, 0, matches_engine, (void*)engine);
|
||
|
|
||
|
if (obj) {
|
||
|
ao2_ref(obj, -1);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* If no match in given container then unregister engine */
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static void speech_observer_loaded(const char *object_type)
|
||
|
{
|
||
|
struct ao2_container *container;
|
||
|
|
||
|
if (strcmp(object_type, AEAP_CONFIG_CLIENT)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
container = ast_aeap_client_configs_get(SPEECH_PROTOCOL);
|
||
|
if (!container) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* An AEAP module reload has occurred. First
|
||
|
* remove all engines that no longer exist.
|
||
|
*/
|
||
|
ast_speech_unregister_engines(should_unregister, container, __ao2_cleanup);
|
||
|
|
||
|
/* Now add or update engines */
|
||
|
ao2_callback(container, 0, load_engine, NULL);
|
||
|
ao2_ref(container, -1);
|
||
|
}
|
||
|
|
||
|
/*! \brief Observer for AEAP reloads */
|
||
|
static const struct ast_sorcery_observer speech_observer = {
|
||
|
.loaded = speech_observer_loaded,
|
||
|
};
|
||
|
|
||
|
static int unload_module(void)
|
||
|
{
|
||
|
struct ao2_container *container;
|
||
|
|
||
|
#ifdef TEST_FRAMEWORK
|
||
|
ao2_cleanup(ast_speech_unregister2("_aeap_test_speech_"));
|
||
|
#endif
|
||
|
|
||
|
ast_sorcery_observer_remove(ast_aeap_sorcery(), AEAP_CONFIG_CLIENT, &speech_observer);
|
||
|
|
||
|
container = ast_aeap_client_configs_get(SPEECH_PROTOCOL);
|
||
|
if (container) {
|
||
|
ao2_callback(container, 0, unload_engine, NULL);
|
||
|
ao2_ref(container, -1);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int load_module(void)
|
||
|
{
|
||
|
struct ao2_container *container;
|
||
|
|
||
|
speech_aeap_params.msg_type = ast_aeap_message_type_json;
|
||
|
|
||
|
container = ast_aeap_client_configs_get(SPEECH_PROTOCOL);
|
||
|
if (container) {
|
||
|
ao2_callback(container, 0, load_engine, NULL);
|
||
|
ao2_ref(container, -1);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Add an observer since a named speech server must be created,
|
||
|
* registered, and eventually removed for all AEAP client
|
||
|
* configuration matching the "speech_to_text" protocol.
|
||
|
*/
|
||
|
if (ast_sorcery_observer_add(ast_aeap_sorcery(), AEAP_CONFIG_CLIENT, &speech_observer)) {
|
||
|
return AST_MODULE_LOAD_DECLINE;
|
||
|
}
|
||
|
|
||
|
#ifdef TEST_FRAMEWORK
|
||
|
speech_engine_alloc_and_register2("_aeap_test_speech_", "ulaw");
|
||
|
#endif
|
||
|
|
||
|
return AST_MODULE_LOAD_SUCCESS;
|
||
|
}
|
||
|
|
||
|
AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "Asterisk External Application Speech Engine",
|
||
|
.support_level = AST_MODULE_SUPPORT_CORE,
|
||
|
.load = load_module,
|
||
|
.unload = unload_module,
|
||
|
.load_pri = AST_MODPRI_CHANNEL_DEPEND,
|
||
|
.requires = "res_speech,res_aeap",
|
||
|
);
|