[asterisk-commits] tilghman: branch group/xmlrpc r244497 - /team/group/xmlrpc/funcs/
SVN commits to the Asterisk project
asterisk-commits at lists.digium.com
Tue Feb 2 19:40:57 CST 2010
Author: tilghman
Date: Tue Feb 2 19:40:54 2010
New Revision: 244497
URL: http://svnview.digium.com/svn/asterisk?view=rev&rev=244497
Log:
Move function into func_curl, for use of the curl library
Removed:
team/group/xmlrpc/funcs/func_xmlrpc.c
Modified:
team/group/xmlrpc/funcs/func_curl.c
Modified: team/group/xmlrpc/funcs/func_curl.c
URL: http://svnview.digium.com/svn/asterisk/team/group/xmlrpc/funcs/func_curl.c?view=diff&rev=244497&r1=244496&r2=244497
==============================================================================
--- team/group/xmlrpc/funcs/func_curl.c (original)
+++ team/group/xmlrpc/funcs/func_curl.c Tue Feb 2 19:40:54 2010
@@ -1,7 +1,7 @@
/*
* Asterisk -- An open source telephony toolkit.
*
- * Copyright (C) 2004 - 2006, Tilghman Lesher
+ * Copyright (C) 2004 - 2010, Tilghman Lesher
*
* Tilghman Lesher <curl-20050919 at the-tilghman.com>
* and Brian Wilkins <bwilkins at cfl.rr.com> (Added POST option)
@@ -55,11 +55,25 @@
#define CURLOPT_SPECIAL_HASHCOMPAT -500
+/* Constants */
+static const char *xmlrpc_config = "func_xmlrpc.conf";
+static const char * const global_useragent = "asterisk-libcurl-agent/1.0";
+
+/* Prototypes */
static void curlds_free(void *data);
-
-static struct ast_datastore_info curl_info = {
- .type = "CURL",
- .destroy = curlds_free,
+static int curl_instance_init(void *data);
+static void curl_instance_cleanup(void *data);
+static int xmlrpc_instance_init(void *data);
+
+/* Structures */
+struct acf_xmlrpc_query {
+ AST_RWLIST_ENTRY(acf_xmlrpc_query) list;
+ AST_DECLARE_STRING_FIELDS(
+ AST_STRING_FIELD(url);
+ AST_STRING_FIELD(method);
+ AST_STRING_FIELD(params);
+ );
+ struct ast_custom_function acf;
};
struct curl_settings {
@@ -67,21 +81,6 @@
CURLoption key;
void *value;
};
-
-AST_LIST_HEAD_STATIC(global_curl_info, curl_settings);
-
-static void curlds_free(void *data)
-{
- AST_LIST_HEAD(global_curl_info, curl_settings) *list = data;
- struct curl_settings *setting;
- if (!list) {
- return;
- }
- while ((setting = AST_LIST_REMOVE_HEAD(list, list))) {
- free(setting);
- }
- AST_LIST_HEAD_DESTROY(list);
-}
enum optiontype {
OT_BOOLEAN,
@@ -90,6 +89,34 @@
OT_STRING,
OT_ENUM,
};
+
+/* Static storage */
+AST_LIST_HEAD_STATIC(global_curl_info, curl_settings);
+static AST_RWLIST_HEAD_STATIC(queries, acf_xmlrpc_query);
+
+AST_THREADSTORAGE(query_buf);
+AST_THREADSTORAGE(value_buf);
+AST_THREADSTORAGE_CUSTOM(curl_instance, curl_instance_init, curl_instance_cleanup);
+AST_THREADSTORAGE_CUSTOM(xmlrpc_instance, xmlrpc_instance_init, curl_instance_cleanup);
+
+static struct ast_datastore_info curl_info = {
+ .type = "CURL",
+ .destroy = curlds_free,
+};
+
+/* Function definitions */
+static void curlds_free(void *data)
+{
+ AST_LIST_HEAD(global_curl_info, curl_settings) *list = data;
+ struct curl_settings *setting;
+ if (!list) {
+ return;
+ }
+ while ((setting = AST_LIST_REMOVE_HEAD(list, list))) {
+ free(setting);
+ }
+ AST_LIST_HEAD_DESTROY(list);
+}
static int parse_curlopt_key(const char *name, CURLoption *key, enum optiontype *ot)
{
@@ -416,8 +443,6 @@
return realsize;
}
-static const char * const global_useragent = "asterisk-libcurl-agent/1.0";
-
static int curl_instance_init(void *data)
{
CURL **curl = data;
@@ -442,7 +467,20 @@
ast_free(data);
}
-AST_THREADSTORAGE_CUSTOM(curl_instance, curl_instance_init, curl_instance_cleanup);
+static int xmlrpc_instance_init(void *data)
+{
+ CURL **curl = data;
+
+ if (!(*curl = curl_easy_init()))
+ return -1;
+
+ curl_easy_setopt(*curl, CURLOPT_NOSIGNAL, 1);
+ curl_easy_setopt(*curl, CURLOPT_TIMEOUT, 5);
+ curl_easy_setopt(*curl, CURLOPT_USERAGENT, global_useragent);
+ curl_easy_setopt(*curl, CURLOPT_POST, 1);
+
+ return 0;
+}
static int acf_curl_helper(struct ast_channel *chan, const char *cmd, char *info, char *buf, struct ast_str **input_str, ssize_t len)
{
@@ -611,19 +649,643 @@
.write = acf_curlopt_write,
};
+static char *encode_xml(struct ast_str *result, int maxlen, const char *input)
+{
+ const char *c;
+ ast_str_reset(result);
+ for (c = input; c && *c; c++) {
+ const char *amp = strchr(c, '&');
+ const char *lt = strchr(c, '<');
+ if ((amp && lt && amp < lt) || (amp && !lt)) {
+ /* Copy up to the first ampersand */
+ ast_str_append_substr(&result, maxlen, c, amp - c);
+ ast_str_append_substr(&result, maxlen, "&", 5);
+ c = amp;
+ } else if ((amp && lt && lt < amp) || (lt && !amp)) {
+ /* Copy up to the first less-than-sign */
+ ast_str_append_substr(&result, maxlen, c, lt - c);
+ ast_str_append_substr(&result, maxlen, "<", 4);
+ c = lt;
+ } else {
+ ast_str_append_substr(&result, maxlen, c, strlen(c));
+ break;
+ }
+ }
+ return ast_str_buffer(result);
+}
+
+static int acf_xmlrpc_read(struct ast_channel *chan, const char *cmd, char *s, struct ast_str *buf, int len)
+{
+ struct acf_xmlrpc_query *query;
+ char varname[15], rowcount[12] = "-1", *template = "/tmp/xmlrpc-XXXXXX";
+ int res, buflen = 0, bogus_chan = 0;
+ AST_DECLARE_APP_ARGS(args,
+ AST_APP_ARG(field)[100];
+ );
+ struct ast_str *qbuf = ast_str_thread_get(&query_buf, 16);
+ struct ast_str *value = ast_str_thread_get(&value_buf, 16);
+ const char *status = "FAILURE";
+ CURL **curl;
+ struct curl_settings *curl_cur;
+ struct ast_datastore *store = NULL;
+ int hashcompat = 0, fd;
+ FILE *f;
+ AST_LIST_HEAD(global_curl_info, curl_settings) *list = NULL;
+ struct curl_slist *slist = NULL;
+ struct ast_xml_doc *xml;
+ struct ast_xml_node *root, *cur;
+
+ if (!qbuf) {
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ }
+
+ ast_str_reset(colnames);
+
+ AST_RWLIST_RDLOCK(&queries);
+ AST_RWLIST_TRAVERSE(&queries, query, list) {
+ if (!strcmp(query.acf->name, cmd)) {
+ break;
+ }
+ }
+
+ if (!query) {
+ ast_log(LOG_ERROR, "No such function '%s'\n", cmd);
+ AST_RWLIST_UNLOCK(&queries);
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ }
+
+ if (!(curl = ast_threadstorage_get(&xmlrpc_instance, sizeof(*curl)))) {
+ AST_RWLIST_UNLOCK(&queries);
+ ast_log(LOG_ERROR, "Cannot allocate curl structure\n");
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ }
+
+ if (!(slist = curl_slist_append(slist, "Content-Type: text/xml"))) {
+ AST_RWLIST_UNLOCK(&queries);
+ ast_log(LOG_ERROR, "Cannot set content type\n");
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ }
+ if (!(slist2 = curl_slist_append(slist, "Expect:"))) {
+ AST_RWLIST_UNLOCK(&queries);
+ ast_log(LOG_ERROR, "Cannot clear Expect header\n");
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ } else {
+ slist = slist2;
+ }
+
+ if (!(fd = mkstemp(template))) {
+ AST_RWLIST_UNLOCK(&queries);
+ ast_log(LOG_ERROR, "Cannot create temporary file: %s\n", strerror(errno));
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ }
+
+ if (!(f = fdopen(fd, "w+"))) {
+ AST_RWLIST_UNLOCK(&queries);
+ ast_log(LOG_ERROR, "Cannot create file point for temporary file: %s\n", strerror(errno));
+ close(fd);
+ unlink(template);
+ if (chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ }
+ return -1;
+ }
+
+ if (!chan) {
+ if (!(chan = ast_dummy_channel_alloc())) {
+ AST_RWLIST_UNLOCK(&queries);
+ fclose(f);
+ unlink(template);
+ return -1;
+ }
+ bogus_chan = 1;
+ }
+
+ if (!bogus_chan) {
+ ast_autoservice_start(chan);
+ }
+
+ AST_LIST_LOCK(&global_curl_info);
+ AST_LIST_TRAVERSE(&global_curl_info, curl_cur, list) {
+ if (curl_cur->key == CURLOPT_SPECIAL_HASHCOMPAT) {
+ hashcompat = (curl_cur->value != NULL) ? 1 : 0;
+ } else {
+ curl_easy_setopt(*curl, curl_cur->key, curl_cur->value);
+ }
+ }
+
+ if (!bogus_chan && (store = ast_channel_datastore_find(chan, &curl_info, NULL))) {
+ list = store->data;
+ AST_LIST_LOCK(list);
+ AST_LIST_TRAVERSE(list, cur, list) {
+ if (cur->key == CURLOPT_SPECIAL_HASHCOMPAT) {
+ hashcompat = (cur->value != NULL) ? 1 : 0;
+ } else {
+ curl_easy_setopt(*curl, cur->key, cur->value);
+ }
+ }
+ }
+
+ curl_easy_setopt(*curl, CURLOPT_URL, query->url);
+ /* Output XML to our temporary file */
+ curl_easy_setopt(*curl, CURLOPT_FILE, f);
+
+ AST_STANDARD_APP_ARGS(args, s);
+ for (x = 0; x < args.argc; x++) {
+ char *c;
+ snprintf(varname, sizeof(varname), "ARG%d", x + 1);
+ pbx_builtin_pushvar_helper(chan, varname, encode_xml(value, 0, args.field[x]));
+ }
+
+ ast_str_substitute_variables(&qbuf, 0, chan, query->params);
+
+ if (bogus_chan) {
+ chan = ast_channel_release(chan);
+ } else {
+ /* Restore prior values */
+ for (x = 0; x < args.argc; x++) {
+ snprintf(varname, sizeof(varname), "ARG%d", x + 1);
+ pbx_builtin_setvar_helper(chan, varname, NULL);
+ }
+ }
+
+ /* Create final post value */
+ ast_str_set(&value, 0, "<?xml version=\"1.0\"?><methodCall><methodName>%s</methodName><params>%s</params></methodCall>",
+ encode_xml(value, 0, query->method), ast_str_buffer(qbuf));
+ curl_easy_setopt(*curl, CURLOPT_POSTFIELDS, ast_str_buffer(qbuf));
+
+ ast_str_set(&value, 0, "Content-length: %d", ast_str_strlen(qbuf));
+ if (!(slist2 = curl_slist_append(slist, ast_str_buffer(value)))) {
+ ast_log(LOG_WARNING, "Cannot set Content-length\n");
+ } else {
+ slist = slist2;
+ }
+ curl_easy_setopt(*curl, CURLOPT_HTTPHEADER, slist);
+
+ curl_easy_perform(*curl);
+
+ if (store) {
+ AST_LIST_UNLOCK(list);
+ }
+ AST_LIST_UNLOCK(&global_curl_info);
+ AST_RWLIST_UNLOCK(&queries);
+
+ /* Decode XML */
+ if (!(xml = ast_xml_open(template))) {
+ goto end_acf_read:
+ }
+ if (!(root = ast_xml_get_root(xml))) {
+ goto end_acf_read;
+ }
+ if (!(cur = ast_xml_find_element(root, "params", NULL, NULL))) {
+ /* No results */
+ if (!(cur = ast_xml_find_element(root, "fault", NULL, NULL))) {
+ /* TODO Retrieve fault */
+ }
+ goto end_acf_read;
+ }
+
+ /* TODO Retrieve values and encode for output */
+
+end_acf_read:
+ if (xml) {
+ ast_xml_close(xml);
+ }
+ fclose(f);
+ unlink(template);
+
+ if (!bogus_chan) {
+ pbx_builtin_setvar_helper(chan, "XMLRPCSTATUS", status);
+ pbx_builtin_setvar_helper(chan, "~ODBCFIELDS~", ast_str_buffer(colnames));
+ ast_autoservice_stop(chan);
+ }
+
+ curl_slist_free_all(slist);
+ return 0;
+}
+
+static int init_xmlrpc_query(struct ast_config *cfg, char *catg, struct acf_xmlrpc_query **query)
+{
+ const char *tmp;
+ int i;
+
+ if (!cfg || !catg) {
+ return EINVAL;
+ }
+
+ if (!(*query = ast_calloc(1, sizeof(struct acf_xmlrpc_query)))) {
+ return ENOMEM;
+ }
+
+ if (ast_string_field_init((*query), 128)) {
+ ast_free(*query);
+ *query = NULL;
+ return ENOMEM;
+ }
+
+ if (((tmp = ast_variable_retrieve(cfg, catg, "url")))) {
+ ast_string_field_set((*query), url, tmp);
+ } else {
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return EINVAL;
+ }
+
+ if ((tmp = ast_variable_retrieve(cfg, catg, "method"))) {
+ ast_string_field_set((*query), method, tmp);
+ } else {
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return EINVAL;
+ }
+
+ if ((tmp = ast_variable_retrieve(cfg, catg, "params")))
+ ast_string_field_set((*query), params, tmp);
+ } else {
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return EINVAL;
+ }
+
+ if (ast_string_field_init((*query).acf, 128)) {
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return ENOMEM;
+ }
+
+ if (!(tmp = ast_variable_retrieve(cfg, catg, "prefix")) || ast_strlen_zero(tmp)) {
+ tmp = "XMLRPC";
+ if (ast_string_field_build((*query).acf, name, "%s_%s", tmp, catg)) {
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return ENOMEM;
+ }
+ }
+
+ if ((tmp = ast_variable_retrieve(cfg, catg, "syntax")) && !ast_strlen_zero(tmp)) {
+ ast_string_field_build((*query).acf, syntax, "%s(%s)", (*query).acf->name, tmp);
+ } else {
+ ast_string_field_build((*query).acf, syntax, "%s(<arg1>[...[,<argN>]])", (*query).acf->name);
+ }
+
+ if (ast_strlen_zero((*query).acf->syntax)) {
+ ast_string_field_free_memory((*query).acf);
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return ENOMEM;
+ }
+
+ if (!(tmp = ast_variable_retrieve(cfg, catg, "synopsis")) || ast_strlen_zero(tmp)) {
+ tmp = "Runs the referenced query with the specified arguments";
+ ast_string_field_set((*query)->acf, synopsis, tmp);
+ }
+
+ if (ast_strlen_zero((*query)->acf->synopsis)) {
+ ast_string_field_free_memory((*query).acf);
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return ENOMEM;
+ }
+
+ ast_string_field_build((*query)->acf, desc,
+ "Runs the following query, as defined in func_xmlrpc.conf, performing\n"
+ "substitution of the arguments into the query as specified by ${ARG1},\n"
+ "${ARG2}, ... ${ARGn}. When setting the function, the values are provided\n"
+ "either in whole as ${VALUE} or parsed as ${VAL1}, ${VAL2}, ... ${VALn}.\n"
+ "\nRead: %s\n%s\n",
+ (*query)->method,
+ (*query)->url);
+
+ if (ast_strlen_zero((*query).acf->desc)) {
+ ast_string_field_free_memory((*query).acf);
+ ast_string_field_free_memory(*query);
+ ast_free(*query);
+ *query = NULL;
+ return ENOMEM;
+ }
+
+ (*query).acf->read = acf_xmlrpc_read;
+
+ return 0;
+}
+
+static int free_acf_query(struct acf_xmlrpc_query *query)
+{
+ if (query) {
+ if (query->acf) {
+ ast_string_field_free_memory(query->acf);
+ ast_string_field_free_memory(query);
+ }
+ ast_free(query);
+ }
+ return 0;
+}
+
+static char *cli_xmlrpc_read(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+ AST_DECLARE_APP_ARGS(args,
+ AST_APP_ARG(field)[100];
+ );
+ struct ast_str *query;
+ char *char_args, varname[10];
+ struct acf_xmlrpc_query *query;
+ struct ast_channel *chan;
+ int i;
+
+ switch (cmd) {
+ case CLI_INIT:
+ e->command = "xmlrpc read";
+ e->usage =
+ "Usage: xmlrpc read <name> <args> [exec]\n"
+ " Evaluates the SQL provided in the ODBC function <name>, and\n"
+ " optionally executes the function. This function is intended for\n"
+ " testing purposes. Remember to quote arguments containing spaces.\n";
+ return NULL;
+ case CLI_GENERATE:
+ if (a->pos == 2) {
+ int wordlen = strlen(a->word), which = 0;
+ /* Complete function name */
+ AST_RWLIST_RDLOCK(&queries);
+ AST_RWLIST_TRAVERSE(&queries, query, list) {
+ if (!strncasecmp(query->acf->name, a->word, wordlen)) {
+ if (++which > a->n) {
+ char *res = ast_strdup(query->acf->name);
+ AST_RWLIST_UNLOCK(&queries);
+ return res;
+ }
+ }
+ }
+ AST_RWLIST_UNLOCK(&queries);
+ return NULL;
+ } else if (a->pos == 4) {
+ return a->n == 0 ? ast_strdup("exec") : NULL;
+ } else {
+ return NULL;
+ }
+ }
+
+ if (a->argc < 4 || a->argc > 5) {
+ return CLI_SHOWUSAGE;
+ }
+
+ query = ast_str_thread_get(&query_buf, 16);
+ if (!query) {
+ return CLI_FAILURE;
+ }
+
+ AST_RWLIST_RDLOCK(&queries);
+ AST_RWLIST_TRAVERSE(&queries, query, list) {
+ if (!strcmp(query->acf->name, a->argv[2])) {
+ break;
+ }
+ }
+
+ if (!query) {
+ ast_cli(a->fd, "No such query '%s'\n", a->argv[2]);
+ AST_RWLIST_UNLOCK(&queries);
+ return CLI_SHOWUSAGE;
+ }
+
+ if (ast_strlen_zero(query->params)) {
+ ast_cli(a->fd, "The function %s has no writequery parameter.\n", a->argv[2]);
+ AST_RWLIST_UNLOCK(&queries);
+ return CLI_SUCCESS;
+ }
+
+ ast_str_make_space(&query, strlen(query->params) * 2 + 300);
+
+ /* Evaluate function */
+ char_args = ast_strdupa(a->argv[3]);
+
+ chan = ast_dummy_channel_alloc();
+
+ AST_STANDARD_APP_ARGS(args, char_args);
+ for (i = 0; i < args.argc; i++) {
+ snprintf(varname, sizeof(varname), "ARG%d", i + 1);
+ pbx_builtin_pushvar_helper(chan, varname, args.field[i]);
+ }
+
+ ast_str_substitute_variables(&query, 0, chan, query->params);
+ chan = ast_channel_release(chan);
+
+ if (a->argc == 5 && !strcmp(a->argv[4], "exec")) {
+ /* Execute the query */
+ struct xmlrpc_obj *obj = NULL;
+ int dsn, executed = 0;
+ SQLHSTMT stmt;
+ int rows = 0, res, x;
+ SQLSMALLINT colcount = 0, collength;
+ SQLLEN indicator;
+ struct ast_str *coldata = ast_str_thread_get(&coldata_buf, 16);
+ char colname[256];
+ SQLULEN maxcol;
+
+ for (dsn = 0; dsn < 5; dsn++) {
+ if (ast_strlen_zero(query->readhandle[dsn])) {
+ continue;
+ }
+ ast_debug(1, "Found handle %s\n", query->readhandle[dsn]);
+ if (!(obj = ast_xmlrpc_request_obj(query->readhandle[dsn], 0))) {
+ continue;
+ }
+
+ ast_debug(1, "Got obj\n");
+ if (!(stmt = ast_xmlrpc_direct_execute(obj, generic_execute, ast_str_buffer(query)))) {
+ ast_xmlrpc_release_obj(obj);
+ obj = NULL;
+ continue;
+ }
+
+ executed = 1;
+
+ res = SQLNumResultCols(stmt, &colcount);
+ if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
+ ast_cli(a->fd, "SQL Column Count error!\n[%s]\n\n", ast_str_buffer(query));
+ SQLCloseCursor(stmt);
+ SQLFreeHandle (SQL_HANDLE_STMT, stmt);
+ ast_xmlrpc_release_obj(obj);
+ obj = NULL;
+ AST_RWLIST_UNLOCK(&queries);
+ return CLI_SUCCESS;
+ }
+
+ res = SQLFetch(stmt);
+ if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
+ SQLCloseCursor(stmt);
+ SQLFreeHandle(SQL_HANDLE_STMT, stmt);
+ ast_xmlrpc_release_obj(obj);
+ obj = NULL;
+ if (res == SQL_NO_DATA) {
+ ast_cli(a->fd, "Returned %d rows. Query executed on handle %d:%s [%s]\n", rows, dsn, query->readhandle[dsn], ast_str_buffer(query));
+ break;
+ } else {
+ ast_cli(a->fd, "Error %d in FETCH [%s]\n", res, ast_str_buffer(query));
+ }
+ AST_RWLIST_UNLOCK(&queries);
+ return CLI_SUCCESS;
+ }
+ for (;;) {
+ for (x = 0; x < colcount; x++) {
+ res = SQLDescribeCol(stmt, x + 1, (unsigned char *)colname, sizeof(colname), &collength, NULL, &maxcol, NULL, NULL);
+ if (((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) || collength == 0) {
+ snprintf(colname, sizeof(colname), "field%d", x);
+ }
+
+ res = ast_xmlrpc_ast_str_SQLGetData(&coldata, maxcol, stmt, x + 1, SQL_CHAR, &indicator);
+ if (indicator == SQL_NULL_DATA) {
+ ast_str_set(&coldata, 0, "(nil)");
+ res = SQL_SUCCESS;
+ }
+
+ if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
+ ast_cli(a->fd, "SQL Get Data error %d!\n[%s]\n\n", res, ast_str_buffer(query));
+ SQLCloseCursor(stmt);
+ SQLFreeHandle(SQL_HANDLE_STMT, stmt);
+ ast_xmlrpc_release_obj(obj);
+ obj = NULL;
+ AST_RWLIST_UNLOCK(&queries);
+ return CLI_SUCCESS;
+ }
+
+ ast_cli(a->fd, "%-20.20s %s\n", colname, ast_str_buffer(coldata));
+ }
+ rows++;
+
+ /* Get next row */
+ res = SQLFetch(stmt);
+ if ((res != SQL_SUCCESS) && (res != SQL_SUCCESS_WITH_INFO)) {
+ break;
+ }
+ ast_cli(a->fd, "%-20.20s %s\n", "----------", "----------");
+ }
+ SQLCloseCursor(stmt);
+ SQLFreeHandle(SQL_HANDLE_STMT, stmt);
+ ast_xmlrpc_release_obj(obj);
+ obj = NULL;
+ ast_cli(a->fd, "Returned %d row%s. Query executed on handle %d [%s]\n", rows, rows == 1 ? "" : "s", dsn, query->readhandle[dsn]);
+ break;
+ }
+ if (obj) {
+ ast_xmlrpc_release_obj(obj);
+ obj = NULL;
+ }
+
+ if (!executed) {
+ ast_cli(a->fd, "Failed to execute query. [%s]\n", ast_str_buffer(query));
+ }
+ } else { /* No execution, just print out the resulting SQL */
+ ast_cli(a->fd, "%s\n", ast_str_buffer(query));
+ }
+ AST_RWLIST_UNLOCK(&queries);
+ return CLI_SUCCESS;
+}
+
+static struct ast_cli_entry cli_func_xmlrpc[] = {
+ AST_CLI_DEFINE(cli_xmlrpc_read, "Test reading a func_xmlrpc function"),
+};
+
+static int reload(void)
+{
+ int res = 0;
+ struct ast_config *cfg;
+ struct acf_xmlrpc_query *oldquery;
+ char *catg;
+ struct ast_flags config_flags = { CONFIG_FLAG_FILEUNCHANGED };
+
+ cfg = ast_config_load(xmlrpc_config, config_flags);
+ if (cfg == CONFIG_STATUS_FILEUNCHANGED || cfg == CONFIG_STATUS_FILEINVALID)
+ return 0;
+
+ AST_RWLIST_WRLOCK(&queries);
+
+ while (!AST_RWLIST_EMPTY(&queries)) {
+ oldquery = AST_RWLIST_REMOVE_HEAD(&queries, list);
+ ast_custom_function_unregister(oldquery->acf);
+ free_acf_query(oldquery);
+ }
+
+ if (!cfg) {
+ ast_log(LOG_WARNING, "Unable to load config for func_xmlrpc: %s\n", xmlrpc_config);
+ goto reload_out;
+ }
+
+ for (catg = ast_category_browse(cfg, NULL);
+ catg;
+ catg = ast_category_browse(cfg, catg)) {
+ struct acf_xmlrpc_query *query = NULL;
+
+ if (init_xmlrpc_query(cfg, catg, &query)) {
+ ast_log(LOG_ERROR, "Cannot initialize query %s\n", catg);
+ } else {
+ AST_RWLIST_INSERT_HEAD(&queries, query, list);
+ ast_custom_function_register(query->acf);
+ }
+ }
+
+ ast_config_destroy(cfg);
+reload_out:
+ AST_RWLIST_UNLOCK(&queries);
+ return res;
+}
+
+
static int unload_module(void)
{
int res;
-
+ struct acf_xmlrpc_query *query;
+
+ AST_RWLIST_WRLOCK(&queries);
+ while (!AST_RWLIST_EMPTY(&queries)) {
+ query = AST_RWLIST_REMOVE_HEAD(&queries, list);
+ ast_custom_function_unregister(query->acf);
+ free_acf_query(query);
+ }
+
+ ast_cli_unregister_multiple(cli_func_xmlrpc, ARRAY_LEN(cli_func_xmlrpc));
res = ast_custom_function_unregister(&acf_curl);
res |= ast_custom_function_unregister(&acf_curlopt);
- return res;
+ /* Allow any threads waiting for this lock to pass (avoids a race) */
+ AST_RWLIST_UNLOCK(&queries);
+ usleep(1);
+ AST_RWLIST_WRLOCK(&queries);
+
+ AST_RWLIST_UNLOCK(&queries);
+ return 0;
}
static int load_module(void)
{
int res;
+ struct ast_config *cfg;
+ char *catg;
+ struct ast_flags config_flags = { 0 };
+
+ AST_RWLIST_WRLOCK(&queries);
if (!ast_module_check("res_curl.so")) {
if (ast_load_resource("res_curl.so") != AST_MODULE_LOAD_SUCCESS) {
@@ -632,11 +1294,48 @@
}
}
+ cfg = ast_config_load(xmlrpc_config, config_flags);
+ if (!cfg || cfg == CONFIG_STATUS_FILEINVALID) {
+ ast_log(LOG_NOTICE, "Unable to load config for func_xmlrpc: %s\n", xmlrpc_config);
+ AST_RWLIST_UNLOCK(&queries);
+ return AST_MODULE_LOAD_DECLINE;
+ }
+
+ for (catg = ast_category_browse(cfg, NULL);
+ catg;
+ catg = ast_category_browse(cfg, catg)) {
+ struct acf_xmlrpc_query *query = NULL;
+ int err;
+
+ if ((err = init_xmlrpc_query(cfg, catg, &query))) {
+ if (err == ENOMEM) {
+ ast_log(LOG_ERROR, "Out of memory\n");
+ } else if (err == EINVAL) {
+ ast_log(LOG_ERROR, "Invalid parameters for category %s\n", catg);
+ } else {
+ ast_log(LOG_ERROR, "%s (%d)\n", strerror(err), err);
+ }
+ } else {
+ AST_RWLIST_INSERT_HEAD(&queries, query, list);
+ ast_custom_function_register(query->acf);
+ }
+ }
+
+ ast_config_destroy(cfg);
+
+ AST_RWLIST_UNLOCK(&queries);
+
res = ast_custom_function_register(&acf_curl);
res |= ast_custom_function_register(&acf_curlopt);
+ ast_cli_register_multiple(cli_func_xmlrpc, ARRAY_LEN(cli_func_xmlrpc));
return res;
}
-AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Load external URL");
-
+AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_DEFAULT, "External URL/XMLRPC lookups",
+ .load = load_module,
+ .unload = unload_module,
+ .reload = reload,
+ );
+
+
More information about the asterisk-commits
mailing list