diff options
Diffstat (limited to 'src/modules/rpc.h')
-rw-r--r-- | src/modules/rpc.h | 296 |
1 files changed, 296 insertions, 0 deletions
diff --git a/src/modules/rpc.h b/src/modules/rpc.h new file mode 100644 index 000000000..cdaec4082 --- /dev/null +++ b/src/modules/rpc.h @@ -0,0 +1,296 @@ +#ifndef RPC_H +#define RPC_H + +#include <string> +#include <map> +#include <stdexcept> + +class RPCValue; + +typedef enum +{ + RPCNull, + RPCBoolean, + RPCInteger, + RPCString, + RPCArray, + RPCObject +} RPCValueType; + +typedef std::map<std::string,RPCValue*> RPCObjectContainer; +typedef std::vector<RPCValue*> RPCArrayContainer; + +class RPCValue : public classbase +{ + protected: + RPCValueType type; + void *value; + + double *CastInteger() + { + return (double*)value; + } + + std::string *CastString() + { + return (std::string*)value; + } + + RPCObjectContainer *CastObject() + { + return (RPCObjectContainer*)value; + } + + RPCArrayContainer *CastArray() + { + return (RPCArrayContainer*)value; + } + + void DestroyValue() + { + // Some versions of GCC complain about declaration in switch statements + RPCArrayContainer *a; + RPCObjectContainer *o; + switch (type) + { + case RPCInteger: + delete this->CastInteger(); + break; + case RPCString: + delete this->CastString(); + break; + case RPCArray: + a = this->CastArray(); + for (RPCArrayContainer::iterator i = a->begin(); i != a->end(); i++) + delete *i; + delete a; + break; + case RPCObject: + o = this->CastObject(); + for (RPCObjectContainer::iterator i = o->begin(); i != o->end(); i++) + delete i->second; + delete o; + break; + default: + break; + } + + value = NULL; + } + + void InitValue() + { + switch (type) + { + case RPCNull: + case RPCBoolean: + value = NULL; + break; + case RPCInteger: + value = new double; + break; + case RPCString: + value = new std::string; + break; + case RPCArray: + value = new RPCArrayContainer; + break; + case RPCObject: + value = new RPCObjectContainer; + break; + } + } + + RPCValue(const RPCValue &v) { } + + public: + RPCValue *parent; + + RPCValue(RPCValue *parent = NULL) : type(RPCNull), value(NULL), parent(parent) { } + RPCValue(RPCValueType type, RPCValue *parent = NULL) : type(type), value(NULL), parent(parent) { InitValue(); } + RPCValue(bool nvalue, RPCValue *parent = NULL) : type(RPCBoolean), value((void*)nvalue), parent(parent) { } + RPCValue(double nvalue, RPCValue *parent = NULL) : type(RPCInteger), parent(parent) { value = new double(nvalue); } + RPCValue(const std::string &nvalue, RPCValue *parent = NULL) : type(RPCString), parent(parent) { value = new std::string(nvalue); } + + virtual ~RPCValue() + { + DestroyValue(); + } + + RPCValueType GetType() + { + return type; + } + + void SetNull() + { + DestroyValue(); + type = RPCNull; + } + + void SetBoolean(bool nvalue) + { + DestroyValue(); + value = (void*)nvalue; + type = RPCBoolean; + } + + void SetInteger(double nvalue) + { + if (type == RPCInteger) + { + *this->CastInteger() = nvalue; + } + else + { + DestroyValue(); + value = new double(nvalue); + type = RPCInteger; + } + } + + void SetString(const std::string &nvalue) + { + if (type == RPCString) + { + this->CastString()->assign(nvalue); + } + else + { + DestroyValue(); + value = new std::string(nvalue); + type = RPCString; + } + } + + void SetArray() + { + if (type == RPCArray) + { + this->CastArray()->clear(); + } + else + { + DestroyValue(); + type = RPCArray; + InitValue(); + } + } + + void SetObject() + { + if (type == RPCObject) + { + this->CastObject()->clear(); + } + else + { + DestroyValue(); + type = RPCObject; + InitValue(); + } + } + + void ArrayAdd(RPCValue *nvalue) + { + if (type != RPCArray) + return; + RPCArrayContainer *a = this->CastArray(); + a->push_back(nvalue); + nvalue->parent = this; + } + + void ObjectAdd(const std::string &key, RPCValue *nvalue) + { + if (type != RPCObject) + return; + RPCObjectContainer *o = this->CastObject(); + o->insert(std::make_pair(key, nvalue)); + nvalue->parent = this; + } + + RPCValue *GetArray(int i) + { + if (type != RPCArray) + return NULL; + RPCArrayContainer *a = this->CastArray(); + if ((i < 0) || (i >= (signed)a->size())) + return NULL; + return a->at(i); + } + + int ArraySize() + { + if (type != RPCArray) + return 0; + RPCArrayContainer *a = this->CastArray(); + return a->size(); + } + + RPCValue *GetObject(const std::string &key) + { + if (type != RPCObject) + return NULL; + RPCObjectContainer *o = this->CastObject(); + RPCObjectContainer::iterator it = o->find(key); + if (it == o->end()) + return NULL; + return it->second; + } + + std::pair<RPCObjectContainer::iterator,RPCObjectContainer::iterator> GetObjectIterator() throw() + { + if (type != RPCObject) + throw std::runtime_error("Cannot get iterator for a non-object RPC value"); + RPCObjectContainer *o = this->CastObject(); + return std::make_pair(o->begin(), o->end()); + } + + std::string GetString() + { + if (type != RPCString) + return std::string(); + return *this->CastString(); + } + + double GetInt() + { + if (type != RPCInteger) + return 0; + return *this->CastInteger(); + } + + bool GetBool() + { + if (type != RPCBoolean) + return 0; + return (value != NULL); + } +}; + +class RPCRequest : public classbase +{ + protected: + + public: + std::string method; + RPCValue *parameters; + RPCValue *result; + std::string provider; + bool claimed; + std::string error; + + RPCRequest(const std::string &provider, const std::string &method, RPCValue *parameters) + : method(method), parameters(parameters), provider(provider), claimed(false) + { + result = new RPCValue(); + } + + ~RPCRequest() + { + if (result) + delete result; + } +}; + +#endif |