ultimatepp/uppsrc/XmlRpc/Server.cpp
cxl 9753418656 .XmlRpc: ThrowXmlRpcError without code overload
git-svn-id: svn://ultimatepp.org/upp/trunk@2708 f0d560ea-af0d-0410-9eb7-867de7ffcac7
2010-09-19 15:45:01 +00:00

150 lines
3.8 KiB
C++

#include "XmlRpc.h"
#include "XmlRpc.h"
#define LLOG(x) // LOG(x)
typedef void (*XmlRpcFnPtr)(XmlRpcData&);
static StaticMutex XmlRpcMapMutex;
VectorMap<String, XmlRpcFnPtr>& XmlRpcMap(const char *group)
{
static VectorMap<String, VectorMap< String, void (*)(XmlRpcData&) > > mm;
return mm.GetAdd(group);
}
void Register(const char *name, void (*method)(XmlRpcData&), const char *group)
{
Mutex::Lock __(XmlRpcMapMutex);
XmlRpcMap(group).Add(name, method);
}
XmlRpcFnPtr XmlRpcMapGet(const char *group, const char *name)
{
Mutex::Lock __(XmlRpcMapMutex);
return XmlRpcMap(group).Get(name, NULL);
}
struct XmlRpcError {
int code;
String text;
};
void ThrowXmlRpcError(int code, const char *s)
{
XmlRpcError e;
e.code = code;
e.text = s;
throw e;
}
void ThrowXmlRpcError(const char *s)
{
ThrowXmlRpcError(-1, s);
}
String XmlRpcExecute(const String& request, const char *group, const char *peeraddr)
{
XmlParser p(request);
XmlRpcData data;
String methodname;
try {
String r = XmlHeader();
r << "<methodResponse>\r\n";
p.ReadPI();
p.PassTag("methodCall");
p.PassTag("methodName");
methodname = p.ReadText();
LLOG("method name: " << methodname);
p.PassEnd();
data.peeraddr = peeraddr;
data.in = ParseXmlRpcParams(p);
XmlRpcMapMutex.Enter();
void (*fn)(XmlRpcData&) = XmlRpcMapGet(group, methodname);
if(fn) {
(*fn)(data);
if(IsValueArray(data.out)) {
ValueArray va = data.out;
if(va.GetCount() && IsError(va[0])) {
LLOG("ProcessingError");
return FormatXmlRpcError(XMLRPC_SERVER_PROCESSING_ERROR, "Processing error: " + GetErrorText(data.out[0]));
}
r << FormatXmlRpcParams(data.out);
}
r << "\r\n</methodResponse>\r\n";
}
else {
XmlRpcMapMutex.Leave();
return FormatXmlRpcError(XMLRPC_UNKNOWN_METHOD_ERROR, "\'" + methodname + "\' method is unknown");
}
p.PassEnd();
return r;
}
catch(XmlRpcError e) {
LLOG("Client error: " << e.text);
return FormatXmlRpcError(e.code, e.text);
}
catch(XmlError e) {
LLOG("XmlError " << e << ": " << p.GetPtr());
return FormatXmlRpcError(XMLRPC_SERVER_XML_ERROR, "XML Error: " + e);
}
catch(ValueTypeMismatch) {
LLOG("ValueTypeMismatch at parameter " << data.ii);
return FormatXmlRpcError(XMLRPC_SERVER_PARAM_ERROR, "Parameter mismatch at parameter " + AsString(data.ii));
}
return Null;
}
int CharFilterNoCr(int c)
{
return c == '\r' ? 0 : c;
}
String ReadLine(Socket& s)
{
return Filter(s.ReadUntil('\n'), CharFilterNoCr);
}
bool XmlRpcPerform(Socket& http, const char *group)
{
LLOG("=== Accepted connection ===================================================");
String request = ReadLine(http);
if(http.IsError()) return false;
VectorMap<String, String> hdr;
for(;;) {
String s = ReadLine(http);
if(s.IsEmpty()) break;
LLOG(s);
int q = s.Find(':');
if(q >= 0)
hdr.GetAdd(s.Mid(0, q)) = TrimLeft(s.Mid(q + 1));
}
if(http.IsError()) return false;
String r = XmlRpcExecute(http.ReadCount(atoi(hdr.Get("Content-Length", "")), 90000),
group, http.GetPeerAddr());
LLOG("--------- Server response:\n" << r << "=============");
String response;
response <<
"HTTP/1.1 200 OK\r\n"
"Date: Mon, 23 May 2005 22:38:34 GMT\r\n"
"Server: U++\r\n"
"Last-Modified: Wed, 08 Jan 2003 23:11:55 GMT\r\n"
"Accept-Ranges: bytes\r\n"
"Content-Length: " << r.GetCount() << "\r\n"
"Connection: close\r\n"
"Content-Type: application/soap+xml; charset=utf-8\r\n\r\n" << r;
http.Write(response);
return true;
}
bool XmlRpcServer(int port, const char *group)
{
Socket rpc;
if(!ServerSocket(rpc, 1234, true, 5))
return false;
for(;;) {
Socket http;
if(rpc.Accept(http))
XmlRpcPerform(http, group);
}
}