/* This file is part of libXMLRPC - a C library for xml-encoded function calls. Author: Dan Libby (dan@libby.com) Epinions.com may be contacted at feedback@epinions-inc.com */ /* Copyright 2000 Epinions, Inc. Subject to the following 3 conditions, Epinions, Inc. permits you, free of charge, to (a) use, copy, distribute, modify, perform and display this software and associated documentation files (the "Software"), and (b) permit others to whom the Software is furnished to do so as well. 1) The above copyright notice and this permission notice shall be included without modification in all copies or substantial portions of the Software. 2) THE SOFTWARE IS PROVIDED "AS IS", WITHOUT ANY WARRANTY OR CONDITION OF ANY KIND, EXPRESS, IMPLIED OR STATUTORY, INCLUDING WITHOUT LIMITATION ANY IMPLIED WARRANTIES OF ACCURACY, MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR NONINFRINGEMENT. 3) IN NO EVENT SHALL EPINIONS, INC. BE LIABLE FOR ANY DIRECT, INDIRECT, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES OR LOST PROFITS ARISING OUT OF OR IN CONNECTION WITH THE SOFTWARE (HOWEVER ARISING, INCLUDING NEGLIGENCE), EVEN IF EPINIONS, INC. IS AWARE OF THE POSSIBILITY OF SUCH DAMAGES. */ /* This is a simple demonstration of how to write an xmlrpc server. * This program reads an XML document from standard input and, * if it is a valid XMLRPC request, generates a response to * stdout. Various output options are available. * * For usage, see below or execute ./server --help * * See also ./sample which contains both this and ./client in * a standalone program. */ #include #include #include "xmlrpc.h" XMLRPC_VALUE method_echo(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { return XMLRPC_RequestGetData(request); } XMLRPC_VALUE method_TestStruct(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { XMLRPC_VALUE output; XMLRPC_VALUE str; XMLRPC_VALUE xe_struct; const char *cardholder; output = XMLRPC_CreateVector(NULL, xmlrpc_vector_array); str = XMLRPC_VectorRewind(XMLRPC_RequestGetData(request)); cardholder = XMLRPC_VectorGetStringWithID(str, "Cardholder"); if(cardholder) { XMLRPC_VectorAppendString(xe_struct, "Cardholder", cardholder, 0); } xe_struct = XMLRPC_CreateVector(NULL, xmlrpc_vector_struct); XMLRPC_VectorAppendString(xe_struct, "Reason", "Whew!!!", 0); XMLRPC_VectorAppendString(xe_struct, "Rubadubdub", "Inmytub", 0); XMLRPC_AddValueToVector(output, xe_struct); return output; } XMLRPC_VALUE method_TestArray(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { const char *string; char *testing[3] = { "One", "Two", "Three and four", }; int i; XMLRPC_VALUE output; XMLRPC_VALUE xIter; char* outBuf; output = XMLRPC_CreateVector(NULL, xmlrpc_vector_array); xIter = XMLRPC_VectorRewind(XMLRPC_VectorRewind(XMLRPC_RequestGetData(request))); while(xIter) { string = XMLRPC_GetValueString(xIter); if(string) { XMLRPC_VectorAppendString(output, NULL, string, 0); } xIter = XMLRPC_VectorNext(XMLRPC_RequestGetData(request)); } for(i = 0; i < 3; i++) { XMLRPC_VectorAppendString(output, NULL, testing[i], 0); } return output; } XMLRPC_VALUE method_TestBoolean(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { int iVal = 1; XMLRPC_VALUE xVal = XMLRPC_VectorGetValueWithID(XMLRPC_VectorRewind(XMLRPC_RequestGetData(request)), "boolean"); if(xVal && XMLRPC_GetValueType(xVal) == xmlrpc_boolean) { iVal = XMLRPC_GetValueBoolean(xVal); } return XMLRPC_CreateValueBoolean(NULL, iVal); } XMLRPC_VALUE method_TestInt(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { int iVal = 25; XMLRPC_VALUE xParams = XMLRPC_RequestGetData(request); XMLRPC_VALUE xArg1Struct = XMLRPC_VectorRewind(xParams); XMLRPC_VALUE xVal = XMLRPC_VectorGetValueWithID(xArg1Struct, "int"); if(xVal && XMLRPC_GetValueType(xVal) == xmlrpc_int) { iVal = XMLRPC_GetValueInt(xVal); } return XMLRPC_CreateValueInt(NULL, iVal); } XMLRPC_VALUE method_TestString(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { const char* pVal = "Hello World"; XMLRPC_VALUE xVal = XMLRPC_VectorGetValueWithID(XMLRPC_VectorRewind(XMLRPC_RequestGetData(request)), "string"); if(xVal && XMLRPC_GetValueType(xVal) == xmlrpc_string) { pVal = XMLRPC_GetValueString(xVal); } return XMLRPC_CreateValueString(NULL, pVal, 0); } XMLRPC_VALUE method_TestDouble(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { double dVal = 25; XMLRPC_VALUE xVal = XMLRPC_VectorGetValueWithID(XMLRPC_VectorRewind(XMLRPC_RequestGetData(request)), "double"); if(xVal && XMLRPC_GetValueType(xVal) == xmlrpc_double) { dVal = XMLRPC_GetValueDouble(xVal); } return XMLRPC_CreateValueDouble(NULL, dVal); } XMLRPC_VALUE method_TestBase64(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { const char* pVal = "U29tZUJhc2U2NFN0cmluZw=="; XMLRPC_VALUE xVal = XMLRPC_VectorGetValueWithID(XMLRPC_VectorRewind(XMLRPC_RequestGetData(request)), "base64"); int buf_len = 0; if(xVal && XMLRPC_GetValueType(xVal) == xmlrpc_base64) { pVal = XMLRPC_GetValueBase64(xVal); buf_len = XMLRPC_GetValueStringLen(xVal); } return XMLRPC_CreateValueBase64(NULL, pVal, buf_len); } XMLRPC_VALUE method_TestDateTime(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { const char* pVal = "19980717T14:08:55"; XMLRPC_VALUE xVal = XMLRPC_VectorGetValueWithID(XMLRPC_VectorRewind(XMLRPC_RequestGetData(request)), "datetime"); if(xVal && XMLRPC_GetValueType(xVal) == xmlrpc_datetime) { pVal = XMLRPC_GetValueDateTime_ISO8601(xVal); } return XMLRPC_CreateValueDateTime_ISO8601(NULL, pVal); } XMLRPC_VALUE method_TestNormal(XMLRPC_SERVER server, XMLRPC_REQUEST request, void* userData) { XMLRPC_VALUE output; output = XMLRPC_CreateVector(NULL, xmlrpc_vector_struct); XMLRPC_AddValuesToVector(output, method_TestStruct(server, request, userData), method_TestArray(server, request, userData), method_TestBoolean(server, request, userData), method_TestInt(server, request, userData), method_TestString(server, request, userData), method_TestDouble(server, request, userData), method_TestDateTime(server, request, userData), method_TestBase64(server, request, userData), NULL); return output; } XMLRPC_VALUE method_TestFault(XMLRPC_SERVER server, XMLRPC_REQUEST input, void* userData) { return XMLRPC_UtilityCreateFault(404, "Page Not Found"); } void print_help() { printf("Usage: server [OPTION VALUE]\n\n"); printf("\t-help this help message\n"); printf("\t-encoding (any standard character encoding)\n"); printf("\t-escaping (may repeat)\n"); printf("\t-method \n"); printf("\t-output \n"); printf("\t-verbosity \n"); printf("\t-version \n"); printf("\n\tavailable methods:\n" "\t\tmethod_Echo\n" "\t\tmethod_TestNormal\n" "\t\tmethod_TestFault\n" "\t\tmethod_TestStruct\n" "\t\tmethod_TestArray\n" "\t\tmethod_TestBoolean\n" "\t\tmethod_TestInt\n" "\t\tmethod_TestString\n" "\t\tmethod_TestDouble\n" "\t\tmethod_TestBase64\n" "\t\tmethod_TestDateTime\n" "\t\tmethod_TestFault\n"); } int main(int argc, char **argv) { int number; int i; XMLRPC_SERVER server; XMLRPC_REQUEST request, response; STRUCT_XMLRPC_REQUEST_OUTPUT_OPTIONS call_options; /* args */ int verbosity = 0; int version = 0; int escaping = 0; int output = 0; char *methodName = "method_TestNormal"; char *encoding = 0; /* for every argument (after the program name) */ for(i=1; i