/**************************************************************************** * apps/netutils/xmlrpc/xmlparser.c * * Copyright (C) 2012 Max Holtzberg. All rights reserved. * Author: Max Holtzberg * * Based on the embeddable lightweight XML-RPC server code discussed * in the article at: http://www.drdobbs.com/web-development/\ * an-embeddable-lightweight-xml-rpc-server/184405364 * * Copyright (c) 2002 Cogito LLC. All rights reserved. * * Redistribution and use in source and binary forms, with or * without modification, is hereby granted without fee provided * that the following conditions are met: * * 1. Redistributions of source code must retain the above * copyright notice, this list of conditions and the * following disclaimer. * 2. 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. * 3. Neither the name of Cogito LLC nor the names of its * contributors may be used to endorse or promote products * derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY COGITO LLC 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 COGITO LLC * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARAY, 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. ****************************************************************************/ /* Lightweight Embedded XML-RPC Server XML Parser * * mtj@cogitollc.com * */ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include "netutils/xmlrpc.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define TAG 0 #define VALUE 1 #define DONE 2 /**************************************************************************** * Private Data ****************************************************************************/ static struct xmlrpc_s g_xmlcall; static char g_data[CONFIG_XMLRPC_STRINGSIZE + 1]; static struct xmlrpc_entry_s *g_entries = NULL; static const char *g_error_strings[] = { "Internal error (unknown)", "Parse Error...", "Function not found...", "Unexpected Integer Argument...", "Unexpected Boolean Argument...", "Unexpected Double Argument...", "Unexpected String Argument...", "Bad Response Argument..." }; #define MAX_ERROR_CODE (sizeof(g_error_strings)/sizeof(char *)) struct parsebuf_s { char *buf; int len; int index; }; /**************************************************************************** * Private Functions ****************************************************************************/ static int xmlrpc_call(struct xmlrpc_s *call) { int ret = XMLRPC_NO_SUCH_FUNCTION; struct xmlrpc_entry_s *entry = g_entries; while (entry != NULL) { if (strcmp(call->name, entry->name) == 0) { ret = entry->func(call); break; } else { entry = entry->next; } } return ret; } static int xmlrpc_getelement(struct parsebuf_s *pbuf, char *data, int size) { int j = 0; int ret = XMLRPC_NO_ERROR; while (pbuf->index < pbuf->len && !isprint(pbuf->buf[pbuf->index])) { pbuf->index++; } if (pbuf->index >= pbuf->len) { return DONE; } if (pbuf->buf[pbuf->index] == '<') { ret = TAG; } else { ret = VALUE; } data[j++] = pbuf->buf[pbuf->index++]; while (j < size) { if (pbuf->buf[pbuf->index] == '>') { data[j++] = pbuf->buf[pbuf->index++]; break; } else if ((pbuf->buf[pbuf->index] == '\n') || (pbuf->buf[pbuf->index] == '<')) { break; } else { data[j++] = pbuf->buf[pbuf->index++]; if (j >= size) { ret = XMLRPC_PARSE_ERROR; } } } data[j] = 0; return ret; } static int xmlrpc_parseparam(struct parsebuf_s *pbuf) { int type; /* Next, we need a tag */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (!((type == TAG) && (!strncmp(g_data, "", 7)))) { return XMLRPC_PARSE_ERROR; } /* Now we get a variable tag, the type of the value */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (type != TAG) { return XMLRPC_PARSE_ERROR; } if (!strncmp(g_data, "", 4)) { g_xmlcall.args[g_xmlcall.argsize] = 'i'; } else if (!strncmp(g_data, "", 5)) { g_xmlcall.args[g_xmlcall.argsize] = 'i'; } else if (!strncmp(g_data, "", 9)) { g_xmlcall.args[g_xmlcall.argsize] = 'b'; } else if (!strncmp(g_data, "", 8)) { g_xmlcall.args[g_xmlcall.argsize] = 'd'; } else if (!strncmp(g_data, "", 8)) { g_xmlcall.args[g_xmlcall.argsize] = 's'; } else { return XMLRPC_PARSE_ERROR; } /* Now, parse the actual value */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (type != VALUE) { return XMLRPC_PARSE_ERROR; } switch (g_xmlcall.args[g_xmlcall.argsize]) { case 'i': case 'b': g_xmlcall.arguments[g_xmlcall.argsize].u.i = atoi(g_data); break; case 'd': g_xmlcall.arguments[g_xmlcall.argsize].u.d = atof(g_data); break; case 's': strlcpy(g_xmlcall.arguments[g_xmlcall.argsize].u.string, g_data, sizeof(g_xmlcall.arguments[g_xmlcall.argsize].u.string)); break; default: return XMLRPC_PARSE_ERROR; } g_xmlcall.argsize++; /* Now we close out the tag, starting with the type */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (!((type == TAG) && (!strncmp(g_data, " close */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (!((type == TAG) && (!strncmp(g_data, "", 8)))) { return XMLRPC_PARSE_ERROR; } /* Finally, close out the tag */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (!((type == TAG) && (!strncmp(g_data, "", 8)))) { return XMLRPC_PARSE_ERROR; } return XMLRPC_NO_ERROR; } static int xmlrpc_parseparams(struct parsebuf_s *pbuf) { int ret = XMLRPC_PARSE_ERROR; int type; /* First, look for the params tag */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if ((type == TAG) && (!strncmp(g_data, "", 8))) { while (1) { /* Get next tag */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if ((type == TAG) && (!strncmp(g_data, "", 7))) { ret = xmlrpc_parseparam(pbuf); } else if ((type == TAG) && (!strncmp(g_data, "", 9))) { return XMLRPC_NO_ERROR; } else { return XMLRPC_PARSE_ERROR; } } } return ret; } static int xmlrpc_parsemethod(struct parsebuf_s *pbuf) { int ret = XMLRPC_PARSE_ERROR; int type; memset(&g_xmlcall, 0, sizeof(struct xmlrpc_s)); /* Look for the methodName tag */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if ((type == TAG) && (!strncmp(g_data, "", 12))) { /* Get the method name for the call */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if (type == VALUE) { /* Save the method name */ strlcpy(g_xmlcall.name, g_data, sizeof(g_xmlcall.name)); /* Find the closing /methodCall */ type = xmlrpc_getelement(pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if ((type == TAG) && (!strncmp(g_data, "", 13))) { /* Now, it's time to parse the parameters */ ret = xmlrpc_parseparams(pbuf); } } } return ret; } static void xmlrpc_sendfault(int fault) { fault = -fault; if (fault >= MAX_ERROR_CODE) { fault = 0; } xmlrpc_buildresponse(&g_xmlcall, "{is}", "faultCode", fault, "faultString", g_error_strings[fault]); } /**************************************************************************** * Public Functions ****************************************************************************/ int xmlrpc_parse(int sock, char *buffer) { struct parsebuf_s pbuf; int type; int ret = XMLRPC_PARSE_ERROR; pbuf.buf = buffer; pbuf.len = strlen(buffer); pbuf.index = 0; /* Parse the xml header tag */ type = xmlrpc_getelement(&pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if ((type == TAG) && (!strncmp(g_data, "", 12))) { /* Parse the remaining tags within the methodCall tag */ xmlrpc_parsemethod(&pbuf); /* Check for the closing /methodCall */ type = xmlrpc_getelement(&pbuf, g_data, CONFIG_XMLRPC_STRINGSIZE); if ((type == TAG) && (!strncmp(g_data, "", 13))) { /* Successful parse, try to call a user function */ ret = xmlrpc_call(&g_xmlcall); } } } if (ret == 0) { write(sock, g_xmlcall.response, strlen(g_xmlcall.response)); } else { /* Send fault response */ g_xmlcall.error = 1; xmlrpc_sendfault(ret); write(sock, g_xmlcall.response, strlen(g_xmlcall.response)); } return ret; } void xmlrpc_register(struct xmlrpc_entry_s *entry) { if (g_entries == NULL) { g_entries = entry; entry->next = NULL; } else { entry->next = g_entries; g_entries = entry; } }