Restructured some for clarity, got rid of the crappy JSON library, general awesomeness.
This commit is contained in:
		
							parent
							
								
									022ff26205
								
							
						
					
					
						commit
						e3ecdb1e4b
					
				
					 34 changed files with 965 additions and 6996 deletions
				
			
		|  | @ -1,4 +1,4 @@ | |||
| SRC = main.cpp ../util/json/json_reader.cpp ../util/json/json_value.cpp ../util/json/json_writer.cpp ../util/socket.cpp ../util/dtsc.cpp | ||||
| SRC = main.cpp ../util/json.cpp ../util/socket.cpp ../util/dtsc.cpp | ||||
| OBJ = $(SRC:.cpp=.o) | ||||
| OUT = DDV_Buffer | ||||
| INCLUDES =  | ||||
|  |  | |||
|  | @ -14,7 +14,7 @@ | |||
| #include <sys/time.h> | ||||
| #include "../util/dtsc.h" //DTSC support
 | ||||
| #include "../util/socket.h" //Socket lib
 | ||||
| #include "../util/json/json.h" | ||||
| #include "../util/json.h" | ||||
| 
 | ||||
| /// Holds all code unique to the Buffer.
 | ||||
| namespace Buffer{ | ||||
|  | @ -27,7 +27,7 @@ namespace Buffer{ | |||
|   }//getNowMS
 | ||||
| 
 | ||||
| 
 | ||||
|   Json::Value Storage = Json::Value(Json::objectValue); ///< Global storage of data.
 | ||||
|   JSON::Value Storage; ///< Global storage of data.
 | ||||
|    | ||||
|   ///A simple signal handler that ignores all signals.
 | ||||
|   void termination_handler (int signum){ | ||||
|  | @ -201,9 +201,9 @@ namespace Buffer{ | |||
|     Socket::Connection std_input(fileno(stdin)); | ||||
|     Socket::Connection StatsSocket = Socket::Connection("/tmp/ddv_statistics", true); | ||||
| 
 | ||||
|     Storage["log"] = Json::Value(Json::objectValue); | ||||
|     Storage["curr"] = Json::Value(Json::objectValue); | ||||
|     Storage["totals"] = Json::Value(Json::objectValue); | ||||
|     Storage["log"] = JSON::Value(); | ||||
|     Storage["curr"] = JSON::Value(); | ||||
|     Storage["totals"] = JSON::Value(); | ||||
|      | ||||
| 
 | ||||
|     while (!feof(stdin) || ip_waiting){ | ||||
|  | @ -228,8 +228,8 @@ namespace Buffer{ | |||
|           StatsSocket = Socket::Connection("/tmp/ddv_statistics", true); | ||||
|         } | ||||
|         if (StatsSocket.connected()){ | ||||
|           StatsSocket.write(Storage.toStyledString()+"\n\n"); | ||||
|           Storage["log"].clear(); | ||||
|           StatsSocket.write(Storage.toString()+"\n\n"); | ||||
|           Storage["log"].null(); | ||||
|         } | ||||
|       } | ||||
|       //invalidate the current buffer
 | ||||
|  |  | |||
|  | @ -1,4 +1,4 @@ | |||
| SRC = main.cpp ../util/socket.cpp ../util/http_parser.cpp ../util/flv_tag.cpp ../util/amf.cpp ../util/dtsc.cpp ../util/util.cpp | ||||
| SRC = main.cpp ../util/socket.cpp ../util/http_parser.cpp ../util/flv_tag.cpp ../util/amf.cpp ../util/dtsc.cpp ../util/config.cpp ../util/base64.cpp | ||||
| OBJ = $(SRC:.cpp=.o) | ||||
| OUT = DDV_Conn_HTTP | ||||
| INCLUDES =  | ||||
|  | @ -22,5 +22,5 @@ clean: | |||
| install: $(OUT) | ||||
| 	cp -f ./$(OUT) /usr/bin/ | ||||
| cversion: | ||||
| 	rm -rf ../util/util.o | ||||
| 	rm -rf ../util/config.o | ||||
| 
 | ||||
|  |  | |||
|  | @ -16,76 +16,15 @@ | |||
| #include "../util/dtsc.h" | ||||
| #include "../util/flv_tag.h" | ||||
| #include "../util/MP4/interface.cpp" | ||||
| #include "../util/base64.h" | ||||
| #include "../util/amf.h" | ||||
| 
 | ||||
| /// Holds everything unique to HTTP Connector.
 | ||||
| namespace Connector_HTTP{ | ||||
| 
 | ||||
|   /// Defines the type of handler used to process this request.
 | ||||
|   enum {HANDLER_NONE, HANDLER_PROGRESSIVE, HANDLER_FLASH, HANDLER_APPLE, HANDLER_MICRO}; | ||||
|   enum {HANDLER_NONE, HANDLER_PROGRESSIVE, HANDLER_FLASH, HANDLER_APPLE, HANDLER_MICRO, HANDLER_JSCRIPT}; | ||||
| 
 | ||||
|   /// Needed for base64_encode function
 | ||||
|   static const std::string base64_chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | ||||
| 
 | ||||
|   /// Helper for base64_decode function
 | ||||
|   static inline bool is_base64(unsigned char c) { | ||||
|     return (isalnum(c) || (c == '+') || (c == '/')); | ||||
|   } | ||||
| 
 | ||||
|   /// Used to base64 encode data. Input is the plaintext as std::string, output is the encoded data as std::string.
 | ||||
|   /// \param input Plaintext data to encode.
 | ||||
|   /// \returns Base64 encoded data.
 | ||||
|   std::string base64_encode(std::string const input) { | ||||
|     std::string ret; | ||||
|     unsigned int in_len = input.size(); | ||||
|     char quad[4], triple[3]; | ||||
|     unsigned int i, x, n = 3; | ||||
|     for (x = 0; x < in_len; x = x + 3){ | ||||
|       if ((in_len - x) / 3 == 0){n = (in_len - x) % 3;} | ||||
|       for (i=0; i < 3; i++){triple[i] = '0';} | ||||
|       for (i=0; i < n; i++){triple[i] = input[x + i];} | ||||
|       quad[0] = base64_chars[(triple[0] & 0xFC) >> 2]; // FC = 11111100
 | ||||
|       quad[1] = base64_chars[((triple[0] & 0x03) << 4) | ((triple[1] & 0xF0) >> 4)]; // 03 = 11
 | ||||
|       quad[2] = base64_chars[((triple[1] & 0x0F) << 2) | ((triple[2] & 0xC0) >> 6)]; // 0F = 1111, C0=11110
 | ||||
|       quad[3] = base64_chars[triple[2] & 0x3F]; // 3F = 111111
 | ||||
|       if (n < 3){quad[3] = '=';} | ||||
|       if (n < 2){quad[2] = '=';} | ||||
|       for(i=0; i < 4; i++){ret += quad[i];} | ||||
|     } | ||||
|     return ret; | ||||
|   }//base64_encode
 | ||||
| 
 | ||||
|   /// Used to base64 decode data. Input is the encoded data as std::string, output is the plaintext data as std::string.
 | ||||
|   /// \param input Base64 encoded data to decode.
 | ||||
|   /// \returns Plaintext decoded data.
 | ||||
|   std::string base64_decode(std::string const& encoded_string) { | ||||
|     int in_len = encoded_string.size(); | ||||
|     int i = 0; | ||||
|     int j = 0; | ||||
|     int in_ = 0; | ||||
|     unsigned char char_array_4[4], char_array_3[3]; | ||||
|     std::string ret; | ||||
|     while (in_len-- && ( encoded_string[in_] != '=') && is_base64(encoded_string[in_])) { | ||||
|       char_array_4[i++] = encoded_string[in_]; in_++; | ||||
|       if (i ==4) { | ||||
|         for (i = 0; i <4; i++){char_array_4[i] = base64_chars.find(char_array_4[i]);} | ||||
|         char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); | ||||
|         char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); | ||||
|         char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; | ||||
|         for (i = 0; (i < 3); i++){ret += char_array_3[i];} | ||||
|         i = 0; | ||||
|       } | ||||
|     } | ||||
|     if (i) { | ||||
|       for (j = i; j <4; j++){char_array_4[j] = 0;} | ||||
|       for (j = 0; j <4; j++){char_array_4[j] = base64_chars.find(char_array_4[j]);} | ||||
|       char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); | ||||
|       char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); | ||||
|       char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; | ||||
|       for (j = 0; (j < i - 1); j++) ret += char_array_3[j]; | ||||
|     } | ||||
|     return ret; | ||||
|   } | ||||
| 
 | ||||
|   /// Returns AMF-format metadata for Adobe HTTP Dynamic Streaming.
 | ||||
|   std::string GetMetaData( ) { | ||||
|  | @ -136,13 +75,13 @@ namespace Connector_HTTP{ | |||
|     Result += "<streamType>live</streamType>\n"; | ||||
|     Result += "<deliveryType>streaming</deliveryType>\n"; | ||||
|     Result += "<bootstrapInfo profile=\"named\" id=\"bootstrap1\">"; | ||||
|     Result += base64_encode(temp->GenerateLiveBootstrap(1)); | ||||
|     Result += Base64::encode(temp->GenerateLiveBootstrap(1)); | ||||
|     Result += "</bootstrapInfo>\n"; | ||||
|     Result += "<media streamId=\"1\" bootstrapInfoId=\"bootstrap1\" url=\""; | ||||
|     Result += MovieId; | ||||
|     Result += "/\">\n"; | ||||
|     Result += "<metadata>"; | ||||
|     Result += base64_encode(GetMetaData()); | ||||
|     Result += Base64::encode(GetMetaData()); | ||||
|     Result += "</metadata>\n"; | ||||
|     Result += "</media>\n"; | ||||
|     Result += "</manifest>\n"; | ||||
|  |  | |||
|  | @ -1,4 +1,4 @@ | |||
| SRC = main.cpp ../util/socket.cpp ../util/flv_tag.cpp ../util/amf.cpp ../util/rtmpchunks.cpp ../util/crypto.cpp ../util/util.cpp ../util/dtsc.cpp | ||||
| SRC = main.cpp ../util/socket.cpp ../util/flv_tag.cpp ../util/amf.cpp ../util/rtmpchunks.cpp ../util/crypto.cpp ../util/config.cpp ../util/dtsc.cpp | ||||
| OBJ = $(SRC:.cpp=.o) | ||||
| OUT = DDV_Conn_RTMP | ||||
| INCLUDES = | ||||
|  | @ -23,5 +23,5 @@ clean: | |||
| install: $(OUT) | ||||
| 	cp -f ./$(OUT) /usr/bin/ | ||||
| cversion: | ||||
| 	rm -rf ../util/util.o | ||||
| 	rm -rf ../util/config.o | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,4 +1,4 @@ | |||
| SRC = main.cpp ../util/json/json_reader.cpp ../util/json/json_value.cpp ../util/json/json_writer.cpp ../util/socket.cpp ../util/http_parser.cpp ../util/md5.cpp ../util/util.cpp | ||||
| SRC = main.cpp ../util/json.cpp ../util/socket.cpp ../util/http_parser.cpp ../util/md5.cpp ../util/config.cpp ../util/procs.cpp ../util/base64.cpp ../util/auth.cpp | ||||
| OBJ = $(SRC:.cpp=.o) | ||||
| OUT = DDV_Controller | ||||
| INCLUDES =  | ||||
|  | @ -23,5 +23,5 @@ $(OUT): $(OBJ) | |||
| clean: | ||||
| 	rm -rf $(OBJ) $(OUT) Makefile.bak *~ | ||||
| cversion: | ||||
| 	rm -rf ../util/util.o | ||||
| 	rm -rf ../util/config.o | ||||
| 
 | ||||
|  |  | |||
|  | @ -23,16 +23,16 @@ | |||
| #include "../util/socket.h" | ||||
| #include "../util/http_parser.h" | ||||
| #include "../util/md5.h" | ||||
| #include "../util/json/json.h" | ||||
| #include "../util/util.h" | ||||
| 
 | ||||
| #include <openssl/rsa.h> | ||||
| #include <openssl/x509.h> | ||||
| #include "../util/json.h" | ||||
| #include "../util/procs.h" | ||||
| #include "../util/config.h" | ||||
| #include "../util/auth.h" | ||||
| 
 | ||||
| #define UPLINK_INTERVAL 30 | ||||
| 
 | ||||
| Socket::Server API_Socket; ///< Main connection socket.
 | ||||
| std::map<std::string, int> lastBuffer; ///< Last moment of contact with all buffers.
 | ||||
| Auth keychecker; ///< Checks key authorization.
 | ||||
| 
 | ||||
| /// Basic signal handler. Disconnects the server_socket if it receives
 | ||||
| /// a SIGINT, SIGHUP or SIGTERM signal, but does nothing for SIGPIPE.
 | ||||
|  | @ -61,114 +61,8 @@ void signal_handler (int signum){ | |||
| }//signal_handler
 | ||||
| 
 | ||||
| 
 | ||||
| /// Needed for base64_encode function
 | ||||
| static const std::string base64_chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | ||||
| 
 | ||||
| /// Helper for base64_decode function
 | ||||
| static inline bool is_base64(unsigned char c) { | ||||
|   return (isalnum(c) || (c == '+') || (c == '/')); | ||||
| } | ||||
| 
 | ||||
| /// Used to base64 encode data. Input is the plaintext as std::string, output is the encoded data as std::string.
 | ||||
| /// \param input Plaintext data to encode.
 | ||||
| /// \returns Base64 encoded data.
 | ||||
| std::string base64_encode(std::string const input) { | ||||
|   std::string ret; | ||||
|   unsigned int in_len = input.size(); | ||||
|   char quad[4], triple[3]; | ||||
|   unsigned int i, x, n = 3; | ||||
|   for (x = 0; x < in_len; x = x + 3){ | ||||
|     if ((in_len - x) / 3 == 0){n = (in_len - x) % 3;} | ||||
|     for (i=0; i < 3; i++){triple[i] = '0';} | ||||
|     for (i=0; i < n; i++){triple[i] = input[x + i];} | ||||
|     quad[0] = base64_chars[(triple[0] & 0xFC) >> 2]; // FC = 11111100
 | ||||
|     quad[1] = base64_chars[((triple[0] & 0x03) << 4) | ((triple[1] & 0xF0) >> 4)]; // 03 = 11
 | ||||
|     quad[2] = base64_chars[((triple[1] & 0x0F) << 2) | ((triple[2] & 0xC0) >> 6)]; // 0F = 1111, C0=11110
 | ||||
|     quad[3] = base64_chars[triple[2] & 0x3F]; // 3F = 111111
 | ||||
|     if (n < 3){quad[3] = '=';} | ||||
|     if (n < 2){quad[2] = '=';} | ||||
|     for(i=0; i < 4; i++){ret += quad[i];} | ||||
|   } | ||||
|   return ret; | ||||
| }//base64_encode
 | ||||
| 
 | ||||
| /// Used to base64 decode data. Input is the encoded data as std::string, output is the plaintext data as std::string.
 | ||||
| /// \param input Base64 encoded data to decode.
 | ||||
| /// \returns Plaintext decoded data.
 | ||||
| std::string base64_decode(std::string const& encoded_string) { | ||||
|   int in_len = encoded_string.size(); | ||||
|   int i = 0; | ||||
|   int j = 0; | ||||
|   int in_ = 0; | ||||
|   unsigned char char_array_4[4], char_array_3[3]; | ||||
|   std::string ret; | ||||
|   while (in_len-- && ( encoded_string[in_] != '=') && is_base64(encoded_string[in_])) { | ||||
|     char_array_4[i++] = encoded_string[in_]; in_++; | ||||
|     if (i ==4) { | ||||
|       for (i = 0; i <4; i++){char_array_4[i] = base64_chars.find(char_array_4[i]);} | ||||
|       char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); | ||||
|       char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); | ||||
|       char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; | ||||
|       for (i = 0; (i < 3); i++){ret += char_array_3[i];} | ||||
|       i = 0; | ||||
|     } | ||||
|   } | ||||
|   if (i) { | ||||
|     for (j = i; j <4; j++){char_array_4[j] = 0;} | ||||
|     for (j = 0; j <4; j++){char_array_4[j] = base64_chars.find(char_array_4[j]);} | ||||
|     char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); | ||||
|     char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); | ||||
|     char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; | ||||
|     for (j = 0; (j < i - 1); j++) ret += char_array_3[j]; | ||||
|   } | ||||
|   return ret; | ||||
| } | ||||
| 
 | ||||
| unsigned char __gbv2keypub_der[] = { | ||||
|   0x30, 0x82, 0x01, 0x22, 0x30, 0x0d, 0x06, 0x09, 0x2a, 0x86, 0x48, 0x86, | ||||
|   0xf7, 0x0d, 0x01, 0x01, 0x01, 0x05, 0x00, 0x03, 0x82, 0x01, 0x0f, 0x00, | ||||
|   0x30, 0x82, 0x01, 0x0a, 0x02, 0x82, 0x01, 0x01, 0x00, 0xe5, 0xd7, 0x9c, | ||||
|   0x7d, 0x73, 0xc6, 0xe6, 0xfb, 0x35, 0x7e, 0xd7, 0x57, 0x99, 0x07, 0xdb, | ||||
|   0x99, 0x70, 0xc9, 0xd0, 0x3e, 0x53, 0x57, 0x3c, 0x1e, 0x55, 0xda, 0x0f, | ||||
|   0x69, 0xbf, 0x26, 0x79, 0xc7, 0xb6, 0xdd, 0x8e, 0x83, 0x32, 0x65, 0x74, | ||||
|   0x0d, 0x74, 0x48, 0x42, 0x49, 0x22, 0x52, 0x58, 0x56, 0xc3, 0xe4, 0x49, | ||||
|   0x5d, 0xac, 0x6a, 0x94, 0xb1, 0x64, 0x14, 0xbf, 0x4d, 0xd5, 0xd7, 0x3a, | ||||
|   0xca, 0x5c, 0x1e, 0x6f, 0x42, 0x30, 0xac, 0x29, 0xaa, 0xa0, 0x85, 0xd2, | ||||
|   0x16, 0xa2, 0x8e, 0x89, 0x12, 0xc4, 0x92, 0x06, 0xea, 0xed, 0x48, 0xf6, | ||||
|   0xdb, 0xed, 0x4f, 0x62, 0x6c, 0xfa, 0xcf, 0xc2, 0xb9, 0x8d, 0x04, 0xb2, | ||||
|   0xba, 0x63, 0xc9, 0xcc, 0xee, 0x23, 0x64, 0x46, 0x14, 0x12, 0xc8, 0x38, | ||||
|   0x67, 0x69, 0x6b, 0xaf, 0xd1, 0x7c, 0xb1, 0xb5, 0x79, 0xe4, 0x4e, 0x3a, | ||||
|   0xa7, 0xe8, 0x28, 0x89, 0x25, 0xc0, 0xd0, 0xd8, 0xc7, 0xd2, 0x26, 0xaa, | ||||
|   0xf5, 0xbf, 0x36, 0x55, 0x01, 0x89, 0x58, 0x1f, 0x1e, 0xf5, 0xa5, 0x42, | ||||
|   0x8f, 0x60, 0x2e, 0xc2, 0xd8, 0x21, 0x0b, 0x6c, 0x8d, 0xbb, 0x72, 0xf2, | ||||
|   0x19, 0x30, 0xe3, 0x4c, 0x3e, 0x80, 0xe7, 0xf2, 0xe3, 0x89, 0x4f, 0xd4, | ||||
|   0xee, 0x96, 0x3e, 0x4a, 0x9b, 0xe5, 0x16, 0x01, 0xf1, 0x98, 0xc9, 0x0b, | ||||
|   0xd6, 0xdf, 0x8a, 0x64, 0x47, 0xc4, 0x44, 0xcc, 0x92, 0x69, 0x28, 0xee, | ||||
|   0x7d, 0xac, 0xdc, 0x30, 0x56, 0x3a, 0xe7, 0xbc, 0xba, 0x45, 0x16, 0x2c, | ||||
|   0x4c, 0x46, 0x6b, 0x2b, 0x20, 0xfb, 0x3d, 0x20, 0x35, 0xbb, 0x48, 0x49, | ||||
|   0x13, 0x65, 0xc9, 0x9a, 0x38, 0x10, 0x84, 0x1a, 0x8c, 0xc9, 0xd7, 0xde, | ||||
|   0x07, 0x10, 0x5a, 0xfb, 0xb4, 0x95, 0xae, 0x18, 0xf2, 0xe3, 0x15, 0xe8, | ||||
|   0xad, 0x7e, 0xe5, 0x3c, 0xa8, 0x47, 0x85, 0xd6, 0x1f, 0x54, 0xb5, 0xa3, | ||||
|   0x79, 0x02, 0x03, 0x01, 0x00, 0x01 | ||||
| }; ///< The GBv2 public key file.
 | ||||
| unsigned int __gbv2keypub_der_len = 294; ///< Length of GBv2 public key data
 | ||||
| 
 | ||||
| RSA * pubkey = 0; ///< Holds the public key.
 | ||||
| /// Attempts to load the GBv2 public key.
 | ||||
| void RSA_Load(){ | ||||
|   const unsigned char * key = __gbv2keypub_der; | ||||
|   pubkey = d2i_RSAPublicKey(0, &key, __gbv2keypub_der_len); | ||||
| } | ||||
| 
 | ||||
| /// Attempts to verify RSA signature using the public key loaded with RSA_Load().
 | ||||
| /// Assumes basesign argument is base64 encoded RSA signature for data.
 | ||||
| /// Returns true if the data could be verified, false otherwise.
 | ||||
| bool RSA_check(std::string & data, std::string basesign){ | ||||
|   std::string sign = base64_decode(basesign); | ||||
|   return (RSA_verify(NID_md5, (unsigned char*)data.c_str(), data.size(), (unsigned char*)sign.c_str(), sign.size(), pubkey) == 1); | ||||
| } | ||||
| 
 | ||||
| Json::Value Storage = Json::Value(Json::objectValue); ///< Global storage of data.
 | ||||
| JSON::Value Storage; ///< Global storage of data.
 | ||||
| 
 | ||||
| void WriteFile( std::string Filename, std::string contents ) { | ||||
|   std::ofstream File; | ||||
|  | @ -205,15 +99,15 @@ class ConnectedUser{ | |||
| }; | ||||
| 
 | ||||
| void Log(std::string kind, std::string message){ | ||||
|   Json::Value m; | ||||
|   m.append((Json::Value::UInt)time(0)); | ||||
|   JSON::Value m; | ||||
|   m.append((long long int)time(0)); | ||||
|   m.append(kind); | ||||
|   m.append(message); | ||||
|   Storage["log"].append(m); | ||||
|   std::cout << "[" << kind << "] " << message << std::endl; | ||||
| } | ||||
| 
 | ||||
| void Authorize( Json::Value & Request, Json::Value & Response, ConnectedUser & conn ) { | ||||
| void Authorize( JSON::Value & Request, JSON::Value & Response, ConnectedUser & conn ) { | ||||
|   time_t Time = time(0); | ||||
|   tm * TimeInfo = localtime(&Time); | ||||
|   std::stringstream Date; | ||||
|  | @ -221,9 +115,9 @@ void Authorize( Json::Value & Request, Json::Value & Response, ConnectedUser & c | |||
|   Date << TimeInfo->tm_mday << "-" << TimeInfo->tm_mon << "-" << TimeInfo->tm_year + 1900; | ||||
|   std::string Challenge = md5( Date.str().c_str() + conn.C.getHost() ); | ||||
|   if( Request.isMember( "authorize" ) ) { | ||||
|     std::string UserID = Request["authorize"]["username"].asString(); | ||||
|     std::string UserID = Request["authorize"]["username"]; | ||||
|     if (Storage["account"].isMember(UserID)){ | ||||
|       if( md5( Storage["account"][UserID]["password"].asString() + Challenge ) == Request["authorize"]["password"].asString() ) { | ||||
|       if( md5( (std::string)Storage["account"][UserID]["password"] + Challenge ) == (std::string)Request["authorize"]["password"] ) { | ||||
|         Response["authorize"]["status"] = "OK"; | ||||
|         conn.Username = UserID; | ||||
|         conn.Authorized = true; | ||||
|  | @ -232,7 +126,7 @@ void Authorize( Json::Value & Request, Json::Value & Response, ConnectedUser & c | |||
|     } | ||||
|     if (Storage["authorize"].isMember("key")){ | ||||
|       UserID = "gearbox"; | ||||
|       if (RSA_check(Challenge, Storage["authorize"]["key"].asString())){ | ||||
|       if (keychecker.PubKey_Check(Challenge, Storage["authorize"]["key"])){ | ||||
|         Response["authorize"]["status"] = "OK"; | ||||
|         conn.Username = UserID; | ||||
|         conn.Authorized = true; | ||||
|  | @ -251,14 +145,14 @@ void Authorize( Json::Value & Request, Json::Value & Response, ConnectedUser & c | |||
|   return; | ||||
| } | ||||
| 
 | ||||
| void CheckProtocols(Json::Value & p){ | ||||
| void CheckProtocols(JSON::Value & p){ | ||||
|   static std::map<std::string, std::string> connports; | ||||
|   bool seenHTTP = false; | ||||
|   bool seenRTMP = false; | ||||
|   std::string tmp; | ||||
|   for (Json::ValueIterator jit = p.begin(); jit != p.end(); jit++){ | ||||
|     if (jit.memberName() == std::string("HTTP")){ | ||||
|       tmp = p[jit.memberName()]["port"].asString(); | ||||
|   for (JSON::ObjIter jit = p.ObjBegin(); jit != p.ObjEnd(); jit++){ | ||||
|     if (jit->first == "HTTP"){ | ||||
|       tmp = (std::string)jit->second["port"]; | ||||
|       seenHTTP = true; | ||||
|       if (connports["HTTP"] != tmp){Util::Procs::Stop("HTTP");} | ||||
|       connports["HTTP"] = tmp; | ||||
|  | @ -266,8 +160,8 @@ void CheckProtocols(Json::Value & p){ | |||
|         Util::Procs::Start("HTTP", std::string("DDV_Conn_HTTP -n -p ")+tmp); | ||||
|       } | ||||
|     } | ||||
|     if (jit.memberName() == std::string("RTMP")){ | ||||
|       tmp = p[jit.memberName()]["port"].asString(); | ||||
|     if (jit->first == "RTMP"){ | ||||
|       tmp = (std::string)jit->second["port"]; | ||||
|       seenRTMP = true; | ||||
|       if (connports["RTMP"] != tmp){Util::Procs::Stop("RTMP");} | ||||
|       connports["RTMP"] = tmp; | ||||
|  | @ -280,39 +174,35 @@ void CheckProtocols(Json::Value & p){ | |||
|   if (!seenRTMP){Util::Procs::Stop("RTMP");} | ||||
| } | ||||
| 
 | ||||
| void CheckConfig(Json::Value & in, Json::Value & out){ | ||||
|   if (in.isObject() && (in.size() > 0)){ | ||||
|     for (Json::ValueIterator jit = in.begin(); jit != in.end(); jit++){ | ||||
|       if (out.isObject() && out.isMember(jit.memberName())){ | ||||
|         if (in[jit.memberName()] != out[jit.memberName()]){ | ||||
|           Log("CONF", std::string("Updated configuration value ")+jit.memberName()); | ||||
| void CheckConfig(JSON::Value & in, JSON::Value & out){ | ||||
|   for (JSON::ObjIter jit = in.ObjBegin(); jit != in.ObjEnd(); jit++){ | ||||
|     if (out.isMember(jit->first)){ | ||||
|       if (jit->second != out[jit->first]){ | ||||
|         Log("CONF", std::string("Updated configuration value ")+jit->first); | ||||
|       } | ||||
|     }else{ | ||||
|         Log("CONF", std::string("New configuration value ")+jit.memberName()); | ||||
|       } | ||||
|     } | ||||
|     if (out.isObject() && (out.size() > 0)){ | ||||
|       for (Json::ValueIterator jit = out.begin(); jit != out.end(); jit++){ | ||||
|         if (!in.isMember(jit.memberName())){ | ||||
|           Log("CONF", std::string("Deleted configuration value ")+jit.memberName()); | ||||
|       Log("CONF", std::string("New configuration value ")+jit->first); | ||||
|     } | ||||
|   } | ||||
|   for (JSON::ObjIter jit = out.ObjBegin(); jit != out.ObjEnd(); jit++){ | ||||
|     if (!in.isMember(jit->first)){ | ||||
|       Log("CONF", std::string("Deleted configuration value ")+jit->first); | ||||
|     } | ||||
|   } | ||||
|   out = in; | ||||
|   out["version"] = TOSTRING(VERSION); | ||||
| } | ||||
| 
 | ||||
| bool streamsEqual(Json::Value & one, Json::Value & two){ | ||||
|   if (one["channel"]["URL"].asString() != two["channel"]["URL"].asString()){return false;} | ||||
|   if (one["preset"]["cmd"].asString() != two["preset"]["cmd"].asString()){return false;} | ||||
| bool streamsEqual(JSON::Value & one, JSON::Value & two){ | ||||
|   if (one["channel"]["URL"] != two["channel"]["URL"]){return false;} | ||||
|   if (one["preset"]["cmd"] != two["preset"]["cmd"]){return false;} | ||||
|   return true; | ||||
| } | ||||
| 
 | ||||
| void startStream(std::string name, Json::Value & data){ | ||||
| void startStream(std::string name, JSON::Value & data){ | ||||
|   Log("BUFF", "(re)starting stream buffer "+name); | ||||
|   std::string URL = data["channel"]["URL"].asString(); | ||||
|   std::string preset = data["preset"]["cmd"].asString(); | ||||
|   std::string URL = data["channel"]["URL"]; | ||||
|   std::string preset = data["preset"]["cmd"]; | ||||
|   std::string cmd1, cmd2; | ||||
|   if (URL.substr(0, 4) == "push"){ | ||||
|     std::string pusher = URL.substr(7); | ||||
|  | @ -325,41 +215,37 @@ void startStream(std::string name, Json::Value & data){ | |||
|   } | ||||
| } | ||||
| 
 | ||||
| void CheckAllStreams(Json::Value & data){ | ||||
| void CheckAllStreams(JSON::Value & data){ | ||||
|   unsigned int currTime = time(0); | ||||
|   for (Json::ValueIterator jit = data.begin(); jit != data.end(); jit++){ | ||||
|     if (!Util::Procs::isActive(jit.memberName())){ | ||||
|       startStream(jit.memberName(), data[jit.memberName()]); | ||||
|   for (JSON::ObjIter jit = data.ObjBegin(); jit != data.ObjEnd(); jit++){ | ||||
|     if (!Util::Procs::isActive(jit->first)){ | ||||
|       startStream(jit->first, jit->second); | ||||
|     } | ||||
|     if (currTime - lastBuffer[jit.memberName()] > 5){ | ||||
|       data[jit.memberName()]["online"] = 0; | ||||
|     if (currTime - lastBuffer[jit->first] > 5){ | ||||
|       jit->second["online"] = 0; | ||||
|     }else{ | ||||
|       data[jit.memberName()]["online"] = 1; | ||||
|       jit->second["online"] = 1; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| void CheckStreams(Json::Value & in, Json::Value & out){ | ||||
|   if (in.isObject() && (in.size() > 0)){ | ||||
|     for (Json::ValueIterator jit = in.begin(); jit != in.end(); jit++){ | ||||
|       if (out.isObject() && out.isMember(jit.memberName())){ | ||||
|         if (!streamsEqual(in[jit.memberName()], out[jit.memberName()])){ | ||||
|           Log("STRM", std::string("Updated stream ")+jit.memberName()); | ||||
|           Util::Procs::Stop(jit.memberName()); | ||||
|           startStream(jit.memberName(), in[jit.memberName()]); | ||||
| void CheckStreams(JSON::Value & in, JSON::Value & out){ | ||||
|   for (JSON::ObjIter jit = in.ObjBegin(); jit != in.ObjEnd(); jit++){ | ||||
|     if (out.isMember(jit->first)){ | ||||
|       if (!streamsEqual(jit->second, out[jit->first])){ | ||||
|         Log("STRM", std::string("Updated stream ")+jit->first); | ||||
|         Util::Procs::Stop(jit->first); | ||||
|         startStream(jit->first, jit->second); | ||||
|       } | ||||
|     }else{ | ||||
|         Log("STRM", std::string("New stream ")+jit.memberName()); | ||||
|         startStream(jit.memberName(), in[jit.memberName()]); | ||||
|       Log("STRM", std::string("New stream ")+jit->first); | ||||
|       startStream(jit->first, jit->second); | ||||
|     } | ||||
|   } | ||||
|   } | ||||
|   if (out.isObject() && (out.size() > 0)){ | ||||
|     for (Json::ValueIterator jit = out.begin(); jit != out.end(); jit++){ | ||||
|       if (!in.isMember(jit.memberName())){ | ||||
|         Log("STRM", std::string("Deleted stream ")+jit.memberName()); | ||||
|         Util::Procs::Stop(jit.memberName()); | ||||
|       } | ||||
|   for (JSON::ObjIter jit = out.ObjBegin(); jit != out.ObjEnd(); jit++){ | ||||
|     if (!in.isMember(jit->first)){ | ||||
|       Log("STRM", std::string("Deleted stream ")+jit->first); | ||||
|       Util::Procs::Stop(jit->first); | ||||
|     } | ||||
|   } | ||||
|   out = in; | ||||
|  | @ -376,7 +262,6 @@ int main(int argc, char ** argv){ | |||
|   sigaction(SIGTERM, &new_action, NULL); | ||||
|   sigaction(SIGPIPE, &new_action, NULL); | ||||
| 
 | ||||
|   RSA_Load(); // Load GearBox public key
 | ||||
|   Util::Config C; | ||||
|   C.confsection = "API"; | ||||
|   C.parseArgs(argc, argv); | ||||
|  | @ -392,15 +277,11 @@ int main(int argc, char ** argv){ | |||
|   Socket::Connection Incoming; | ||||
|   std::vector< ConnectedUser > users; | ||||
|   std::vector<Socket::Connection> buffers; | ||||
|   Json::Value Request = Json::Value(Json::objectValue); | ||||
|   Json::Value Response = Json::Value(Json::objectValue); | ||||
|   Json::Reader JsonParse; | ||||
|   JSON::Value Request; | ||||
|   JSON::Value Response; | ||||
|   std::string jsonp; | ||||
|   ConnectedUser * uplink = 0; | ||||
|   JsonParse.parse(ReadFile("config.json"), Storage, false); | ||||
|   if (!Storage.isMember("config")){Storage["config"] = Json::Value(Json::objectValue);} | ||||
|   if (!Storage.isMember("log")){Storage["log"] = Json::Value(Json::arrayValue);} | ||||
|   if (!Storage.isMember("statistics")){Storage["statistics"] = Json::Value(Json::objectValue);} | ||||
|   Storage = JSON::fromString(ReadFile("config.json")); | ||||
|   while (API_Socket.connected()){ | ||||
|     usleep(100000); //sleep for 100 ms - prevents 100% CPU time
 | ||||
| 
 | ||||
|  | @ -433,14 +314,14 @@ int main(int argc, char ** argv){ | |||
|         } | ||||
|       } | ||||
|       if (gotUplink){ | ||||
|         Response.clear(); //make sure no data leaks from previous requests
 | ||||
|         Response.null(); //make sure no data leaks from previous requests
 | ||||
|         Response["config"] = Storage["config"]; | ||||
|         Response["streams"] = Storage["streams"]; | ||||
|         Response["log"] = Storage["log"]; | ||||
|         Response["statistics"] = Storage["statistics"]; | ||||
|         Response["now"] = (unsigned int)lastuplink; | ||||
|         uplink->H.Clean(); | ||||
|         uplink->H.SetBody("command="+HTTP::Parser::urlencode(Response.toStyledString())); | ||||
|         uplink->H.SetBody("command="+HTTP::Parser::urlencode(Response.toString())); | ||||
|         uplink->H.BuildRequest(); | ||||
|         uplink->writebuffer += uplink->H.BuildResponse("200", "OK"); | ||||
|         uplink->H.Clean(); | ||||
|  | @ -465,24 +346,18 @@ int main(int argc, char ** argv){ | |||
|         if (it->Received() != ""){ | ||||
|           size_t newlines = it->Received().find("\n\n"); | ||||
|           while (newlines != std::string::npos){ | ||||
|             if (JsonParse.parse(it->Received().substr(0, newlines), Request, false)){ | ||||
|             Request = it->Received().substr(0, newlines); | ||||
|             if (Request.isMember("totals") && Request["totals"].isMember("buffer")){ | ||||
|                 std::string thisbuffer = Request["totals"]["buffer"].asString(); | ||||
|               std::string thisbuffer = Request["totals"]["buffer"]; | ||||
|               lastBuffer[thisbuffer] = time(0); | ||||
|               Storage["statistics"][thisbuffer]["curr"] = Request["curr"]; | ||||
|               std::stringstream st; | ||||
|                 st << Request["totals"]["now"].asUInt(); | ||||
|               st << (long long int)Request["totals"]["now"]; | ||||
|               std::string nowstr = st.str(); | ||||
|               Storage["statistics"][thisbuffer]["totals"][nowstr] = Request["totals"]; | ||||
|                 if (!Storage["statistics"][thisbuffer].isMember("log")){ | ||||
|                   Storage["statistics"][thisbuffer]["log"] = Json::Value(Json::arrayValue); | ||||
|               for (JSON::ObjIter jit = Request["log"].ObjBegin(); jit != Request["log"].ObjEnd(); jit++){ | ||||
|                 Storage["statistics"][thisbuffer]["log"].append(jit->second); | ||||
|               } | ||||
|                 for (Json::ValueIterator jit = Request["log"].begin(); jit != Request["log"].end(); jit++){ | ||||
|                   Storage["statistics"][thisbuffer]["log"].append(Request["log"][jit.memberName()]); | ||||
|                 } | ||||
|               } | ||||
|             }else{ | ||||
|               Log("STAT", "Failed to parse stats info from buffer: "+it->Received().substr(0, newlines)); | ||||
|             } | ||||
|             it->Received().erase(0, newlines+2); | ||||
|             newlines = it->Received().find("\n\n"); | ||||
|  | @ -501,15 +376,12 @@ int main(int argc, char ** argv){ | |||
|           it->C.iwrite(it->writebuffer); | ||||
|         } | ||||
|         if (it->H.Read(it->C)){ | ||||
|           Response.clear(); //make sure no data leaks from previous requests
 | ||||
|           Response.null(); //make sure no data leaks from previous requests
 | ||||
|           if (it->clientMode){ | ||||
|             // In clientMode, requests are reversed. These are connections we initiated to GearBox.
 | ||||
|             // They are assumed to be authorized, but authorization to gearbox is still done.
 | ||||
|             // This authorization uses the compiled-in username and password (account).
 | ||||
|             if (!JsonParse.parse(it->H.body, Request, false)){ | ||||
|               Log("UPLK", "Failed to parse body JSON: "+it->H.body); | ||||
|               Response["authorize"]["status"] = "INVALID"; | ||||
|             }else{ | ||||
|             Request = JSON::fromString(it->H.body); | ||||
|             if (Request["authorize"]["status"] != "OK"){ | ||||
|               if (Request["authorize"].isMember("challenge")){ | ||||
|                 it->logins++; | ||||
|  | @ -522,10 +394,10 @@ int main(int argc, char ** argv){ | |||
|                   Response["log"] = Storage["log"]; | ||||
|                   Response["statistics"] = Storage["statistics"]; | ||||
|                   Response["authorize"]["username"] = TOSTRING(COMPILED_USERNAME); | ||||
|                     Log("UPLK", "Responding to login challenge: " + Request["authorize"]["challenge"].asString()); | ||||
|                     Response["authorize"]["password"] = md5(TOSTRING(COMPILED_PASSWORD) + Request["authorize"]["challenge"].asString()); | ||||
|                   Log("UPLK", "Responding to login challenge: " + (std::string)Request["authorize"]["challenge"]); | ||||
|                   Response["authorize"]["password"] = md5(TOSTRING(COMPILED_PASSWORD) + (std::string)Request["authorize"]["challenge"]); | ||||
|                   it->H.Clean(); | ||||
|                     it->H.SetBody("command="+HTTP::Parser::urlencode(Response.toStyledString())); | ||||
|                   it->H.SetBody("command="+HTTP::Parser::urlencode(Response.toString())); | ||||
|                   it->H.BuildRequest(); | ||||
|                   it->writebuffer += it->H.BuildResponse("200", "OK"); | ||||
|                   it->H.Clean(); | ||||
|  | @ -536,17 +408,13 @@ int main(int argc, char ** argv){ | |||
|               if (Request.isMember("config")){CheckConfig(Request["config"], Storage["config"]);} | ||||
|               if (Request.isMember("streams")){CheckStreams(Request["streams"], Storage["streams"]);} | ||||
|               if (Request.isMember("clearstatlogs")){ | ||||
|                   Storage["log"].clear(); | ||||
|                   Storage["statistics"].clear(); | ||||
|                 } | ||||
|                 Storage["log"].null(); | ||||
|                 Storage["statistics"].null(); | ||||
|               } | ||||
|             } | ||||
|           }else{ | ||||
|             if (!JsonParse.parse(it->H.GetVar("command"), Request, false)){ | ||||
|               Log("HTTP", "Failed to parse command JSON: "+it->H.GetVar("command")); | ||||
|               Response["authorize"]["status"] = "INVALID"; | ||||
|             }else{ | ||||
|               std::cout << "Request: " << Request.toStyledString() << std::endl; | ||||
|             Request = JSON::fromString(it->H.GetVar("command")); | ||||
|             std::cout << "Request: " << Request.toString() << std::endl; | ||||
|             Authorize(Request, Response, (*it)); | ||||
|             if (it->Authorized){ | ||||
|               //Parse config and streams from the request.
 | ||||
|  | @ -557,15 +425,14 @@ int main(int argc, char ** argv){ | |||
|               Response["config"] = Storage["config"]; | ||||
|               Response["streams"] = Storage["streams"]; | ||||
|               //add required data to the current unix time to the config, for syncing reasons
 | ||||
|                 Response["config"]["time"] = (Json::Value::UInt)time(0); | ||||
|               Response["config"]["time"] = (long long int)time(0); | ||||
|               if (!Response["config"].isMember("serverid")){Response["config"]["serverid"] = "";} | ||||
|               //sent any available logs and statistics
 | ||||
|               Response["log"] = Storage["log"]; | ||||
|               Response["statistics"] = Storage["statistics"]; | ||||
|               //clear log and statistics to prevent useless data transfer
 | ||||
|                 Storage["log"].clear(); | ||||
|                 Storage["statistics"].clear(); | ||||
|               } | ||||
|               Storage["log"].null(); | ||||
|               Storage["statistics"].null(); | ||||
|             } | ||||
|             jsonp = ""; | ||||
|             if (it->H.GetVar("callback") != ""){jsonp = it->H.GetVar("callback");} | ||||
|  | @ -574,9 +441,9 @@ int main(int argc, char ** argv){ | |||
|             it->H.protocol = "HTTP/1.0"; | ||||
|             it->H.SetHeader("Content-Type", "text/javascript"); | ||||
|             if (jsonp == ""){ | ||||
|               it->H.SetBody(Response.toStyledString()+"\n\n"); | ||||
|               it->H.SetBody(Response.toString()+"\n\n"); | ||||
|             }else{ | ||||
|               it->H.SetBody(jsonp+"("+Response.toStyledString()+");\n\n"); | ||||
|               it->H.SetBody(jsonp+"("+Response.toString()+");\n\n"); | ||||
|             } | ||||
|             it->writebuffer += it->H.BuildResponse("200", "OK"); | ||||
|             it->H.Clean(); | ||||
|  | @ -586,7 +453,7 @@ int main(int argc, char ** argv){ | |||
|     } | ||||
|   } | ||||
|   Util::Procs::StopAll(); | ||||
|   WriteFile("config.json", Storage.toStyledString()); | ||||
|   WriteFile("config.json", Storage.toString()); | ||||
|   std::cout << "Killed all processes, wrote config to disk. Exiting." << std::endl; | ||||
|   return 0; | ||||
| } | ||||
|  |  | |||
							
								
								
									
										5
									
								
								Makefile
									
										
									
									
									
								
							
							
						
						
									
										5
									
								
								Makefile
									
										
									
									
									
								
							|  | @ -6,29 +6,34 @@ client-debug: | |||
| 	cd Connector_RTMP; $(MAKE) | ||||
| 	cd Connector_RAW; $(MAKE) | ||||
| 	cd Buffer; $(MAKE) | ||||
| 	cd DDV_Controller; $(MAKE) | ||||
| client: client-debug | ||||
| client-clean: | ||||
| 	cd Connector_HTTP; $(MAKE) clean | ||||
| 	cd Connector_RTMP; $(MAKE) clean | ||||
| 	cd Connector_RAW; $(MAKE) clean | ||||
| 	cd Buffer; $(MAKE) clean | ||||
| 	cd DDV_Controller; $(MAKE) clean | ||||
| clean: client-clean | ||||
| client-release: | ||||
| 	cd Connector_HTTP; $(MAKE) DEBUG=0 OPTIMIZE=-O2 | ||||
| 	cd Connector_RTMP; $(MAKE) DEBUG=0 OPTIMIZE=-O2 | ||||
| 	cd Connector_RAW; $(MAKE) DEBUG=0 OPTIMIZE=-O2 | ||||
| 	cd Buffer; $(MAKE) DEBUG=0 OPTIMIZE=-O2 | ||||
| 	cd DDV_Controller; $(MAKE) DEBUG=0 OPTIMIZE=-O2 | ||||
| release: client-release | ||||
| release-install: client-clean client-release | ||||
| 	cd Connector_RTMP; $(MAKE) install | ||||
| 	cd Connector_HTTP; $(MAKE) install | ||||
| 	cd Connector_RAW; $(MAKE) install | ||||
| 	cd Buffer; $(MAKE) install | ||||
| 	cd DDV_Controller; $(MAKE) install | ||||
| debug-install: client-clean client-debug | ||||
| 	cd Connector_RTMP; $(MAKE) install | ||||
| 	cd Connector_HTTP; $(MAKE) install | ||||
| 	cd Connector_RAW; $(MAKE) install | ||||
| 	cd Buffer; $(MAKE) install | ||||
| 	cd DDV_Controller; $(MAKE) install | ||||
| docs: | ||||
| 	doxygen ./Doxyfile > /dev/null | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										45
									
								
								util/auth.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								util/auth.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,45 @@ | |||
| #include "auth.h" | ||||
| #include "base64.h" | ||||
| 
 | ||||
| static unsigned char __gbv2keypub_der[] = { | ||||
|   0x30, 0x82, 0x01, 0x22, 0x30, 0x0d, 0x06, 0x09, 0x2a, 0x86, 0x48, 0x86, | ||||
|   0xf7, 0x0d, 0x01, 0x01, 0x01, 0x05, 0x00, 0x03, 0x82, 0x01, 0x0f, 0x00, | ||||
|   0x30, 0x82, 0x01, 0x0a, 0x02, 0x82, 0x01, 0x01, 0x00, 0xe5, 0xd7, 0x9c, | ||||
|   0x7d, 0x73, 0xc6, 0xe6, 0xfb, 0x35, 0x7e, 0xd7, 0x57, 0x99, 0x07, 0xdb, | ||||
|   0x99, 0x70, 0xc9, 0xd0, 0x3e, 0x53, 0x57, 0x3c, 0x1e, 0x55, 0xda, 0x0f, | ||||
|   0x69, 0xbf, 0x26, 0x79, 0xc7, 0xb6, 0xdd, 0x8e, 0x83, 0x32, 0x65, 0x74, | ||||
|   0x0d, 0x74, 0x48, 0x42, 0x49, 0x22, 0x52, 0x58, 0x56, 0xc3, 0xe4, 0x49, | ||||
|   0x5d, 0xac, 0x6a, 0x94, 0xb1, 0x64, 0x14, 0xbf, 0x4d, 0xd5, 0xd7, 0x3a, | ||||
|   0xca, 0x5c, 0x1e, 0x6f, 0x42, 0x30, 0xac, 0x29, 0xaa, 0xa0, 0x85, 0xd2, | ||||
|   0x16, 0xa2, 0x8e, 0x89, 0x12, 0xc4, 0x92, 0x06, 0xea, 0xed, 0x48, 0xf6, | ||||
|   0xdb, 0xed, 0x4f, 0x62, 0x6c, 0xfa, 0xcf, 0xc2, 0xb9, 0x8d, 0x04, 0xb2, | ||||
|   0xba, 0x63, 0xc9, 0xcc, 0xee, 0x23, 0x64, 0x46, 0x14, 0x12, 0xc8, 0x38, | ||||
|   0x67, 0x69, 0x6b, 0xaf, 0xd1, 0x7c, 0xb1, 0xb5, 0x79, 0xe4, 0x4e, 0x3a, | ||||
|   0xa7, 0xe8, 0x28, 0x89, 0x25, 0xc0, 0xd0, 0xd8, 0xc7, 0xd2, 0x26, 0xaa, | ||||
|   0xf5, 0xbf, 0x36, 0x55, 0x01, 0x89, 0x58, 0x1f, 0x1e, 0xf5, 0xa5, 0x42, | ||||
|   0x8f, 0x60, 0x2e, 0xc2, 0xd8, 0x21, 0x0b, 0x6c, 0x8d, 0xbb, 0x72, 0xf2, | ||||
|   0x19, 0x30, 0xe3, 0x4c, 0x3e, 0x80, 0xe7, 0xf2, 0xe3, 0x89, 0x4f, 0xd4, | ||||
|   0xee, 0x96, 0x3e, 0x4a, 0x9b, 0xe5, 0x16, 0x01, 0xf1, 0x98, 0xc9, 0x0b, | ||||
|   0xd6, 0xdf, 0x8a, 0x64, 0x47, 0xc4, 0x44, 0xcc, 0x92, 0x69, 0x28, 0xee, | ||||
|   0x7d, 0xac, 0xdc, 0x30, 0x56, 0x3a, 0xe7, 0xbc, 0xba, 0x45, 0x16, 0x2c, | ||||
|   0x4c, 0x46, 0x6b, 0x2b, 0x20, 0xfb, 0x3d, 0x20, 0x35, 0xbb, 0x48, 0x49, | ||||
|   0x13, 0x65, 0xc9, 0x9a, 0x38, 0x10, 0x84, 0x1a, 0x8c, 0xc9, 0xd7, 0xde, | ||||
|   0x07, 0x10, 0x5a, 0xfb, 0xb4, 0x95, 0xae, 0x18, 0xf2, 0xe3, 0x15, 0xe8, | ||||
|   0xad, 0x7e, 0xe5, 0x3c, 0xa8, 0x47, 0x85, 0xd6, 0x1f, 0x54, 0xb5, 0xa3, | ||||
|   0x79, 0x02, 0x03, 0x01, 0x00, 0x01 | ||||
| }; ///< The GBv2 public key file.
 | ||||
| static unsigned int __gbv2keypub_der_len = 294; ///< Length of GBv2 public key data
 | ||||
| 
 | ||||
| /// Attempts to load the GBv2 public key.
 | ||||
| Auth::Auth(){ | ||||
|   const unsigned char * key = __gbv2keypub_der; | ||||
|   pubkey = d2i_RSAPublicKey(0, &key, __gbv2keypub_der_len); | ||||
| } | ||||
| 
 | ||||
| /// Attempts to verify RSA signature using the public key.
 | ||||
| /// Assumes basesign argument is base64 encoded RSA signature for data.
 | ||||
| /// Returns true if the data could be verified, false otherwise.
 | ||||
| bool Auth::PubKey_Check(std::string & data, std::string basesign){ | ||||
|   std::string sign = Base64::decode(basesign); | ||||
|   return (RSA_verify(NID_md5, (unsigned char*)data.c_str(), data.size(), (unsigned char*)sign.c_str(), sign.size(), pubkey) == 1); | ||||
| } | ||||
							
								
								
									
										11
									
								
								util/auth.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								util/auth.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | |||
| #include <string> | ||||
| #include <openssl/rsa.h> | ||||
| #include <openssl/x509.h> | ||||
| 
 | ||||
| class Auth{ | ||||
|   private: | ||||
|     RSA * pubkey; ///< Holds the public key.
 | ||||
|   public: | ||||
|     Auth(); ///< Attempts to load the GBv2 public key.
 | ||||
|     bool PubKey_Check(std::string & data, std::string basesign); ///< Attempts to verify RSA signature using the public key.
 | ||||
| }; | ||||
							
								
								
									
										64
									
								
								util/base64.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								util/base64.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | |||
| #include "base64.h" | ||||
| 
 | ||||
| /// Needed for base64_encode function
 | ||||
| const std::string Base64::chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | ||||
| 
 | ||||
|   /// Helper for base64_decode function
 | ||||
| inline bool Base64::is_base64(unsigned char c) { | ||||
|   return (isalnum(c) || (c == '+') || (c == '/')); | ||||
| } | ||||
| 
 | ||||
| /// Used to base64 encode data. Input is the plaintext as std::string, output is the encoded data as std::string.
 | ||||
| /// \param input Plaintext data to encode.
 | ||||
| /// \returns Base64 encoded data.
 | ||||
| std::string Base64::encode(std::string const input) { | ||||
|   std::string ret; | ||||
|   unsigned int in_len = input.size(); | ||||
|   char quad[4], triple[3]; | ||||
|   unsigned int i, x, n = 3; | ||||
|   for (x = 0; x < in_len; x = x + 3){ | ||||
|     if ((in_len - x) / 3 == 0){n = (in_len - x) % 3;} | ||||
|     for (i=0; i < 3; i++){triple[i] = '0';} | ||||
|     for (i=0; i < n; i++){triple[i] = input[x + i];} | ||||
|     quad[0] = chars[(triple[0] & 0xFC) >> 2]; // FC = 11111100
 | ||||
|     quad[1] = chars[((triple[0] & 0x03) << 4) | ((triple[1] & 0xF0) >> 4)]; // 03 = 11
 | ||||
|     quad[2] = chars[((triple[1] & 0x0F) << 2) | ((triple[2] & 0xC0) >> 6)]; // 0F = 1111, C0=11110
 | ||||
|     quad[3] = chars[triple[2] & 0x3F]; // 3F = 111111
 | ||||
|     if (n < 3){quad[3] = '=';} | ||||
|     if (n < 2){quad[2] = '=';} | ||||
|     for(i=0; i < 4; i++){ret += quad[i];} | ||||
|   } | ||||
|   return ret; | ||||
| }//base64_encode
 | ||||
| 
 | ||||
| /// Used to base64 decode data. Input is the encoded data as std::string, output is the plaintext data as std::string.
 | ||||
| /// \param input Base64 encoded data to decode.
 | ||||
| /// \returns Plaintext decoded data.
 | ||||
| std::string Base64::decode(std::string const& encoded_string) { | ||||
|   int in_len = encoded_string.size(); | ||||
|   int i = 0; | ||||
|   int j = 0; | ||||
|   int in_ = 0; | ||||
|   unsigned char char_array_4[4], char_array_3[3]; | ||||
|   std::string ret; | ||||
|   while (in_len-- && ( encoded_string[in_] != '=') && is_base64(encoded_string[in_])) { | ||||
|     char_array_4[i++] = encoded_string[in_]; in_++; | ||||
|     if (i ==4) { | ||||
|       for (i = 0; i <4; i++){char_array_4[i] = chars.find(char_array_4[i]);} | ||||
|       char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); | ||||
|       char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); | ||||
|       char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; | ||||
|       for (i = 0; (i < 3); i++){ret += char_array_3[i];} | ||||
|       i = 0; | ||||
|     } | ||||
|   } | ||||
|   if (i) { | ||||
|     for (j = i; j <4; j++){char_array_4[j] = 0;} | ||||
|     for (j = 0; j <4; j++){char_array_4[j] = chars.find(char_array_4[j]);} | ||||
|     char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); | ||||
|     char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); | ||||
|     char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; | ||||
|     for (j = 0; (j < i - 1); j++) ret += char_array_3[j]; | ||||
|   } | ||||
|   return ret; | ||||
| } | ||||
							
								
								
									
										11
									
								
								util/base64.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								util/base64.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | |||
| #include <string> | ||||
| 
 | ||||
| /// Holds base64 decoding and encoding functions.
 | ||||
| class Base64{ | ||||
|   private: | ||||
|     static const std::string chars; | ||||
|     static inline bool is_base64(unsigned char c); | ||||
|   public: | ||||
|     static std::string encode(std::string const input); | ||||
|     static std::string decode(std::string const& encoded_string); | ||||
| }; | ||||
							
								
								
									
										150
									
								
								util/config.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								util/config.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,150 @@ | |||
| /// \file config.cpp
 | ||||
| /// Contains generic functions for managing configuration.
 | ||||
| 
 | ||||
| #include "config.h" | ||||
| #include <string.h> | ||||
| #include <sys/types.h> | ||||
| #include <signal.h> | ||||
| 
 | ||||
| #ifdef __FreeBSD__ | ||||
| #include <sys/wait.h> | ||||
| #else | ||||
| #include <wait.h> | ||||
| #endif | ||||
| #include <errno.h> | ||||
| #include <iostream> | ||||
| #include <sys/types.h> | ||||
| #include <fcntl.h> | ||||
| #include <pwd.h> | ||||
| #include <getopt.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <fstream> | ||||
| 
 | ||||
| /// Creates a new configuration manager.
 | ||||
| Util::Config::Config(){ | ||||
|   listen_port = 4242; | ||||
|   daemon_mode = true; | ||||
|   interface = "0.0.0.0"; | ||||
|   configfile = "/etc/ddvtech.conf"; | ||||
|   username = "root"; | ||||
|   ignore_daemon = false; | ||||
|   ignore_interface = false; | ||||
|   ignore_port = false; | ||||
|   ignore_user = false; | ||||
| } | ||||
| 
 | ||||
| /// Parses commandline arguments.
 | ||||
| /// Calls exit if an unknown option is encountered, printing a help message.
 | ||||
| /// confsection must be either already set or never be set at all when this function is called.
 | ||||
| /// In other words: do not change confsection after calling this function.
 | ||||
| void Util::Config::parseArgs(int argc, char ** argv){ | ||||
|   int opt = 0; | ||||
|   static const char *optString = "ndvp:i:u:c:h?"; | ||||
|   static const struct option longOpts[] = { | ||||
|     {"help",0,0,'h'}, | ||||
|     {"port",1,0,'p'}, | ||||
|     {"interface",1,0,'i'}, | ||||
|     {"username",1,0,'u'}, | ||||
|     {"no-daemon",0,0,'n'}, | ||||
|     {"daemon",0,0,'d'}, | ||||
|     {"configfile",1,0,'c'}, | ||||
|     {"version",0,0,'v'} | ||||
|   }; | ||||
|   while ((opt = getopt_long(argc, argv, optString, longOpts, 0)) != -1){ | ||||
|     switch (opt){ | ||||
|       case 'p': listen_port = atoi(optarg); ignore_port = true; break; | ||||
|       case 'i': interface = optarg; ignore_interface = true; break; | ||||
|       case 'n': daemon_mode = false; ignore_daemon = true; break; | ||||
|       case 'd': daemon_mode = true; ignore_daemon = true; break; | ||||
|       case 'c': configfile = optarg; break; | ||||
|       case 'u': username = optarg; ignore_user = true; break; | ||||
|       case 'v': | ||||
|         printf("%s\n", TOSTRING(VERSION)); | ||||
|         exit(1); | ||||
|         break; | ||||
|       case 'h': | ||||
|       case '?': | ||||
|         std::string doingdaemon = "true"; | ||||
|         if (!daemon_mode){doingdaemon = "false";} | ||||
|         if (confsection == ""){ | ||||
|           printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -u[sername] VAL\n"); | ||||
|           printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), username.c_str()); | ||||
|         }else{ | ||||
|           printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -c[onfigfile] VAL, -u[sername] VAL\n"); | ||||
|           printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  configfile: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), configfile.c_str(), username.c_str()); | ||||
|           printf("Username root means no change to UID, no matter what the UID is.\n"); | ||||
|           printf("If the configfile exists, it is always loaded first. Commandline settings then overwrite the config file.\n"); | ||||
|           printf("\nThis process takes it directives from the %s section of the configfile.\n", confsection.c_str()); | ||||
|         } | ||||
|         printf("This is %s version %s\n", argv[0], TOSTRING(VERSION)); | ||||
|         exit(1); | ||||
|         break; | ||||
|     } | ||||
|   }//commandline options parser
 | ||||
| } | ||||
| 
 | ||||
| /// Parses the configuration file at configfile, if it exists.
 | ||||
| /// Assumes confsection is set.
 | ||||
| void Util::Config::parseFile(){ | ||||
|   std::ifstream conf(configfile.c_str(), std::ifstream::in); | ||||
|   std::string tmpstr; | ||||
|   bool acc_comm = false; | ||||
|   size_t foundeq; | ||||
|   if (conf.fail()){ | ||||
|     #if DEBUG >= 3 | ||||
|     fprintf(stderr, "Configuration file %s not found - using build-in defaults...\n", configfile.c_str()); | ||||
|     #endif | ||||
|   }else{ | ||||
|     while (conf.good()){ | ||||
|       getline(conf, tmpstr); | ||||
|       if (tmpstr[0] == '['){//new section? check if we care.
 | ||||
|         if (tmpstr == confsection){acc_comm = true;}else{acc_comm = false;} | ||||
|       }else{ | ||||
|         if (!acc_comm){break;}//skip all lines in this section if we do not care about it
 | ||||
|         foundeq = tmpstr.find('='); | ||||
|         if (foundeq != std::string::npos){ | ||||
|           if ((tmpstr.substr(0, foundeq) == "port") && !ignore_port){listen_port = atoi(tmpstr.substr(foundeq+1).c_str());} | ||||
|           if ((tmpstr.substr(0, foundeq) == "interface") && !ignore_interface){interface = tmpstr.substr(foundeq+1);} | ||||
|           if ((tmpstr.substr(0, foundeq) == "username") && !ignore_user){username = tmpstr.substr(foundeq+1);} | ||||
|           if ((tmpstr.substr(0, foundeq) == "daemon") && !ignore_daemon){daemon_mode = true;} | ||||
|           if ((tmpstr.substr(0, foundeq) == "nodaemon") && !ignore_daemon){daemon_mode = false;} | ||||
|         }//found equals sign
 | ||||
|       }//section contents
 | ||||
|     }//configfile line loop
 | ||||
|   }//configuration
 | ||||
| } | ||||
| 
 | ||||
| /// Sets the current process' running user
 | ||||
| void Util::setUser(std::string username){ | ||||
|   if (username != "root"){ | ||||
|     struct passwd * user_info = getpwnam(username.c_str()); | ||||
|     if (!user_info){ | ||||
|       #if DEBUG >= 1 | ||||
|       fprintf(stderr, "Error: could not setuid %s: could not get PID\n", username.c_str()); | ||||
|       #endif | ||||
|       return; | ||||
|     }else{ | ||||
|       if (setuid(user_info->pw_uid) != 0){ | ||||
|         #if DEBUG >= 1 | ||||
|         fprintf(stderr, "Error: could not setuid %s: not allowed\n", username.c_str()); | ||||
|         #endif | ||||
|       }else{ | ||||
|         #if DEBUG >= 3 | ||||
|         fprintf(stderr, "Changed user to %s\n", username.c_str()); | ||||
|         #endif | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// Will turn the current process into a daemon.
 | ||||
| /// Works by calling daemon(1,0):
 | ||||
| /// Does not change directory to root.
 | ||||
| /// Does redirect output to /dev/null
 | ||||
| void Util::Daemonize(){ | ||||
|   #if DEBUG >= 3 | ||||
|   fprintf(stderr, "Going into background mode...\n"); | ||||
|   #endif | ||||
|   daemon(1, 0); | ||||
| } | ||||
							
								
								
									
										37
									
								
								util/config.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								util/config.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,37 @@ | |||
| /// \file config.h
 | ||||
| /// Contains generic function headers for managing configuration.
 | ||||
| 
 | ||||
| #include <string> | ||||
| 
 | ||||
| #define STRINGIFY(x) #x | ||||
| #define TOSTRING(x) STRINGIFY(x) | ||||
| 
 | ||||
| /// Contains utility code, not directly related to streaming media
 | ||||
| namespace Util{ | ||||
| 
 | ||||
|   /// Deals with parsing configuration from files or commandline options.
 | ||||
|   class Config{ | ||||
|   private: | ||||
|     bool ignore_daemon; | ||||
|     bool ignore_interface; | ||||
|     bool ignore_port; | ||||
|     bool ignore_user; | ||||
|   public: | ||||
|     std::string confsection; | ||||
|     std::string configfile; | ||||
|     bool daemon_mode; | ||||
|     std::string interface; | ||||
|     int listen_port; | ||||
|     std::string username; | ||||
|     Config(); | ||||
|     void parseArgs(int argc, char ** argv); | ||||
|     void parseFile(); | ||||
|   }; | ||||
| 
 | ||||
|   /// Will set the active user to the named username.
 | ||||
|   void setUser(std::string user); | ||||
| 
 | ||||
|   /// Will turn the current process into a daemon.
 | ||||
|   void Daemonize(); | ||||
| 
 | ||||
| }; | ||||
							
								
								
									
										421
									
								
								util/json.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										421
									
								
								util/json.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,421 @@ | |||
| /// \file json.cpp Holds all JSON-related code.
 | ||||
| 
 | ||||
| #include "json.h" | ||||
| #include <sstream> | ||||
| 
 | ||||
| int JSON::Value::c2hex(int c){ | ||||
|   if (c >= '0' && c <= '9') return c - '0'; | ||||
|   if (c >= 'a' && c <= 'f') return c - 'a' + 10; | ||||
|   if (c >= 'A' && c <= 'F') return c - 'A' + 10; | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string JSON::Value::read_string(int separator, std::istream & fromstream){ | ||||
|   std::string out; | ||||
|   bool escaped = false; | ||||
|   while (fromstream.good()){ | ||||
|     int c = fromstream.get(); | ||||
|     if (c == '\\'){ | ||||
|       escaped = true; | ||||
|       continue; | ||||
|     } | ||||
|     if (escaped){ | ||||
|       switch (c){ | ||||
|         case 'b': out += '\b'; break; | ||||
|         case 'f': out += '\f'; break; | ||||
|         case 'n': out += '\n'; break; | ||||
|         case 'r': out += '\r'; break; | ||||
|         case 't': out += '\t'; break; | ||||
|         case 'u':{ | ||||
|            int d1 = fromstream.get(); | ||||
|            int d2 = fromstream.get(); | ||||
|            int d3 = fromstream.get(); | ||||
|            int d4 = fromstream.get(); | ||||
|            c = c2hex(d4) + (c2hex(d3) << 4) + (c2hex(d2) << 8) + (c2hex(d1) << 16); | ||||
|         } | ||||
|         default: | ||||
|           out += (char)c; | ||||
|           break; | ||||
|       } | ||||
|     }else{ | ||||
|       if (c == separator){ | ||||
|         return out; | ||||
|       }else{ | ||||
|         out += (char)c; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   return out; | ||||
| } | ||||
| 
 | ||||
| std::string JSON::Value::string_escape(std::string val){ | ||||
|   std::string out = "\""; | ||||
|   for (unsigned int i = 0; i < val.size(); ++i){ | ||||
|     switch (val[i]){ | ||||
|       case '"': out += "\\\""; break; | ||||
|       case '\\': out += "\\\\"; break; | ||||
|       case '\n': out += "\\n"; break; | ||||
|       case '\b': out += "\\b"; break; | ||||
|       case '\f': out += "\\f"; break; | ||||
|       case '\r': out += "\\r"; break; | ||||
|       case '\t': out += "\\t"; break; | ||||
|       default: out += val[i]; | ||||
|     } | ||||
|   } | ||||
|   out += "\""; | ||||
|   return out; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /// Sets this JSON::Value to null;
 | ||||
| JSON::Value::Value(){ | ||||
|   null(); | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to read from this position in the std::istream
 | ||||
| JSON::Value::Value(std::istream & fromstream){ | ||||
|   null(); | ||||
|   bool reading_object = false; | ||||
|   bool reading_obj_name = false; | ||||
|   bool reading_array = false; | ||||
|   while (fromstream.good()){ | ||||
|     int c = fromstream.peek(); | ||||
|     switch (c){ | ||||
|       case '{': | ||||
|         reading_object = true; | ||||
|         reading_obj_name = true; | ||||
|         c = fromstream.get(); | ||||
|         myType = OBJECT; | ||||
|         break; | ||||
|       case '[': | ||||
|         reading_array = true; | ||||
|         c = fromstream.get(); | ||||
|         myType = ARRAY; | ||||
|         append(JSON::Value(fromstream)); | ||||
|         break; | ||||
|       case '\'': | ||||
|       case '"': | ||||
|         c = fromstream.get(); | ||||
|         if (!reading_object || !reading_obj_name){ | ||||
|           myType = STRING; | ||||
|           strVal = read_string(c, fromstream); | ||||
|           return; | ||||
|         }else{ | ||||
|           std::string tmpstr = read_string(c, fromstream); | ||||
|           (*this)[tmpstr] = JSON::Value(fromstream); | ||||
|         } | ||||
|         break; | ||||
|       case '0': | ||||
|       case '1': | ||||
|       case '2': | ||||
|       case '3': | ||||
|       case '4': | ||||
|       case '5': | ||||
|       case '6': | ||||
|       case '7': | ||||
|       case '8': | ||||
|       case '9': | ||||
|         c = fromstream.get(); | ||||
|         myType = INTEGER; | ||||
|         intVal *= 10; | ||||
|         intVal += c - '0'; | ||||
|         break; | ||||
|       case ',': | ||||
|         if (!reading_object && !reading_array) return; | ||||
|         c = fromstream.get(); | ||||
|         if (reading_object){ | ||||
|           reading_obj_name = true; | ||||
|         }else{ | ||||
|           append(JSON::Value(fromstream)); | ||||
|         } | ||||
|         break; | ||||
|       case '}': | ||||
|         if (reading_object){c = fromstream.get();} | ||||
|         return; | ||||
|         break; | ||||
|       case ']': | ||||
|         if (reading_array){c = fromstream.get();} | ||||
|         return; | ||||
|         break; | ||||
|       case 't': | ||||
|       case 'T': | ||||
|         myType = BOOL; | ||||
|         intVal = 1; | ||||
|         return; | ||||
|         break; | ||||
|       case 'f': | ||||
|       case 'F': | ||||
|         myType = BOOL; | ||||
|         intVal = 0; | ||||
|         return; | ||||
|         break; | ||||
|       case 'n': | ||||
|       case 'N': | ||||
|         myType = EMPTY; | ||||
|         return; | ||||
|         break; | ||||
|       default: | ||||
|         c = fromstream.get();//ignore this character
 | ||||
|         continue; | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given string.
 | ||||
| JSON::Value::Value(const std::string & val){ | ||||
|   myType = STRING; | ||||
|   strVal = val; | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given string.
 | ||||
| JSON::Value::Value(const char * val){ | ||||
|   myType = STRING; | ||||
|   strVal = val; | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given integer.
 | ||||
| JSON::Value::Value(long long int val){ | ||||
|   myType = INTEGER; | ||||
|   intVal = val; | ||||
| } | ||||
| 
 | ||||
| /// Compares a JSON::Value to another for equality.
 | ||||
| bool JSON::Value::operator==(const JSON::Value & rhs) const{ | ||||
|   if (myType != rhs.myType) return false; | ||||
|   if (myType == INTEGER || myType == BOOL){return intVal == rhs.intVal;} | ||||
|   if (myType == STRING){return strVal == rhs.strVal;} | ||||
|   if (myType == EMPTY){return true;} | ||||
|   if (myType == OBJECT){ | ||||
|     if (objVal.size() != rhs.objVal.size()) return false; | ||||
|     for (std::map<std::string, Value>::const_iterator it = objVal.begin(); it != objVal.end(); ++it){ | ||||
|       if (!rhs.isMember(it->first)){return false;} | ||||
|       if (it->second != rhs.objVal.find(it->first)->second){return false;} | ||||
|     } | ||||
|     return true; | ||||
|   } | ||||
|   return true; | ||||
| } | ||||
| 
 | ||||
| /// Compares a JSON::Value to another for equality.
 | ||||
| bool JSON::Value::operator!=(const JSON::Value & rhs) const{ | ||||
|   return !((*this) == rhs); | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given boolean.
 | ||||
| JSON::Value & JSON::Value::operator=(const bool &rhs){ | ||||
|   null(); | ||||
|   myType = BOOL; | ||||
|   if (rhs) intVal = 1; | ||||
|   return *this; | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given string.
 | ||||
| JSON::Value & JSON::Value::operator=(const std::string &rhs){ | ||||
|   null(); | ||||
|   myType = STRING; | ||||
|   strVal = rhs; | ||||
|   return *this; | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given string.
 | ||||
| JSON::Value & JSON::Value::operator=(const char * rhs){ | ||||
|   return ((*this) = (std::string)rhs); | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given integer.
 | ||||
| JSON::Value & JSON::Value::operator=(const long long int &rhs){ | ||||
|   null(); | ||||
|   myType = INTEGER; | ||||
|   intVal = rhs; | ||||
|   return *this; | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given integer.
 | ||||
| JSON::Value & JSON::Value::operator=(const int &rhs){ | ||||
|   return ((*this) = (long long int)rhs); | ||||
| } | ||||
| 
 | ||||
| /// Sets this JSON::Value to the given integer.
 | ||||
| JSON::Value & JSON::Value::operator=(const unsigned int &rhs){ | ||||
|   return ((*this) = (long long int)rhs); | ||||
| } | ||||
| 
 | ||||
| /// Automatic conversion to long long int - returns 0 if not an integer type.
 | ||||
| JSON::Value::operator long long int(){ | ||||
|   return intVal; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /// Automatic conversion to std::string.
 | ||||
| /// Returns the raw string value if available, otherwise calls toString().
 | ||||
| JSON::Value::operator std::string(){ | ||||
|   if (myType == STRING){ | ||||
|     return strVal; | ||||
|   }else{ | ||||
|     return toString(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// Retrieves or sets the JSON::Value at this position in the object.
 | ||||
| /// Converts destructively to object if not already an object.
 | ||||
| JSON::Value & JSON::Value::operator[](const std::string i){ | ||||
|   if (myType != OBJECT){ | ||||
|     null(); | ||||
|     myType = OBJECT; | ||||
|   } | ||||
|   return objVal[i]; | ||||
| } | ||||
| 
 | ||||
| /// Retrieves or sets the JSON::Value at this position in the object.
 | ||||
| /// Converts destructively to object if not already an object.
 | ||||
| JSON::Value & JSON::Value::operator[](const char * i){ | ||||
|   if (myType != OBJECT){ | ||||
|     null(); | ||||
|     myType = OBJECT; | ||||
|   } | ||||
|   return objVal[i]; | ||||
| } | ||||
| 
 | ||||
| /// Retrieves or sets the JSON::Value at this position in the array.
 | ||||
| /// Converts destructively to array if not already an array.
 | ||||
| JSON::Value & JSON::Value::operator[](unsigned int i){ | ||||
|   if (myType != ARRAY){ | ||||
|     null(); | ||||
|     myType = ARRAY; | ||||
|   } | ||||
|   while (i >= arrVal.size()){ | ||||
|     append(JSON::Value()); | ||||
|   } | ||||
|   return arrVal[i]; | ||||
| } | ||||
| 
 | ||||
| /// Converts this JSON::Value to valid JSON notation and returns it.
 | ||||
| /// Makes absolutely no attempts to pretty-print anything. :-)
 | ||||
| std::string JSON::Value::toString(){ | ||||
|   switch (myType){ | ||||
|     case INTEGER: { | ||||
|       std::stringstream st; | ||||
|       st << intVal; | ||||
|       return st.str(); | ||||
|       break; | ||||
|     } | ||||
|     case STRING: { | ||||
|       return string_escape(strVal); | ||||
|       break; | ||||
|     } | ||||
|     case ARRAY: { | ||||
|       std::string tmp = "["; | ||||
|       for (ArrIter it = ArrBegin(); it != ArrEnd(); it++){ | ||||
|         tmp += it->toString(); | ||||
|         if (it + 1 != ArrEnd()){tmp += ",";} | ||||
|       } | ||||
|       tmp += "]"; | ||||
|       return tmp; | ||||
|       break; | ||||
|     } | ||||
|     case OBJECT: { | ||||
|       std::string tmp2 = "{"; | ||||
|       ObjIter it3 = ObjEnd(); | ||||
|       --it3; | ||||
|       for (ObjIter it2 = ObjBegin(); it2 != ObjEnd(); it2++){ | ||||
|         tmp2 += "\"" + it2->first + "\":"; | ||||
|         tmp2 += it2->second.toString(); | ||||
|         if (it2 != it3){tmp2 += ",";} | ||||
|       } | ||||
|       tmp2 += "}"; | ||||
|       return tmp2; | ||||
|       break; | ||||
|     } | ||||
|     case EMPTY: | ||||
|     default: | ||||
|       return "null"; | ||||
|   } | ||||
|   return "null";//should never get here...
 | ||||
| } | ||||
| 
 | ||||
| /// Appends the given value to the end of this JSON::Value array.
 | ||||
| /// Turns this value into an array if it is not already one.
 | ||||
| void JSON::Value::append(const JSON::Value & rhs){ | ||||
|   if (myType != ARRAY){ | ||||
|     null(); | ||||
|     myType = ARRAY; | ||||
|   } | ||||
|   arrVal.push_back(rhs); | ||||
| } | ||||
| 
 | ||||
| /// Prepends the given value to the beginning of this JSON::Value array.
 | ||||
| /// Turns this value into an array if it is not already one.
 | ||||
| void JSON::Value::prepend(const JSON::Value & rhs){ | ||||
|   if (myType != ARRAY){ | ||||
|     null(); | ||||
|     myType = ARRAY; | ||||
|   } | ||||
|   arrVal.push_front(rhs); | ||||
| } | ||||
| 
 | ||||
| /// For array and object JSON::Value objects, reduces them
 | ||||
| /// so they contain at most size elements, throwing away
 | ||||
| /// the first elements and keeping the last ones.
 | ||||
| /// Does nothing for other JSON::Value types, nor does it
 | ||||
| /// do anything if the size is already lower or equal to the
 | ||||
| /// given size.
 | ||||
| void JSON::Value::shrink(unsigned int size){ | ||||
|   if (myType == ARRAY){ | ||||
|     while (arrVal.size() > size){arrVal.pop_front();} | ||||
|     return; | ||||
|   } | ||||
|   if (myType == OBJECT){ | ||||
|     while (objVal.size() > size){objVal.erase(objVal.begin());} | ||||
|     return; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// For object JSON::Value objects, removes the member with
 | ||||
| /// the given name, if it exists. Has no effect otherwise.
 | ||||
| void JSON::Value::removeMember(const std::string & name){ | ||||
|   objVal.erase(name); | ||||
| } | ||||
| 
 | ||||
| /// For object JSON::Value objects, returns true if the
 | ||||
| /// given name is a member. Returns false otherwise.
 | ||||
| bool JSON::Value::isMember(const std::string & name) const{ | ||||
|   return objVal.count(name) > 0; | ||||
| } | ||||
| 
 | ||||
| /// Returns an iterator to the begin of the object map, if any.
 | ||||
| JSON::ObjIter JSON::Value::ObjBegin(){ | ||||
|   return objVal.begin(); | ||||
| } | ||||
| 
 | ||||
| /// Returns an iterator to the end of the object map, if any.
 | ||||
| JSON::ObjIter JSON::Value::ObjEnd(){ | ||||
|   return objVal.end(); | ||||
| } | ||||
| 
 | ||||
| /// Returns an iterator to the begin of the array, if any.
 | ||||
| JSON::ArrIter JSON::Value::ArrBegin(){ | ||||
|   return arrVal.begin(); | ||||
| } | ||||
| 
 | ||||
| /// Returns an iterator to the end of the array, if any.
 | ||||
| JSON::ArrIter JSON::Value::ArrEnd(){ | ||||
|   return arrVal.end(); | ||||
| } | ||||
| 
 | ||||
| /// Completely clears the contents of this value,
 | ||||
| /// changing its type to NULL in the process.
 | ||||
| void JSON::Value::null(){ | ||||
|   objVal.clear(); | ||||
|   arrVal.clear(); | ||||
|   strVal.clear(); | ||||
|   intVal = 0; | ||||
|   myType = EMPTY; | ||||
| } | ||||
| 
 | ||||
| /// Converts a std::string to a JSON::Value.
 | ||||
| JSON::Value JSON::fromString(std::string json){ | ||||
|   std::istringstream is(json); | ||||
|   return JSON::Value(is); | ||||
| } | ||||
							
								
								
									
										73
									
								
								util/json.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								util/json.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,73 @@ | |||
| /// \file json.h Holds all JSON-related headers.
 | ||||
| 
 | ||||
| #include <string> | ||||
| #include <deque> | ||||
| #include <map> | ||||
| #include <istream> | ||||
| 
 | ||||
| /// JSON-related classes and functions
 | ||||
| namespace JSON{ | ||||
| 
 | ||||
|   /// Lists all types of JSON::Value.
 | ||||
|   enum ValueType{ EMPTY, BOOL, INTEGER, STRING, ARRAY, OBJECT }; | ||||
| 
 | ||||
|   class Value;//forward declaration for below typedef
 | ||||
| 
 | ||||
|   typedef std::map<std::string, Value>::iterator ObjIter; | ||||
|   typedef std::deque<Value>::iterator ArrIter; | ||||
|    | ||||
|   /// A JSON::Value is either a string or an integer, but may also be an object, array or null.
 | ||||
|   class Value{ | ||||
|     private: | ||||
|       ValueType myType; | ||||
|       long long int intVal; | ||||
|       std::string strVal; | ||||
|       std::deque<Value> arrVal; | ||||
|       std::map<std::string, Value> objVal; | ||||
|       std::string read_string(int separator, std::istream & fromstream); | ||||
|       std::string string_escape(std::string val); | ||||
|       int c2hex(int c); | ||||
|     public: | ||||
|       //constructors
 | ||||
|       Value(); | ||||
|       Value(std::istream & fromstream); | ||||
|       Value(const std::string & val); | ||||
|       Value(const char * val); | ||||
|       Value(long long int val); | ||||
|       Value(bool val); | ||||
|       //comparison operators
 | ||||
|       bool operator==(const Value &rhs) const; | ||||
|       bool operator!=(const Value &rhs) const; | ||||
|       //assignment operators
 | ||||
|       Value & operator=(const std::string &rhs); | ||||
|       Value & operator=(const char * rhs); | ||||
|       Value & operator=(const long long int &rhs); | ||||
|       Value & operator=(const int &rhs); | ||||
|       Value & operator=(const unsigned int &rhs); | ||||
|       Value & operator=(const bool &rhs); | ||||
|       //converts to basic types
 | ||||
|       operator long long int(); | ||||
|       operator std::string(); | ||||
|       operator bool(); | ||||
|       //array operator for maps and arrays
 | ||||
|       Value & operator[](const std::string i); | ||||
|       Value & operator[](const char * i); | ||||
|       Value & operator[](unsigned int i); | ||||
|       //handy functions and others
 | ||||
|       std::string toString(); | ||||
|       void append(const Value & rhs); | ||||
|       void prepend(const Value & rhs); | ||||
|       void shrink(unsigned int size); | ||||
|       void removeMember(const std::string & name); | ||||
|       bool isMember(const std::string & name) const; | ||||
|       ObjIter ObjBegin(); | ||||
|       ObjIter ObjEnd(); | ||||
|       ArrIter ArrBegin(); | ||||
|       ArrIter ArrEnd(); | ||||
|       unsigned int size(); | ||||
|       void null(); | ||||
|   }; | ||||
| 
 | ||||
|   Value fromString(std::string json); | ||||
|    | ||||
| }; | ||||
|  | @ -1,19 +0,0 @@ | |||
| #ifndef JSON_AUTOLINK_H_INCLUDED | ||||
| # define JSON_AUTOLINK_H_INCLUDED | ||||
| 
 | ||||
| # include "config.h" | ||||
| 
 | ||||
| # ifdef JSON_IN_CPPTL | ||||
| #  include <cpptl/cpptl_autolink.h> | ||||
| # endif | ||||
| 
 | ||||
| # if !defined(JSON_NO_AUTOLINK)  &&  !defined(JSON_DLL_BUILD)  &&  !defined(JSON_IN_CPPTL) | ||||
| #  define CPPTL_AUTOLINK_NAME "json" | ||||
| #  undef CPPTL_AUTOLINK_DLL | ||||
| #  ifdef JSON_DLL | ||||
| #   define CPPTL_AUTOLINK_DLL | ||||
| #  endif | ||||
| #  include "autolink.h" | ||||
| # endif | ||||
| 
 | ||||
| #endif // JSON_AUTOLINK_H_INCLUDED
 | ||||
|  | @ -1,43 +0,0 @@ | |||
| #ifndef JSON_CONFIG_H_INCLUDED | ||||
| # define JSON_CONFIG_H_INCLUDED | ||||
| 
 | ||||
| /// If defined, indicates that json library is embedded in CppTL library.
 | ||||
| //# define JSON_IN_CPPTL 1
 | ||||
| 
 | ||||
| /// If defined, indicates that json may leverage CppTL library
 | ||||
| //#  define JSON_USE_CPPTL 1
 | ||||
| /// If defined, indicates that cpptl vector based map should be used instead of std::map
 | ||||
| /// as Value container.
 | ||||
| //#  define JSON_USE_CPPTL_SMALLMAP 1
 | ||||
| /// If defined, indicates that Json specific container should be used
 | ||||
| /// (hash table & simple deque container with customizable allocator).
 | ||||
| /// THIS FEATURE IS STILL EXPERIMENTAL!
 | ||||
| //#  define JSON_VALUE_USE_INTERNAL_MAP 1
 | ||||
| /// Force usage of standard new/malloc based allocator instead of memory pool based allocator.
 | ||||
| /// The memory pools allocator used optimization (initializing Value and ValueInternalLink
 | ||||
| /// as if it was a POD) that may cause some validation tool to report errors.
 | ||||
| /// Only has effects if JSON_VALUE_USE_INTERNAL_MAP is defined.
 | ||||
| //#  define JSON_USE_SIMPLE_INTERNAL_ALLOCATOR 1
 | ||||
| 
 | ||||
| /// If defined, indicates that Json use exception to report invalid type manipulation
 | ||||
| /// instead of C assert macro.
 | ||||
| # define JSON_USE_EXCEPTION 1 | ||||
| 
 | ||||
| # ifdef JSON_IN_CPPTL | ||||
| #  include <cpptl/config.h> | ||||
| #  ifndef JSON_USE_CPPTL | ||||
| #   define JSON_USE_CPPTL 1 | ||||
| #  endif | ||||
| # endif | ||||
| 
 | ||||
| # ifdef JSON_IN_CPPTL | ||||
| #  define JSON_API CPPTL_API | ||||
| # elif defined(JSON_DLL_BUILD) | ||||
| #  define JSON_API __declspec(dllexport) | ||||
| # elif defined(JSON_DLL) | ||||
| #  define JSON_API __declspec(dllimport) | ||||
| # else | ||||
| #  define JSON_API | ||||
| # endif | ||||
| 
 | ||||
| #endif // JSON_CONFIG_H_INCLUDED
 | ||||
|  | @ -1,42 +0,0 @@ | |||
| #ifndef CPPTL_JSON_FEATURES_H_INCLUDED | ||||
| # define CPPTL_JSON_FEATURES_H_INCLUDED | ||||
| 
 | ||||
| # include "forwards.h" | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
|    /** \brief Configuration passed to reader and writer.
 | ||||
|     * This configuration object can be used to force the Reader or Writer | ||||
|     * to behave in a standard conforming way. | ||||
|     */ | ||||
|    class JSON_API Features | ||||
|    { | ||||
|    public: | ||||
|       /** \brief A configuration that allows all features and assumes all strings are UTF-8.
 | ||||
|        * - C & C++ comments are allowed | ||||
|        * - Root object can be any JSON value | ||||
|        * - Assumes Value strings are encoded in UTF-8 | ||||
|        */ | ||||
|       static Features all(); | ||||
| 
 | ||||
|       /** \brief A configuration that is strictly compatible with the JSON specification.
 | ||||
|        * - Comments are forbidden. | ||||
|        * - Root object must be either an array or an object value. | ||||
|        * - Assumes Value strings are encoded in UTF-8 | ||||
|        */ | ||||
|       static Features strictMode(); | ||||
| 
 | ||||
|       /** \brief Initialize the configuration like JsonConfig::allFeatures;
 | ||||
|        */ | ||||
|       Features(); | ||||
| 
 | ||||
|       /// \c true if comments are allowed. Default: \c true.
 | ||||
|       bool allowComments_; | ||||
| 
 | ||||
|       /// \c true if root must be either an array or an object value. Default: \c false.
 | ||||
|       bool strictRoot_; | ||||
|    }; | ||||
| 
 | ||||
| } // namespace Json
 | ||||
| 
 | ||||
| #endif // CPPTL_JSON_FEATURES_H_INCLUDED
 | ||||
|  | @ -1,39 +0,0 @@ | |||
| #ifndef JSON_FORWARDS_H_INCLUDED | ||||
| # define JSON_FORWARDS_H_INCLUDED | ||||
| 
 | ||||
| # include "config.h" | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
|    // writer.h
 | ||||
|    class FastWriter; | ||||
|    class StyledWriter; | ||||
| 
 | ||||
|    // reader.h
 | ||||
|    class Reader; | ||||
| 
 | ||||
|    // features.h
 | ||||
|    class Features; | ||||
| 
 | ||||
|    // value.h
 | ||||
|    typedef int Int; | ||||
|    typedef unsigned int UInt; | ||||
|    class StaticString; | ||||
|    class Path; | ||||
|    class PathArgument; | ||||
|    class Value; | ||||
|    class ValueIteratorBase; | ||||
|    class ValueIterator; | ||||
|    class ValueConstIterator; | ||||
| #ifdef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    class ValueAllocator; | ||||
|    class ValueMapAllocator; | ||||
|    class ValueInternalLink; | ||||
|    class ValueInternalArray; | ||||
|    class ValueInternalMap; | ||||
| #endif // #ifdef JSON_VALUE_USE_INTERNAL_MAP
 | ||||
| 
 | ||||
| } // namespace Json
 | ||||
| 
 | ||||
| 
 | ||||
| #endif // JSON_FORWARDS_H_INCLUDED
 | ||||
|  | @ -1,10 +0,0 @@ | |||
| #ifndef JSON_JSON_H_INCLUDED | ||||
| # define JSON_JSON_H_INCLUDED | ||||
| 
 | ||||
| # include "autolink.h" | ||||
| # include "value.h" | ||||
| # include "reader.h" | ||||
| # include "writer.h" | ||||
| # include "features.h" | ||||
| 
 | ||||
| #endif // JSON_JSON_H_INCLUDED
 | ||||
|  | @ -1,125 +0,0 @@ | |||
| #ifndef JSONCPP_BATCHALLOCATOR_H_INCLUDED | ||||
| # define JSONCPP_BATCHALLOCATOR_H_INCLUDED | ||||
| 
 | ||||
| # include <stdlib.h> | ||||
| # include <assert.h> | ||||
| 
 | ||||
| # ifndef JSONCPP_DOC_EXCLUDE_IMPLEMENTATION | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
| /* Fast memory allocator.
 | ||||
|  * | ||||
|  * This memory allocator allocates memory for a batch of object (specified by | ||||
|  * the page size, the number of object in each page). | ||||
|  * | ||||
|  * It does not allow the destruction of a single object. All the allocated objects | ||||
|  * can be destroyed at once. The memory can be either released or reused for future | ||||
|  * allocation. | ||||
|  *  | ||||
|  * The in-place new operator must be used to construct the object using the pointer | ||||
|  * returned by allocate. | ||||
|  */ | ||||
| template<typename AllocatedType | ||||
|         ,const unsigned int objectPerAllocation> | ||||
| class BatchAllocator | ||||
| { | ||||
| public: | ||||
|    typedef AllocatedType Type; | ||||
| 
 | ||||
|    BatchAllocator( unsigned int objectsPerPage = 255 ) | ||||
|       : freeHead_( 0 ) | ||||
|       , objectsPerPage_( objectsPerPage ) | ||||
|    { | ||||
| //      printf( "Size: %d => %s\n", sizeof(AllocatedType), typeid(AllocatedType).name() );
 | ||||
|       assert( sizeof(AllocatedType) * objectPerAllocation >= sizeof(AllocatedType *) ); // We must be able to store a slist in the object free space.
 | ||||
|       assert( objectsPerPage >= 16 ); | ||||
|       batches_ = allocateBatch( 0 );   // allocated a dummy page
 | ||||
|       currentBatch_ = batches_; | ||||
|    } | ||||
| 
 | ||||
|    ~BatchAllocator() | ||||
|    { | ||||
|       for ( BatchInfo *batch = batches_; batch;  ) | ||||
|       { | ||||
|          BatchInfo *nextBatch = batch->next_; | ||||
|          free( batch ); | ||||
|          batch = nextBatch; | ||||
|       } | ||||
|    } | ||||
| 
 | ||||
|    /// allocate space for an array of objectPerAllocation object.
 | ||||
|    /// @warning it is the responsability of the caller to call objects constructors.
 | ||||
|    AllocatedType *allocate() | ||||
|    { | ||||
|       if ( freeHead_ ) // returns node from free list.
 | ||||
|       { | ||||
|          AllocatedType *object = freeHead_; | ||||
|          freeHead_ = *(AllocatedType **)object; | ||||
|          return object; | ||||
|       } | ||||
|       if ( currentBatch_->used_ == currentBatch_->end_ ) | ||||
|       { | ||||
|          currentBatch_ = currentBatch_->next_; | ||||
|          while ( currentBatch_  &&  currentBatch_->used_ == currentBatch_->end_ ) | ||||
|             currentBatch_ = currentBatch_->next_; | ||||
| 
 | ||||
|          if ( !currentBatch_  ) // no free batch found, allocate a new one
 | ||||
|          {  | ||||
|             currentBatch_ = allocateBatch( objectsPerPage_ ); | ||||
|             currentBatch_->next_ = batches_; // insert at the head of the list
 | ||||
|             batches_ = currentBatch_; | ||||
|          } | ||||
|       } | ||||
|       AllocatedType *allocated = currentBatch_->used_; | ||||
|       currentBatch_->used_ += objectPerAllocation; | ||||
|       return allocated; | ||||
|    } | ||||
| 
 | ||||
|    /// Release the object.
 | ||||
|    /// @warning it is the responsability of the caller to actually destruct the object.
 | ||||
|    void release( AllocatedType *object ) | ||||
|    { | ||||
|       assert( object != 0 ); | ||||
|       *(AllocatedType **)object = freeHead_; | ||||
|       freeHead_ = object; | ||||
|    } | ||||
| 
 | ||||
| private: | ||||
|    struct BatchInfo | ||||
|    { | ||||
|       BatchInfo *next_; | ||||
|       AllocatedType *used_; | ||||
|       AllocatedType *end_; | ||||
|       AllocatedType buffer_[objectPerAllocation]; | ||||
|    }; | ||||
| 
 | ||||
|    // disabled copy constructor and assignement operator.
 | ||||
|    BatchAllocator( const BatchAllocator & ); | ||||
|    void operator =( const BatchAllocator &); | ||||
| 
 | ||||
|    static BatchInfo *allocateBatch( unsigned int objectsPerPage ) | ||||
|    { | ||||
|       const unsigned int mallocSize = sizeof(BatchInfo) - sizeof(AllocatedType)* objectPerAllocation | ||||
|                                 + sizeof(AllocatedType) * objectPerAllocation * objectsPerPage; | ||||
|       BatchInfo *batch = static_cast<BatchInfo*>( malloc( mallocSize ) ); | ||||
|       batch->next_ = 0; | ||||
|       batch->used_ = batch->buffer_; | ||||
|       batch->end_ = batch->buffer_ + objectsPerPage; | ||||
|       return batch; | ||||
|    } | ||||
| 
 | ||||
|    BatchInfo *batches_; | ||||
|    BatchInfo *currentBatch_; | ||||
|    /// Head of a single linked list within the allocated space of freeed object
 | ||||
|    AllocatedType *freeHead_; | ||||
|    unsigned int objectsPerPage_; | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| } // namespace Json
 | ||||
| 
 | ||||
| # endif // ifndef JSONCPP_DOC_INCLUDE_IMPLEMENTATION
 | ||||
| 
 | ||||
| #endif // JSONCPP_BATCHALLOCATOR_H_INCLUDED
 | ||||
| 
 | ||||
|  | @ -1,448 +0,0 @@ | |||
| // included by json_value.cpp | ||||
| // everything is within Json namespace | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class ValueInternalArray | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| 
 | ||||
| ValueArrayAllocator::~ValueArrayAllocator() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class DefaultValueArrayAllocator | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR | ||||
| class DefaultValueArrayAllocator : public ValueArrayAllocator | ||||
| { | ||||
| public: // overridden from ValueArrayAllocator | ||||
|    virtual ~DefaultValueArrayAllocator() | ||||
|    { | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalArray *newArray() | ||||
|    { | ||||
|       return new ValueInternalArray(); | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalArray *newArrayCopy( const ValueInternalArray &other ) | ||||
|    { | ||||
|       return new ValueInternalArray( other ); | ||||
|    } | ||||
| 
 | ||||
|    virtual void destructArray( ValueInternalArray *array ) | ||||
|    { | ||||
|       delete array; | ||||
|    } | ||||
| 
 | ||||
|    virtual void reallocateArrayPageIndex( Value **&indexes,  | ||||
|                                           ValueInternalArray::PageIndex &indexCount, | ||||
|                                           ValueInternalArray::PageIndex minNewIndexCount ) | ||||
|    { | ||||
|       ValueInternalArray::PageIndex newIndexCount = (indexCount*3)/2 + 1; | ||||
|       if ( minNewIndexCount > newIndexCount ) | ||||
|          newIndexCount = minNewIndexCount; | ||||
|       void *newIndexes = realloc( indexes, sizeof(Value*) * newIndexCount ); | ||||
|       if ( !newIndexes ) | ||||
|          throw std::bad_alloc(); | ||||
|       indexCount = newIndexCount; | ||||
|       indexes = static_cast<Value **>( newIndexes ); | ||||
|    } | ||||
|    virtual void releaseArrayPageIndex( Value **indexes,  | ||||
|                                        ValueInternalArray::PageIndex indexCount ) | ||||
|    { | ||||
|       if ( indexes ) | ||||
|          free( indexes ); | ||||
|    } | ||||
| 
 | ||||
|    virtual Value *allocateArrayPage() | ||||
|    { | ||||
|       return static_cast<Value *>( malloc( sizeof(Value) * ValueInternalArray::itemsPerPage ) ); | ||||
|    } | ||||
| 
 | ||||
|    virtual void releaseArrayPage( Value *value ) | ||||
|    { | ||||
|       if ( value ) | ||||
|          free( value ); | ||||
|    } | ||||
| }; | ||||
| 
 | ||||
| #else // #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR | ||||
| /// @todo make this thread-safe (lock when accessign batch allocator) | ||||
| class DefaultValueArrayAllocator : public ValueArrayAllocator | ||||
| { | ||||
| public: // overridden from ValueArrayAllocator | ||||
|    virtual ~DefaultValueArrayAllocator() | ||||
|    { | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalArray *newArray() | ||||
|    { | ||||
|       ValueInternalArray *array = arraysAllocator_.allocate(); | ||||
|       new (array) ValueInternalArray(); // placement new | ||||
|       return array; | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalArray *newArrayCopy( const ValueInternalArray &other ) | ||||
|    { | ||||
|       ValueInternalArray *array = arraysAllocator_.allocate(); | ||||
|       new (array) ValueInternalArray( other ); // placement new | ||||
|       return array; | ||||
|    } | ||||
| 
 | ||||
|    virtual void destructArray( ValueInternalArray *array ) | ||||
|    { | ||||
|       if ( array ) | ||||
|       { | ||||
|          array->~ValueInternalArray(); | ||||
|          arraysAllocator_.release( array ); | ||||
|       } | ||||
|    } | ||||
| 
 | ||||
|    virtual void reallocateArrayPageIndex( Value **&indexes,  | ||||
|                                           ValueInternalArray::PageIndex &indexCount, | ||||
|                                           ValueInternalArray::PageIndex minNewIndexCount ) | ||||
|    { | ||||
|       ValueInternalArray::PageIndex newIndexCount = (indexCount*3)/2 + 1; | ||||
|       if ( minNewIndexCount > newIndexCount ) | ||||
|          newIndexCount = minNewIndexCount; | ||||
|       void *newIndexes = realloc( indexes, sizeof(Value*) * newIndexCount ); | ||||
|       if ( !newIndexes ) | ||||
|          throw std::bad_alloc(); | ||||
|       indexCount = newIndexCount; | ||||
|       indexes = static_cast<Value **>( newIndexes ); | ||||
|    } | ||||
|    virtual void releaseArrayPageIndex( Value **indexes,  | ||||
|                                        ValueInternalArray::PageIndex indexCount ) | ||||
|    { | ||||
|       if ( indexes ) | ||||
|          free( indexes ); | ||||
|    } | ||||
| 
 | ||||
|    virtual Value *allocateArrayPage() | ||||
|    { | ||||
|       return static_cast<Value *>( pagesAllocator_.allocate() ); | ||||
|    } | ||||
| 
 | ||||
|    virtual void releaseArrayPage( Value *value ) | ||||
|    { | ||||
|       if ( value ) | ||||
|          pagesAllocator_.release( value ); | ||||
|    } | ||||
| private: | ||||
|    BatchAllocator<ValueInternalArray,1> arraysAllocator_; | ||||
|    BatchAllocator<Value,ValueInternalArray::itemsPerPage> pagesAllocator_; | ||||
| }; | ||||
| #endif // #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR | ||||
| 
 | ||||
| static ValueArrayAllocator *&arrayAllocator() | ||||
| { | ||||
|    static DefaultValueArrayAllocator defaultAllocator; | ||||
|    static ValueArrayAllocator *arrayAllocator = &defaultAllocator; | ||||
|    return arrayAllocator; | ||||
| } | ||||
| 
 | ||||
| static struct DummyArrayAllocatorInitializer { | ||||
|    DummyArrayAllocatorInitializer()  | ||||
|    { | ||||
|       arrayAllocator();      // ensure arrayAllocator() statics are initialized before main(). | ||||
|    } | ||||
| } dummyArrayAllocatorInitializer; | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class ValueInternalArray | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| bool  | ||||
| ValueInternalArray::equals( const IteratorState &x,  | ||||
|                             const IteratorState &other ) | ||||
| { | ||||
|    return x.array_ == other.array_   | ||||
|           &&  x.currentItemIndex_ == other.currentItemIndex_   | ||||
|           &&  x.currentPageIndex_ == other.currentPageIndex_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::increment( IteratorState &it ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( it.array_  && | ||||
|       (it.currentPageIndex_ - it.array_->pages_)*itemsPerPage + it.currentItemIndex_ | ||||
|       != it.array_->size_, | ||||
|       "ValueInternalArray::increment(): moving iterator beyond end" ); | ||||
|    ++(it.currentItemIndex_); | ||||
|    if ( it.currentItemIndex_ == itemsPerPage ) | ||||
|    { | ||||
|       it.currentItemIndex_ = 0; | ||||
|       ++(it.currentPageIndex_); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::decrement( IteratorState &it ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( it.array_  &&  it.currentPageIndex_ == it.array_->pages_  | ||||
|                         &&  it.currentItemIndex_ == 0, | ||||
|       "ValueInternalArray::decrement(): moving iterator beyond end" ); | ||||
|    if ( it.currentItemIndex_ == 0 ) | ||||
|    { | ||||
|       it.currentItemIndex_ = itemsPerPage-1; | ||||
|       --(it.currentPageIndex_); | ||||
|    } | ||||
|    else | ||||
|    { | ||||
|       --(it.currentItemIndex_); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalArray::unsafeDereference( const IteratorState &it ) | ||||
| { | ||||
|    return (*(it.currentPageIndex_))[it.currentItemIndex_]; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalArray::dereference( const IteratorState &it ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( it.array_  && | ||||
|       (it.currentPageIndex_ - it.array_->pages_)*itemsPerPage + it.currentItemIndex_ | ||||
|       < it.array_->size_, | ||||
|       "ValueInternalArray::dereference(): dereferencing invalid iterator" ); | ||||
|    return unsafeDereference( it ); | ||||
| } | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::makeBeginIterator( IteratorState &it ) const | ||||
| { | ||||
|    it.array_ = const_cast<ValueInternalArray *>( this ); | ||||
|    it.currentItemIndex_ = 0; | ||||
|    it.currentPageIndex_ = pages_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::makeIterator( IteratorState &it, ArrayIndex index ) const | ||||
| { | ||||
|    it.array_ = const_cast<ValueInternalArray *>( this ); | ||||
|    it.currentItemIndex_ = index % itemsPerPage; | ||||
|    it.currentPageIndex_ = pages_ + index / itemsPerPage; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::makeEndIterator( IteratorState &it ) const | ||||
| { | ||||
|    makeIterator( it, size_ ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalArray::ValueInternalArray() | ||||
|    : pages_( 0 ) | ||||
|    , size_( 0 ) | ||||
|    , pageCount_( 0 ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalArray::ValueInternalArray( const ValueInternalArray &other ) | ||||
|    : pages_( 0 ) | ||||
|    , pageCount_( 0 ) | ||||
|    , size_( other.size_ ) | ||||
| { | ||||
|    PageIndex minNewPages = other.size_ / itemsPerPage; | ||||
|    arrayAllocator()->reallocateArrayPageIndex( pages_, pageCount_, minNewPages ); | ||||
|    JSON_ASSERT_MESSAGE( pageCount_ >= minNewPages,  | ||||
|                         "ValueInternalArray::reserve(): bad reallocation" ); | ||||
|    IteratorState itOther; | ||||
|    other.makeBeginIterator( itOther ); | ||||
|    Value *value; | ||||
|    for ( ArrayIndex index = 0; index < size_; ++index, increment(itOther) ) | ||||
|    { | ||||
|       if ( index % itemsPerPage == 0 ) | ||||
|       { | ||||
|          PageIndex pageIndex = index / itemsPerPage; | ||||
|          value = arrayAllocator()->allocateArrayPage(); | ||||
|          pages_[pageIndex] = value; | ||||
|       } | ||||
|       new (value) Value( dereference( itOther ) ); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalArray & | ||||
| ValueInternalArray::operator =( const ValueInternalArray &other ) | ||||
| { | ||||
|    ValueInternalArray temp( other ); | ||||
|    swap( temp ); | ||||
|    return *this; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalArray::~ValueInternalArray() | ||||
| { | ||||
|    // destroy all constructed items | ||||
|    IteratorState it; | ||||
|    IteratorState itEnd; | ||||
|    makeBeginIterator( it); | ||||
|    makeEndIterator( itEnd ); | ||||
|    for ( ; !equals(it,itEnd); increment(it) ) | ||||
|    { | ||||
|       Value *value = &dereference(it); | ||||
|       value->~Value(); | ||||
|    } | ||||
|    // release all pages | ||||
|    PageIndex lastPageIndex = size_ / itemsPerPage; | ||||
|    for ( PageIndex pageIndex = 0; pageIndex < lastPageIndex; ++pageIndex ) | ||||
|       arrayAllocator()->releaseArrayPage( pages_[pageIndex] ); | ||||
|    // release pages index | ||||
|    arrayAllocator()->releaseArrayPageIndex( pages_, pageCount_ ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::swap( ValueInternalArray &other ) | ||||
| { | ||||
|    Value **tempPages = pages_; | ||||
|    pages_ = other.pages_; | ||||
|    other.pages_ = tempPages; | ||||
|    ArrayIndex tempSize = size_; | ||||
|    size_ = other.size_; | ||||
|    other.size_ = tempSize; | ||||
|    PageIndex tempPageCount = pageCount_; | ||||
|    pageCount_ = other.pageCount_; | ||||
|    other.pageCount_ = tempPageCount; | ||||
| } | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::clear() | ||||
| { | ||||
|    ValueInternalArray dummy; | ||||
|    swap( dummy ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::resize( ArrayIndex newSize ) | ||||
| { | ||||
|    if ( newSize == 0 ) | ||||
|       clear(); | ||||
|    else if ( newSize < size_ ) | ||||
|    { | ||||
|       IteratorState it; | ||||
|       IteratorState itEnd; | ||||
|       makeIterator( it, newSize ); | ||||
|       makeIterator( itEnd, size_ ); | ||||
|       for ( ; !equals(it,itEnd); increment(it) ) | ||||
|       { | ||||
|          Value *value = &dereference(it); | ||||
|          value->~Value(); | ||||
|       } | ||||
|       PageIndex pageIndex = (newSize + itemsPerPage - 1) / itemsPerPage; | ||||
|       PageIndex lastPageIndex = size_ / itemsPerPage; | ||||
|       for ( ; pageIndex < lastPageIndex; ++pageIndex ) | ||||
|          arrayAllocator()->releaseArrayPage( pages_[pageIndex] ); | ||||
|       size_ = newSize; | ||||
|    } | ||||
|    else if ( newSize > size_ ) | ||||
|       resolveReference( newSize ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalArray::makeIndexValid( ArrayIndex index ) | ||||
| { | ||||
|    // Need to enlarge page index ? | ||||
|    if ( index >= pageCount_ * itemsPerPage ) | ||||
|    { | ||||
|       PageIndex minNewPages = (index + 1) / itemsPerPage; | ||||
|       arrayAllocator()->reallocateArrayPageIndex( pages_, pageCount_, minNewPages ); | ||||
|       JSON_ASSERT_MESSAGE( pageCount_ >= minNewPages, "ValueInternalArray::reserve(): bad reallocation" ); | ||||
|    } | ||||
| 
 | ||||
|    // Need to allocate new pages ? | ||||
|    ArrayIndex nextPageIndex =  | ||||
|       (size_ % itemsPerPage) != 0 ? size_ - (size_%itemsPerPage) + itemsPerPage | ||||
|                                   : size_; | ||||
|    if ( nextPageIndex <= index ) | ||||
|    { | ||||
|       PageIndex pageIndex = nextPageIndex / itemsPerPage; | ||||
|       PageIndex pageToAllocate = (index - nextPageIndex) / itemsPerPage + 1; | ||||
|       for ( ; pageToAllocate-- > 0; ++pageIndex ) | ||||
|          pages_[pageIndex] = arrayAllocator()->allocateArrayPage(); | ||||
|    } | ||||
| 
 | ||||
|    // Initialize all new entries | ||||
|    IteratorState it; | ||||
|    IteratorState itEnd; | ||||
|    makeIterator( it, size_ ); | ||||
|    size_ = index + 1; | ||||
|    makeIterator( itEnd, size_ ); | ||||
|    for ( ; !equals(it,itEnd); increment(it) ) | ||||
|    { | ||||
|       Value *value = &dereference(it); | ||||
|       new (value) Value(); // Construct a default value using placement new | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalArray::resolveReference( ArrayIndex index ) | ||||
| { | ||||
|    if ( index >= size_ ) | ||||
|       makeIndexValid( index ); | ||||
|    return pages_[index/itemsPerPage][index%itemsPerPage]; | ||||
| } | ||||
| 
 | ||||
| Value * | ||||
| ValueInternalArray::find( ArrayIndex index ) const | ||||
| { | ||||
|    if ( index >= size_ ) | ||||
|       return 0; | ||||
|    return &(pages_[index/itemsPerPage][index%itemsPerPage]); | ||||
| } | ||||
| 
 | ||||
| ValueInternalArray::ArrayIndex  | ||||
| ValueInternalArray::size() const | ||||
| { | ||||
|    return size_; | ||||
| } | ||||
| 
 | ||||
| int  | ||||
| ValueInternalArray::distance( const IteratorState &x, const IteratorState &y ) | ||||
| { | ||||
|    return indexOf(y) - indexOf(x); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalArray::ArrayIndex  | ||||
| ValueInternalArray::indexOf( const IteratorState &iterator ) | ||||
| { | ||||
|    if ( !iterator.array_ ) | ||||
|       return ArrayIndex(-1); | ||||
|    return ArrayIndex( | ||||
|       (iterator.currentPageIndex_ - iterator.array_->pages_) * itemsPerPage  | ||||
|       + iterator.currentItemIndex_ ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| int  | ||||
| ValueInternalArray::compare( const ValueInternalArray &other ) const | ||||
| { | ||||
|    int sizeDiff( size_ - other.size_ ); | ||||
|    if ( sizeDiff != 0 ) | ||||
|       return sizeDiff; | ||||
|     | ||||
|    for ( ArrayIndex index =0; index < size_; ++index ) | ||||
|    { | ||||
|       int diff = pages_[index/itemsPerPage][index%itemsPerPage].compare(  | ||||
|          other.pages_[index/itemsPerPage][index%itemsPerPage] ); | ||||
|       if ( diff != 0 ) | ||||
|          return diff; | ||||
|    } | ||||
|    return 0; | ||||
| } | ||||
|  | @ -1,607 +0,0 @@ | |||
| // included by json_value.cpp | ||||
| // everything is within Json namespace | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class ValueInternalMap | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| 
 | ||||
| /** \internal MUST be safely initialized using memset( this, 0, sizeof(ValueInternalLink) ); | ||||
|    * This optimization is used by the fast allocator. | ||||
|    */ | ||||
| ValueInternalLink::ValueInternalLink() | ||||
|    : previous_( 0 ) | ||||
|    , next_( 0 ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| ValueInternalLink::~ValueInternalLink() | ||||
| {  | ||||
|    for ( int index =0; index < itemPerLink; ++index ) | ||||
|    { | ||||
|       if ( !items_[index].isItemAvailable() ) | ||||
|       { | ||||
|          if ( !items_[index].isMemberNameStatic() ) | ||||
|             free( keys_[index] ); | ||||
|       } | ||||
|       else | ||||
|          break; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| ValueMapAllocator::~ValueMapAllocator() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR | ||||
| class DefaultValueMapAllocator : public ValueMapAllocator | ||||
| { | ||||
| public: // overridden from ValueMapAllocator | ||||
|    virtual ValueInternalMap *newMap() | ||||
|    { | ||||
|       return new ValueInternalMap(); | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalMap *newMapCopy( const ValueInternalMap &other ) | ||||
|    { | ||||
|       return new ValueInternalMap( other ); | ||||
|    } | ||||
| 
 | ||||
|    virtual void destructMap( ValueInternalMap *map ) | ||||
|    { | ||||
|       delete map; | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalLink *allocateMapBuckets( unsigned int size ) | ||||
|    { | ||||
|       return new ValueInternalLink[size]; | ||||
|    } | ||||
| 
 | ||||
|    virtual void releaseMapBuckets( ValueInternalLink *links ) | ||||
|    { | ||||
|       delete [] links; | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalLink *allocateMapLink() | ||||
|    { | ||||
|       return new ValueInternalLink(); | ||||
|    } | ||||
| 
 | ||||
|    virtual void releaseMapLink( ValueInternalLink *link ) | ||||
|    { | ||||
|       delete link; | ||||
|    } | ||||
| }; | ||||
| #else | ||||
| /// @todo make this thread-safe (lock when accessign batch allocator) | ||||
| class DefaultValueMapAllocator : public ValueMapAllocator | ||||
| { | ||||
| public: // overridden from ValueMapAllocator | ||||
|    virtual ValueInternalMap *newMap() | ||||
|    { | ||||
|       ValueInternalMap *map = mapsAllocator_.allocate(); | ||||
|       new (map) ValueInternalMap(); // placement new | ||||
|       return map; | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalMap *newMapCopy( const ValueInternalMap &other ) | ||||
|    { | ||||
|       ValueInternalMap *map = mapsAllocator_.allocate(); | ||||
|       new (map) ValueInternalMap( other ); // placement new | ||||
|       return map; | ||||
|    } | ||||
| 
 | ||||
|    virtual void destructMap( ValueInternalMap *map ) | ||||
|    { | ||||
|       if ( map ) | ||||
|       { | ||||
|          map->~ValueInternalMap(); | ||||
|          mapsAllocator_.release( map ); | ||||
|       } | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalLink *allocateMapBuckets( unsigned int size ) | ||||
|    { | ||||
|       return new ValueInternalLink[size]; | ||||
|    } | ||||
| 
 | ||||
|    virtual void releaseMapBuckets( ValueInternalLink *links ) | ||||
|    { | ||||
|       delete [] links; | ||||
|    } | ||||
| 
 | ||||
|    virtual ValueInternalLink *allocateMapLink() | ||||
|    { | ||||
|       ValueInternalLink *link = linksAllocator_.allocate(); | ||||
|       memset( link, 0, sizeof(ValueInternalLink) ); | ||||
|       return link; | ||||
|    } | ||||
| 
 | ||||
|    virtual void releaseMapLink( ValueInternalLink *link ) | ||||
|    { | ||||
|       link->~ValueInternalLink(); | ||||
|       linksAllocator_.release( link ); | ||||
|    } | ||||
| private: | ||||
|    BatchAllocator<ValueInternalMap,1> mapsAllocator_; | ||||
|    BatchAllocator<ValueInternalLink,1> linksAllocator_; | ||||
| }; | ||||
| #endif | ||||
| 
 | ||||
| static ValueMapAllocator *&mapAllocator() | ||||
| { | ||||
|    static DefaultValueMapAllocator defaultAllocator; | ||||
|    static ValueMapAllocator *mapAllocator = &defaultAllocator; | ||||
|    return mapAllocator; | ||||
| } | ||||
| 
 | ||||
| static struct DummyMapAllocatorInitializer { | ||||
|    DummyMapAllocatorInitializer()  | ||||
|    { | ||||
|       mapAllocator();      // ensure mapAllocator() statics are initialized before main(). | ||||
|    } | ||||
| } dummyMapAllocatorInitializer; | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| // h(K) = value * K >> w ; with w = 32 & K prime w.r.t. 2^32. | ||||
| 
 | ||||
| /* | ||||
| use linked list hash map.  | ||||
| buckets array is a container. | ||||
| linked list element contains 6 key/values. (memory = (16+4) * 6 + 4 = 124) | ||||
| value have extra state: valid, available, deleted | ||||
| */ | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalMap::ValueInternalMap() | ||||
|    : buckets_( 0 ) | ||||
|    , tailLink_( 0 ) | ||||
|    , bucketsSize_( 0 ) | ||||
|    , itemCount_( 0 ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalMap::ValueInternalMap( const ValueInternalMap &other ) | ||||
|    : buckets_( 0 ) | ||||
|    , tailLink_( 0 ) | ||||
|    , bucketsSize_( 0 ) | ||||
|    , itemCount_( 0 ) | ||||
| { | ||||
|    reserve( other.itemCount_ ); | ||||
|    IteratorState it; | ||||
|    IteratorState itEnd; | ||||
|    other.makeBeginIterator( it ); | ||||
|    other.makeEndIterator( itEnd ); | ||||
|    for ( ; !equals(it,itEnd); increment(it) ) | ||||
|    { | ||||
|       bool isStatic; | ||||
|       const char *memberName = key( it, isStatic ); | ||||
|       const Value &aValue = value( it ); | ||||
|       resolveReference(memberName, isStatic) = aValue; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalMap & | ||||
| ValueInternalMap::operator =( const ValueInternalMap &other ) | ||||
| { | ||||
|    ValueInternalMap dummy( other ); | ||||
|    swap( dummy ); | ||||
|    return *this; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalMap::~ValueInternalMap() | ||||
| { | ||||
|    if ( buckets_ ) | ||||
|    { | ||||
|       for ( BucketIndex bucketIndex =0; bucketIndex < bucketsSize_; ++bucketIndex ) | ||||
|       { | ||||
|          ValueInternalLink *link = buckets_[bucketIndex].next_; | ||||
|          while ( link ) | ||||
|          { | ||||
|             ValueInternalLink *linkToRelease = link; | ||||
|             link = link->next_; | ||||
|             mapAllocator()->releaseMapLink( linkToRelease ); | ||||
|          } | ||||
|       } | ||||
|       mapAllocator()->releaseMapBuckets( buckets_ ); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::swap( ValueInternalMap &other ) | ||||
| { | ||||
|    ValueInternalLink *tempBuckets = buckets_; | ||||
|    buckets_ = other.buckets_; | ||||
|    other.buckets_ = tempBuckets; | ||||
|    ValueInternalLink *tempTailLink = tailLink_; | ||||
|    tailLink_ = other.tailLink_; | ||||
|    other.tailLink_ = tempTailLink; | ||||
|    BucketIndex tempBucketsSize = bucketsSize_; | ||||
|    bucketsSize_ = other.bucketsSize_; | ||||
|    other.bucketsSize_ = tempBucketsSize; | ||||
|    BucketIndex tempItemCount = itemCount_; | ||||
|    itemCount_ = other.itemCount_; | ||||
|    other.itemCount_ = tempItemCount; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::clear() | ||||
| { | ||||
|    ValueInternalMap dummy; | ||||
|    swap( dummy ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalMap::BucketIndex  | ||||
| ValueInternalMap::size() const | ||||
| { | ||||
|    return itemCount_; | ||||
| } | ||||
| 
 | ||||
| bool  | ||||
| ValueInternalMap::reserveDelta( BucketIndex growth ) | ||||
| { | ||||
|    return reserve( itemCount_ + growth ); | ||||
| } | ||||
| 
 | ||||
| bool  | ||||
| ValueInternalMap::reserve( BucketIndex newItemCount ) | ||||
| { | ||||
|    if ( !buckets_  &&  newItemCount > 0 ) | ||||
|    { | ||||
|       buckets_ = mapAllocator()->allocateMapBuckets( 1 ); | ||||
|       bucketsSize_ = 1; | ||||
|       tailLink_ = &buckets_[0]; | ||||
|    } | ||||
| //   BucketIndex idealBucketCount = (newItemCount + ValueInternalLink::itemPerLink) / ValueInternalLink::itemPerLink; | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| const Value * | ||||
| ValueInternalMap::find( const char *key ) const | ||||
| { | ||||
|    if ( !bucketsSize_ ) | ||||
|       return 0; | ||||
|    HashKey hashedKey = hash( key ); | ||||
|    BucketIndex bucketIndex = hashedKey % bucketsSize_; | ||||
|    for ( const ValueInternalLink *current = &buckets_[bucketIndex];  | ||||
|          current != 0;  | ||||
|          current = current->next_ ) | ||||
|    { | ||||
|       for ( BucketIndex index=0; index < ValueInternalLink::itemPerLink; ++index ) | ||||
|       { | ||||
|          if ( current->items_[index].isItemAvailable() ) | ||||
|             return 0; | ||||
|          if ( strcmp( key, current->keys_[index] ) == 0 ) | ||||
|             return ¤t->items_[index]; | ||||
|       } | ||||
|    } | ||||
|    return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value * | ||||
| ValueInternalMap::find( const char *key ) | ||||
| { | ||||
|    const ValueInternalMap *constThis = this; | ||||
|    return const_cast<Value *>( constThis->find( key ) ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalMap::resolveReference( const char *key, | ||||
|                                     bool isStatic ) | ||||
| { | ||||
|    HashKey hashedKey = hash( key ); | ||||
|    if ( bucketsSize_ ) | ||||
|    { | ||||
|       BucketIndex bucketIndex = hashedKey % bucketsSize_; | ||||
|       ValueInternalLink **previous = 0; | ||||
|       BucketIndex index; | ||||
|       for ( ValueInternalLink *current = &buckets_[bucketIndex];  | ||||
|             current != 0;  | ||||
|             previous = ¤t->next_, current = current->next_ ) | ||||
|       { | ||||
|          for ( index=0; index < ValueInternalLink::itemPerLink; ++index ) | ||||
|          { | ||||
|             if ( current->items_[index].isItemAvailable() ) | ||||
|                return setNewItem( key, isStatic, current, index ); | ||||
|             if ( strcmp( key, current->keys_[index] ) == 0 ) | ||||
|                return current->items_[index]; | ||||
|          } | ||||
|       } | ||||
|    } | ||||
| 
 | ||||
|    reserveDelta( 1 ); | ||||
|    return unsafeAdd( key, isStatic, hashedKey ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::remove( const char *key ) | ||||
| { | ||||
|    HashKey hashedKey = hash( key ); | ||||
|    if ( !bucketsSize_ ) | ||||
|       return; | ||||
|    BucketIndex bucketIndex = hashedKey % bucketsSize_; | ||||
|    for ( ValueInternalLink *link = &buckets_[bucketIndex];  | ||||
|          link != 0;  | ||||
|          link = link->next_ ) | ||||
|    { | ||||
|       BucketIndex index; | ||||
|       for ( index =0; index < ValueInternalLink::itemPerLink; ++index ) | ||||
|       { | ||||
|          if ( link->items_[index].isItemAvailable() ) | ||||
|             return; | ||||
|          if ( strcmp( key, link->keys_[index] ) == 0 ) | ||||
|          { | ||||
|             doActualRemove( link, index, bucketIndex ); | ||||
|             return; | ||||
|          } | ||||
|       } | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::doActualRemove( ValueInternalLink *link,  | ||||
|                                   BucketIndex index, | ||||
|                                   BucketIndex bucketIndex ) | ||||
| { | ||||
|    // find last item of the bucket and swap it with the 'removed' one. | ||||
|    // set removed items flags to 'available'. | ||||
|    // if last page only contains 'available' items, then desallocate it (it's empty) | ||||
|    ValueInternalLink *&lastLink = getLastLinkInBucket( index ); | ||||
|    BucketIndex lastItemIndex = 1; // a link can never be empty, so start at 1 | ||||
|    for ( ;    | ||||
|          lastItemIndex < ValueInternalLink::itemPerLink;  | ||||
|          ++lastItemIndex ) // may be optimized with dicotomic search | ||||
|    { | ||||
|       if ( lastLink->items_[lastItemIndex].isItemAvailable() ) | ||||
|          break; | ||||
|    } | ||||
|     | ||||
|    BucketIndex lastUsedIndex = lastItemIndex - 1; | ||||
|    Value *valueToDelete = &link->items_[index]; | ||||
|    Value *valueToPreserve = &lastLink->items_[lastUsedIndex]; | ||||
|    if ( valueToDelete != valueToPreserve ) | ||||
|       valueToDelete->swap( *valueToPreserve ); | ||||
|    if ( lastUsedIndex == 0 )  // page is now empty | ||||
|    {  // remove it from bucket linked list and delete it. | ||||
|       ValueInternalLink *linkPreviousToLast = lastLink->previous_; | ||||
|       if ( linkPreviousToLast != 0 )   // can not deleted bucket link. | ||||
|       { | ||||
|          mapAllocator()->releaseMapLink( lastLink ); | ||||
|          linkPreviousToLast->next_ = 0; | ||||
|          lastLink = linkPreviousToLast; | ||||
|       } | ||||
|    } | ||||
|    else | ||||
|    { | ||||
|       Value dummy; | ||||
|       valueToPreserve->swap( dummy ); // restore deleted to default Value. | ||||
|       valueToPreserve->setItemUsed( false ); | ||||
|    } | ||||
|    --itemCount_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalLink *& | ||||
| ValueInternalMap::getLastLinkInBucket( BucketIndex bucketIndex ) | ||||
| { | ||||
|    if ( bucketIndex == bucketsSize_ - 1 ) | ||||
|       return tailLink_; | ||||
|    ValueInternalLink *&previous = buckets_[bucketIndex+1].previous_; | ||||
|    if ( !previous ) | ||||
|       previous = &buckets_[bucketIndex]; | ||||
|    return previous; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalMap::setNewItem( const char *key,  | ||||
|                               bool isStatic, | ||||
|                               ValueInternalLink *link,  | ||||
|                               BucketIndex index ) | ||||
| { | ||||
|    char *duplicatedKey = valueAllocator()->makeMemberName( key ); | ||||
|    ++itemCount_; | ||||
|    link->keys_[index] = duplicatedKey; | ||||
|    link->items_[index].setItemUsed(); | ||||
|    link->items_[index].setMemberNameIsStatic( isStatic ); | ||||
|    return link->items_[index]; // items already default constructed. | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalMap::unsafeAdd( const char *key,  | ||||
|                              bool isStatic,  | ||||
|                              HashKey hashedKey ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( bucketsSize_ > 0, "ValueInternalMap::unsafeAdd(): internal logic error." ); | ||||
|    BucketIndex bucketIndex = hashedKey % bucketsSize_; | ||||
|    ValueInternalLink *&previousLink = getLastLinkInBucket( bucketIndex ); | ||||
|    ValueInternalLink *link = previousLink; | ||||
|    BucketIndex index; | ||||
|    for ( index =0; index < ValueInternalLink::itemPerLink; ++index ) | ||||
|    { | ||||
|       if ( link->items_[index].isItemAvailable() ) | ||||
|          break; | ||||
|    } | ||||
|    if ( index == ValueInternalLink::itemPerLink ) // need to add a new page | ||||
|    { | ||||
|       ValueInternalLink *newLink = mapAllocator()->allocateMapLink(); | ||||
|       index = 0; | ||||
|       link->next_ = newLink; | ||||
|       previousLink = newLink; | ||||
|       link = newLink; | ||||
|    } | ||||
|    return setNewItem( key, isStatic, link, index ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueInternalMap::HashKey  | ||||
| ValueInternalMap::hash( const char *key ) const | ||||
| { | ||||
|    HashKey hash = 0; | ||||
|    while ( *key ) | ||||
|       hash += *key++ * 37; | ||||
|    return hash; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| int  | ||||
| ValueInternalMap::compare( const ValueInternalMap &other ) const | ||||
| { | ||||
|    int sizeDiff( itemCount_ - other.itemCount_ ); | ||||
|    if ( sizeDiff != 0 ) | ||||
|       return sizeDiff; | ||||
|    // Strict order guaranty is required. Compare all keys FIRST, then compare values. | ||||
|    IteratorState it; | ||||
|    IteratorState itEnd; | ||||
|    makeBeginIterator( it ); | ||||
|    makeEndIterator( itEnd ); | ||||
|    for ( ; !equals(it,itEnd); increment(it) ) | ||||
|    { | ||||
|       if ( !other.find( key( it ) ) ) | ||||
|          return 1; | ||||
|    } | ||||
| 
 | ||||
|    // All keys are equals, let's compare values | ||||
|    makeBeginIterator( it ); | ||||
|    for ( ; !equals(it,itEnd); increment(it) ) | ||||
|    { | ||||
|       const Value *otherValue = other.find( key( it ) ); | ||||
|       int valueDiff = value(it).compare( *otherValue ); | ||||
|       if ( valueDiff != 0 ) | ||||
|          return valueDiff; | ||||
|    } | ||||
|    return 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::makeBeginIterator( IteratorState &it ) const | ||||
| { | ||||
|    it.map_ = const_cast<ValueInternalMap *>( this ); | ||||
|    it.bucketIndex_ = 0; | ||||
|    it.itemIndex_ = 0; | ||||
|    it.link_ = buckets_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::makeEndIterator( IteratorState &it ) const | ||||
| { | ||||
|    it.map_ = const_cast<ValueInternalMap *>( this ); | ||||
|    it.bucketIndex_ = bucketsSize_; | ||||
|    it.itemIndex_ = 0; | ||||
|    it.link_ = 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| ValueInternalMap::equals( const IteratorState &x, const IteratorState &other ) | ||||
| { | ||||
|    return x.map_ == other.map_   | ||||
|           &&  x.bucketIndex_ == other.bucketIndex_   | ||||
|           &&  x.link_ == other.link_ | ||||
|           &&  x.itemIndex_ == other.itemIndex_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::incrementBucket( IteratorState &iterator ) | ||||
| { | ||||
|    ++iterator.bucketIndex_; | ||||
|    JSON_ASSERT_MESSAGE( iterator.bucketIndex_ <= iterator.map_->bucketsSize_, | ||||
|       "ValueInternalMap::increment(): attempting to iterate beyond end." ); | ||||
|    if ( iterator.bucketIndex_ == iterator.map_->bucketsSize_ ) | ||||
|       iterator.link_ = 0; | ||||
|    else | ||||
|       iterator.link_ = &(iterator.map_->buckets_[iterator.bucketIndex_]); | ||||
|    iterator.itemIndex_ = 0; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::increment( IteratorState &iterator ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( iterator.map_, "Attempting to iterator using invalid iterator." ); | ||||
|    ++iterator.itemIndex_; | ||||
|    if ( iterator.itemIndex_ == ValueInternalLink::itemPerLink ) | ||||
|    { | ||||
|       JSON_ASSERT_MESSAGE( iterator.link_ != 0, | ||||
|          "ValueInternalMap::increment(): attempting to iterate beyond end." ); | ||||
|       iterator.link_ = iterator.link_->next_; | ||||
|       if ( iterator.link_ == 0 ) | ||||
|          incrementBucket( iterator ); | ||||
|    } | ||||
|    else if ( iterator.link_->items_[iterator.itemIndex_].isItemAvailable() ) | ||||
|    { | ||||
|       incrementBucket( iterator ); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueInternalMap::decrement( IteratorState &iterator ) | ||||
| { | ||||
|    if ( iterator.itemIndex_ == 0 ) | ||||
|    { | ||||
|       JSON_ASSERT_MESSAGE( iterator.map_, "Attempting to iterate using invalid iterator." ); | ||||
|       if ( iterator.link_ == &iterator.map_->buckets_[iterator.bucketIndex_] ) | ||||
|       { | ||||
|          JSON_ASSERT_MESSAGE( iterator.bucketIndex_ > 0, "Attempting to iterate beyond beginning." ); | ||||
|          --(iterator.bucketIndex_); | ||||
|       } | ||||
|       iterator.link_ = iterator.link_->previous_; | ||||
|       iterator.itemIndex_ = ValueInternalLink::itemPerLink - 1; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| const char * | ||||
| ValueInternalMap::key( const IteratorState &iterator ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( iterator.link_, "Attempting to iterate using invalid iterator." ); | ||||
|    return iterator.link_->keys_[iterator.itemIndex_]; | ||||
| } | ||||
| 
 | ||||
| const char * | ||||
| ValueInternalMap::key( const IteratorState &iterator, bool &isStatic ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( iterator.link_, "Attempting to iterate using invalid iterator." ); | ||||
|    isStatic = iterator.link_->items_[iterator.itemIndex_].isMemberNameStatic(); | ||||
|    return iterator.link_->keys_[iterator.itemIndex_]; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| ValueInternalMap::value( const IteratorState &iterator ) | ||||
| { | ||||
|    JSON_ASSERT_MESSAGE( iterator.link_, "Attempting to iterate using invalid iterator." ); | ||||
|    return iterator.link_->items_[iterator.itemIndex_]; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| int  | ||||
| ValueInternalMap::distance( const IteratorState &x, const IteratorState &y ) | ||||
| { | ||||
|    int offset = 0; | ||||
|    IteratorState it = x; | ||||
|    while ( !equals( it, y ) ) | ||||
|       increment( it ); | ||||
|    return offset; | ||||
| } | ||||
|  | @ -1,885 +0,0 @@ | |||
| #include "reader.h" | ||||
| #include "value.h" | ||||
| #include <utility> | ||||
| #include <cstdio> | ||||
| #include <cassert> | ||||
| #include <cstring> | ||||
| #include <iostream> | ||||
| #include <stdexcept> | ||||
| 
 | ||||
| #if _MSC_VER >= 1400 // VC++ 8.0
 | ||||
| #pragma warning( disable : 4996 )   // disable warning about strdup being deprecated.
 | ||||
| #endif | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
| // Implementation of class Features
 | ||||
| // ////////////////////////////////
 | ||||
| 
 | ||||
| Features::Features() | ||||
|    : allowComments_( true ) | ||||
|    , strictRoot_( false ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Features  | ||||
| Features::all() | ||||
| { | ||||
|    return Features(); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Features  | ||||
| Features::strictMode() | ||||
| { | ||||
|    Features features; | ||||
|    features.allowComments_ = false; | ||||
|    features.strictRoot_ = true; | ||||
|    return features; | ||||
| } | ||||
| 
 | ||||
| // Implementation of class Reader
 | ||||
| // ////////////////////////////////
 | ||||
| 
 | ||||
| 
 | ||||
| static inline bool  | ||||
| in( Reader::Char c, Reader::Char c1, Reader::Char c2, Reader::Char c3, Reader::Char c4 ) | ||||
| { | ||||
|    return c == c1  ||  c == c2  ||  c == c3  ||  c == c4; | ||||
| } | ||||
| 
 | ||||
| static inline bool  | ||||
| in( Reader::Char c, Reader::Char c1, Reader::Char c2, Reader::Char c3, Reader::Char c4, Reader::Char c5 ) | ||||
| { | ||||
|    return c == c1  ||  c == c2  ||  c == c3  ||  c == c4  ||  c == c5; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| static bool  | ||||
| containsNewLine( Reader::Location begin,  | ||||
|                  Reader::Location end ) | ||||
| { | ||||
|    for ( ;begin < end; ++begin ) | ||||
|       if ( *begin == '\n'  ||  *begin == '\r' ) | ||||
|          return true; | ||||
|    return false; | ||||
| } | ||||
| 
 | ||||
| static std::string codePointToUTF8(unsigned int cp) | ||||
| { | ||||
|    std::string result; | ||||
|     | ||||
|    // based on description from http://en.wikipedia.org/wiki/UTF-8
 | ||||
| 
 | ||||
|    if (cp <= 0x7f)  | ||||
|    { | ||||
|       result.resize(1); | ||||
|       result[0] = static_cast<char>(cp); | ||||
|    }  | ||||
|    else if (cp <= 0x7FF)  | ||||
|    { | ||||
|       result.resize(2); | ||||
|       result[1] = static_cast<char>(0x80 | (0x3f & cp)); | ||||
|       result[0] = static_cast<char>(0xC0 | (0x1f & (cp >> 6))); | ||||
|    }  | ||||
|    else if (cp <= 0xFFFF)  | ||||
|    { | ||||
|       result.resize(3); | ||||
|       result[2] = static_cast<char>(0x80 | (0x3f & cp)); | ||||
|       result[1] = 0x80 | static_cast<char>((0x3f & (cp >> 6))); | ||||
|       result[0] = 0xE0 | static_cast<char>((0xf & (cp >> 12))); | ||||
|    } | ||||
|    else if (cp <= 0x10FFFF)  | ||||
|    { | ||||
|       result.resize(4); | ||||
|       result[3] = static_cast<char>(0x80 | (0x3f & cp)); | ||||
|       result[2] = static_cast<char>(0x80 | (0x3f & (cp >> 6))); | ||||
|       result[1] = static_cast<char>(0x80 | (0x3f & (cp >> 12))); | ||||
|       result[0] = static_cast<char>(0xF0 | (0x7 & (cp >> 18))); | ||||
|    } | ||||
| 
 | ||||
|    return result; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Class Reader
 | ||||
| // //////////////////////////////////////////////////////////////////
 | ||||
| 
 | ||||
| Reader::Reader() | ||||
|    : features_( Features::all() ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Reader::Reader( const Features &features ) | ||||
|    : features_( features ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool | ||||
| Reader::parse( const std::string &document,  | ||||
|                Value &root, | ||||
|                bool collectComments ) | ||||
| { | ||||
|    document_ = document; | ||||
|    const char *begin = document_.c_str(); | ||||
|    const char *end = begin + document_.length(); | ||||
|    return parse( begin, end, root, collectComments ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool | ||||
| Reader::parse( std::istream& sin, | ||||
|                Value &root, | ||||
|                bool collectComments ) | ||||
| { | ||||
|    //std::istream_iterator<char> begin(sin);
 | ||||
|    //std::istream_iterator<char> end;
 | ||||
|    // Those would allow streamed input from a file, if parse() were a
 | ||||
|    // template function.
 | ||||
| 
 | ||||
|    // Since std::string is reference-counted, this at least does not
 | ||||
|    // create an extra copy.
 | ||||
|    std::string doc; | ||||
|    std::getline(sin, doc, (char)EOF); | ||||
|    return parse( doc, root, collectComments ); | ||||
| } | ||||
| 
 | ||||
| bool  | ||||
| Reader::parse( const char *beginDoc, const char *endDoc,  | ||||
|                Value &root, | ||||
|                bool collectComments ) | ||||
| { | ||||
|    if ( !features_.allowComments_ ) | ||||
|    { | ||||
|       collectComments = false; | ||||
|    } | ||||
| 
 | ||||
|    begin_ = beginDoc; | ||||
|    end_ = endDoc; | ||||
|    collectComments_ = collectComments; | ||||
|    current_ = begin_; | ||||
|    lastValueEnd_ = 0; | ||||
|    lastValue_ = 0; | ||||
|    commentsBefore_ = ""; | ||||
|    errors_.clear(); | ||||
|    while ( !nodes_.empty() ) | ||||
|       nodes_.pop(); | ||||
|    nodes_.push( &root ); | ||||
|     | ||||
|    bool successful = readValue(); | ||||
|    Token token; | ||||
|    skipCommentTokens( token ); | ||||
|    if ( collectComments_  &&  !commentsBefore_.empty() ) | ||||
|       root.setComment( commentsBefore_, commentAfter ); | ||||
|    if ( features_.strictRoot_ ) | ||||
|    { | ||||
|       if ( !root.isArray()  &&  !root.isObject() ) | ||||
|       { | ||||
|          // Set error location to start of doc, ideally should be first token found in doc
 | ||||
|          token.type_ = tokenError; | ||||
|          token.start_ = beginDoc; | ||||
|          token.end_ = endDoc; | ||||
|          addError( "A valid JSON document must be either an array or an object value.", | ||||
|                    token ); | ||||
|          return false; | ||||
|       } | ||||
|    } | ||||
|    return successful; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool | ||||
| Reader::readValue() | ||||
| { | ||||
|    Token token; | ||||
|    skipCommentTokens( token ); | ||||
|    bool successful = true; | ||||
| 
 | ||||
|    if ( collectComments_  &&  !commentsBefore_.empty() ) | ||||
|    { | ||||
|       currentValue().setComment( commentsBefore_, commentBefore ); | ||||
|       commentsBefore_ = ""; | ||||
|    } | ||||
| 
 | ||||
| 
 | ||||
|    switch ( token.type_ ) | ||||
|    { | ||||
|    case tokenObjectBegin: | ||||
|       successful = readObject( token ); | ||||
|       break; | ||||
|    case tokenArrayBegin: | ||||
|       successful = readArray( token ); | ||||
|       break; | ||||
|    case tokenNumber: | ||||
|       successful = decodeNumber( token ); | ||||
|       break; | ||||
|    case tokenString: | ||||
|       successful = decodeString( token ); | ||||
|       break; | ||||
|    case tokenTrue: | ||||
|       currentValue() = true; | ||||
|       break; | ||||
|    case tokenFalse: | ||||
|       currentValue() = false; | ||||
|       break; | ||||
|    case tokenNull: | ||||
|       currentValue() = Value(); | ||||
|       break; | ||||
|    default: | ||||
|       return addError( "Syntax error: value, object or array expected.", token ); | ||||
|    } | ||||
| 
 | ||||
|    if ( collectComments_ ) | ||||
|    { | ||||
|       lastValueEnd_ = current_; | ||||
|       lastValue_ = ¤tValue(); | ||||
|    } | ||||
| 
 | ||||
|    return successful; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| Reader::skipCommentTokens( Token &token ) | ||||
| { | ||||
|    if ( features_.allowComments_ ) | ||||
|    { | ||||
|       do | ||||
|       { | ||||
|          readToken( token ); | ||||
|       } | ||||
|       while ( token.type_ == tokenComment ); | ||||
|    } | ||||
|    else | ||||
|    { | ||||
|       readToken( token ); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::expectToken( TokenType type, Token &token, const char *message ) | ||||
| { | ||||
|    readToken( token ); | ||||
|    if ( token.type_ != type ) | ||||
|       return addError( message, token ); | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::readToken( Token &token ) | ||||
| { | ||||
|    skipSpaces(); | ||||
|    token.start_ = current_; | ||||
|    Char c = getNextChar(); | ||||
|    bool ok = true; | ||||
|    switch ( c ) | ||||
|    { | ||||
|    case '{': | ||||
|       token.type_ = tokenObjectBegin; | ||||
|       break; | ||||
|    case '}': | ||||
|       token.type_ = tokenObjectEnd; | ||||
|       break; | ||||
|    case '[': | ||||
|       token.type_ = tokenArrayBegin; | ||||
|       break; | ||||
|    case ']': | ||||
|       token.type_ = tokenArrayEnd; | ||||
|       break; | ||||
|    case '"': | ||||
|       token.type_ = tokenString; | ||||
|       ok = readString(); | ||||
|       break; | ||||
|    case '/': | ||||
|       token.type_ = tokenComment; | ||||
|       ok = readComment(); | ||||
|       break; | ||||
|    case '0': | ||||
|    case '1': | ||||
|    case '2': | ||||
|    case '3': | ||||
|    case '4': | ||||
|    case '5': | ||||
|    case '6': | ||||
|    case '7': | ||||
|    case '8': | ||||
|    case '9': | ||||
|    case '-': | ||||
|       token.type_ = tokenNumber; | ||||
|       readNumber(); | ||||
|       break; | ||||
|    case 't': | ||||
|       token.type_ = tokenTrue; | ||||
|       ok = match( "rue", 3 ); | ||||
|       break; | ||||
|    case 'f': | ||||
|       token.type_ = tokenFalse; | ||||
|       ok = match( "alse", 4 ); | ||||
|       break; | ||||
|    case 'n': | ||||
|       token.type_ = tokenNull; | ||||
|       ok = match( "ull", 3 ); | ||||
|       break; | ||||
|    case ',': | ||||
|       token.type_ = tokenArraySeparator; | ||||
|       break; | ||||
|    case ':': | ||||
|       token.type_ = tokenMemberSeparator; | ||||
|       break; | ||||
|    case 0: | ||||
|       token.type_ = tokenEndOfStream; | ||||
|       break; | ||||
|    default: | ||||
|       ok = false; | ||||
|       break; | ||||
|    } | ||||
|    if ( !ok ) | ||||
|       token.type_ = tokenError; | ||||
|    token.end_ = current_; | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| Reader::skipSpaces() | ||||
| { | ||||
|    while ( current_ != end_ ) | ||||
|    { | ||||
|       Char c = *current_; | ||||
|       if ( c == ' '  ||  c == '\t'  ||  c == '\r'  ||  c == '\n' ) | ||||
|          ++current_; | ||||
|       else | ||||
|          break; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::match( Location pattern,  | ||||
|                int patternLength ) | ||||
| { | ||||
|    if ( end_ - current_ < patternLength ) | ||||
|       return false; | ||||
|    int index = patternLength; | ||||
|    while ( index-- ) | ||||
|       if ( current_[index] != pattern[index] ) | ||||
|          return false; | ||||
|    current_ += patternLength; | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool | ||||
| Reader::readComment() | ||||
| { | ||||
|    Location commentBegin = current_ - 1; | ||||
|    Char c = getNextChar(); | ||||
|    bool successful = false; | ||||
|    if ( c == '*' ) | ||||
|       successful = readCStyleComment(); | ||||
|    else if ( c == '/' ) | ||||
|       successful = readCppStyleComment(); | ||||
|    if ( !successful ) | ||||
|       return false; | ||||
| 
 | ||||
|    if ( collectComments_ ) | ||||
|    { | ||||
|       CommentPlacement placement = commentBefore; | ||||
|       if ( lastValueEnd_  &&  !containsNewLine( lastValueEnd_, commentBegin ) ) | ||||
|       { | ||||
|          if ( c != '*'  ||  !containsNewLine( commentBegin, current_ ) ) | ||||
|             placement = commentAfterOnSameLine; | ||||
|       } | ||||
| 
 | ||||
|       addComment( commentBegin, current_, placement ); | ||||
|    } | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| Reader::addComment( Location begin,  | ||||
|                     Location end,  | ||||
|                     CommentPlacement placement ) | ||||
| { | ||||
|    assert( collectComments_ ); | ||||
|    if ( placement == commentAfterOnSameLine ) | ||||
|    { | ||||
|       assert( lastValue_ != 0 ); | ||||
|       lastValue_->setComment( std::string( begin, end ), placement ); | ||||
|    } | ||||
|    else | ||||
|    { | ||||
|       if ( !commentsBefore_.empty() ) | ||||
|          commentsBefore_ += "\n"; | ||||
|       commentsBefore_ += std::string( begin, end ); | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::readCStyleComment() | ||||
| { | ||||
|    while ( current_ != end_ ) | ||||
|    { | ||||
|       Char c = getNextChar(); | ||||
|       if ( c == '*'  &&  *current_ == '/' ) | ||||
|          break; | ||||
|    } | ||||
|    return getNextChar() == '/'; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::readCppStyleComment() | ||||
| { | ||||
|    while ( current_ != end_ ) | ||||
|    { | ||||
|       Char c = getNextChar(); | ||||
|       if (  c == '\r'  ||  c == '\n' ) | ||||
|          break; | ||||
|    } | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| Reader::readNumber() | ||||
| { | ||||
|    while ( current_ != end_ ) | ||||
|    { | ||||
|       if ( !(*current_ >= '0'  &&  *current_ <= '9')  && | ||||
|            !in( *current_, '.', 'e', 'E', '+', '-' ) ) | ||||
|          break; | ||||
|       ++current_; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| bool | ||||
| Reader::readString() | ||||
| { | ||||
|    Char c = 0; | ||||
|    while ( current_ != end_ ) | ||||
|    { | ||||
|       c = getNextChar(); | ||||
|       if ( c == '\\' ) | ||||
|          getNextChar(); | ||||
|       else if ( c == '"' ) | ||||
|          break; | ||||
|    } | ||||
|    return c == '"'; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::readObject( Token &tokenStart ) | ||||
| { | ||||
|    Token tokenName; | ||||
|    std::string name; | ||||
|    currentValue() = Value( objectValue ); | ||||
|    while ( readToken( tokenName ) ) | ||||
|    { | ||||
|       bool initialTokenOk = true; | ||||
|       while ( tokenName.type_ == tokenComment  &&  initialTokenOk ) | ||||
|          initialTokenOk = readToken( tokenName ); | ||||
|       if  ( !initialTokenOk ) | ||||
|          break; | ||||
|       if ( tokenName.type_ == tokenObjectEnd  &&  name.empty() )  // empty object
 | ||||
|          return true; | ||||
|       if ( tokenName.type_ != tokenString ) | ||||
|          break; | ||||
|        | ||||
|       name = ""; | ||||
|       if ( !decodeString( tokenName, name ) ) | ||||
|          return recoverFromError( tokenObjectEnd ); | ||||
| 
 | ||||
|       Token colon; | ||||
|       if ( !readToken( colon ) ||  colon.type_ != tokenMemberSeparator ) | ||||
|       { | ||||
|          return addErrorAndRecover( "Missing ':' after object member name",  | ||||
|                                     colon,  | ||||
|                                     tokenObjectEnd ); | ||||
|       } | ||||
|       Value &value = currentValue()[ name ]; | ||||
|       nodes_.push( &value ); | ||||
|       bool ok = readValue(); | ||||
|       nodes_.pop(); | ||||
|       if ( !ok ) // error already set
 | ||||
|          return recoverFromError( tokenObjectEnd ); | ||||
| 
 | ||||
|       Token comma; | ||||
|       if ( !readToken( comma ) | ||||
|             ||  ( comma.type_ != tokenObjectEnd  &&   | ||||
|                   comma.type_ != tokenArraySeparator && | ||||
| 		  comma.type_ != tokenComment ) ) | ||||
|       { | ||||
|          return addErrorAndRecover( "Missing ',' or '}' in object declaration",  | ||||
|                                     comma,  | ||||
|                                     tokenObjectEnd ); | ||||
|       } | ||||
|       bool finalizeTokenOk = true; | ||||
|       while ( comma.type_ == tokenComment && | ||||
|               finalizeTokenOk ) | ||||
|          finalizeTokenOk = readToken( comma ); | ||||
|       if ( comma.type_ == tokenObjectEnd ) | ||||
|          return true; | ||||
|    } | ||||
|    return addErrorAndRecover( "Missing '}' or object member name",  | ||||
|                               tokenName,  | ||||
|                               tokenObjectEnd ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::readArray( Token &tokenStart ) | ||||
| { | ||||
|    currentValue() = Value( arrayValue ); | ||||
|    skipSpaces(); | ||||
|    if ( *current_ == ']' ) // empty array
 | ||||
|    { | ||||
|       Token endArray; | ||||
|       readToken( endArray ); | ||||
|       return true; | ||||
|    } | ||||
|    int index = 0; | ||||
|    while ( true ) | ||||
|    { | ||||
|       Value &value = currentValue()[ index++ ]; | ||||
|       nodes_.push( &value ); | ||||
|       bool ok = readValue(); | ||||
|       nodes_.pop(); | ||||
|       if ( !ok ) // error already set
 | ||||
|          return recoverFromError( tokenArrayEnd ); | ||||
| 
 | ||||
|       Token token; | ||||
|       // Accept Comment after last item in the array.
 | ||||
|       ok = readToken( token ); | ||||
|       while ( token.type_ == tokenComment  &&  ok ) | ||||
|       { | ||||
|          ok = readToken( token ); | ||||
|       } | ||||
|       bool badTokenType = ( token.type_ == tokenArraySeparator  &&   | ||||
|                             token.type_ == tokenArrayEnd ); | ||||
|       if ( !ok  ||  badTokenType ) | ||||
|       { | ||||
|          return addErrorAndRecover( "Missing ',' or ']' in array declaration",  | ||||
|                                     token,  | ||||
|                                     tokenArrayEnd ); | ||||
|       } | ||||
|       if ( token.type_ == tokenArrayEnd ) | ||||
|          break; | ||||
|    } | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::decodeNumber( Token &token ) | ||||
| { | ||||
|    bool isDouble = false; | ||||
|    for ( Location inspect = token.start_; inspect != token.end_; ++inspect ) | ||||
|    { | ||||
|       isDouble = isDouble   | ||||
|                  ||  in( *inspect, '.', 'e', 'E', '+' )   | ||||
|                  ||  ( *inspect == '-'  &&  inspect != token.start_ ); | ||||
|    } | ||||
|    if ( isDouble ) | ||||
|       return decodeDouble( token ); | ||||
|    Location current = token.start_; | ||||
|    bool isNegative = *current == '-'; | ||||
|    if ( isNegative ) | ||||
|       ++current; | ||||
|    Value::UInt threshold = (isNegative ? Value::UInt(-Value::minInt)  | ||||
|                                        : Value::maxUInt) / 10; | ||||
|    Value::UInt value = 0; | ||||
|    while ( current < token.end_ ) | ||||
|    { | ||||
|       Char c = *current++; | ||||
|       if ( c < '0'  ||  c > '9' ) | ||||
|          return addError( "'" + std::string( token.start_, token.end_ ) + "' is not a number.", token ); | ||||
|       if ( value >= threshold ) | ||||
|          return decodeDouble( token ); | ||||
|       value = value * 10 + Value::UInt(c - '0'); | ||||
|    } | ||||
|    if ( isNegative ) | ||||
|       currentValue() = -Value::Int( value ); | ||||
|    else if ( value <= Value::UInt(Value::maxInt) ) | ||||
|       currentValue() = Value::Int( value ); | ||||
|    else | ||||
|       currentValue() = value; | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::decodeDouble( Token &token ) | ||||
| { | ||||
|    double value = 0; | ||||
|    const int bufferSize = 32; | ||||
|    int count; | ||||
|    int length = int(token.end_ - token.start_); | ||||
|    if ( length <= bufferSize ) | ||||
|    { | ||||
|       Char buffer[bufferSize]; | ||||
|       memcpy( buffer, token.start_, length ); | ||||
|       buffer[length] = 0; | ||||
|       count = sscanf( buffer, "%lf", &value ); | ||||
|    } | ||||
|    else | ||||
|    { | ||||
|       std::string buffer( token.start_, token.end_ ); | ||||
|       count = sscanf( buffer.c_str(), "%lf", &value ); | ||||
|    } | ||||
| 
 | ||||
|    if ( count != 1 ) | ||||
|       return addError( "'" + std::string( token.start_, token.end_ ) + "' is not a number.", token ); | ||||
|    currentValue() = value; | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::decodeString( Token &token ) | ||||
| { | ||||
|    std::string decoded; | ||||
|    if ( !decodeString( token, decoded ) ) | ||||
|       return false; | ||||
|    currentValue() = decoded; | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::decodeString( Token &token, std::string &decoded ) | ||||
| { | ||||
|    decoded.reserve( token.end_ - token.start_ - 2 ); | ||||
|    Location current = token.start_ + 1; // skip '"'
 | ||||
|    Location end = token.end_ - 1;      // do not include '"'
 | ||||
|    while ( current != end ) | ||||
|    { | ||||
|       Char c = *current++; | ||||
|       if ( c == '"' ) | ||||
|          break; | ||||
|       else if ( c == '\\' ) | ||||
|       { | ||||
|          if ( current == end ) | ||||
|             return addError( "Empty escape sequence in string", token, current ); | ||||
|          Char escape = *current++; | ||||
|          switch ( escape ) | ||||
|          { | ||||
|          case '"': decoded += '"'; break; | ||||
|          case '/': decoded += '/'; break; | ||||
|          case '\\': decoded += '\\'; break; | ||||
|          case 'b': decoded += '\b'; break; | ||||
|          case 'f': decoded += '\f'; break; | ||||
|          case 'n': decoded += '\n'; break; | ||||
|          case 'r': decoded += '\r'; break; | ||||
|          case 't': decoded += '\t'; break; | ||||
|          case 'u': | ||||
|             { | ||||
|                unsigned int unicode; | ||||
|                if ( !decodeUnicodeCodePoint( token, current, end, unicode ) ) | ||||
|                   return false; | ||||
|                decoded += codePointToUTF8(unicode); | ||||
|             } | ||||
|             break; | ||||
|          default: | ||||
|             return addError( "Bad escape sequence in string", token, current ); | ||||
|          } | ||||
|       } | ||||
|       else | ||||
|       { | ||||
|          decoded += c; | ||||
|       } | ||||
|    } | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| bool | ||||
| Reader::decodeUnicodeCodePoint( Token &token,  | ||||
|                                      Location ¤t,  | ||||
|                                      Location end,  | ||||
|                                      unsigned int &unicode ) | ||||
| { | ||||
| 
 | ||||
|    if ( !decodeUnicodeEscapeSequence( token, current, end, unicode ) ) | ||||
|       return false; | ||||
|    if (unicode >= 0xD800 && unicode <= 0xDBFF) | ||||
|    { | ||||
|       // surrogate pairs
 | ||||
|       if (end - current < 6) | ||||
|          return addError( "additional six characters expected to parse unicode surrogate pair.", token, current ); | ||||
|       unsigned int surrogatePair; | ||||
|       if (*(current++) == '\\' && *(current++)== 'u') | ||||
|       { | ||||
|          if (decodeUnicodeEscapeSequence( token, current, end, surrogatePair )) | ||||
|          { | ||||
|             unicode = 0x10000 + ((unicode & 0x3FF) << 10) + (surrogatePair & 0x3FF); | ||||
|          }  | ||||
|          else | ||||
|             return false; | ||||
|       }  | ||||
|       else | ||||
|          return addError( "expecting another \\u token to begin the second half of a unicode surrogate pair", token, current ); | ||||
|    } | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| bool  | ||||
| Reader::decodeUnicodeEscapeSequence( Token &token,  | ||||
|                                      Location ¤t,  | ||||
|                                      Location end,  | ||||
|                                      unsigned int &unicode ) | ||||
| { | ||||
|    if ( end - current < 4 ) | ||||
|       return addError( "Bad unicode escape sequence in string: four digits expected.", token, current ); | ||||
|    unicode = 0; | ||||
|    for ( int index =0; index < 4; ++index ) | ||||
|    { | ||||
|       Char c = *current++; | ||||
|       unicode *= 16; | ||||
|       if ( c >= '0'  &&  c <= '9' ) | ||||
|          unicode += c - '0'; | ||||
|       else if ( c >= 'a'  &&  c <= 'f' ) | ||||
|          unicode += c - 'a' + 10; | ||||
|       else if ( c >= 'A'  &&  c <= 'F' ) | ||||
|          unicode += c - 'A' + 10; | ||||
|       else | ||||
|          return addError( "Bad unicode escape sequence in string: hexadecimal digit expected.", token, current ); | ||||
|    } | ||||
|    return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::addError( const std::string &message,  | ||||
|                   Token &token, | ||||
|                   Location extra ) | ||||
| { | ||||
|    ErrorInfo info; | ||||
|    info.token_ = token; | ||||
|    info.message_ = message; | ||||
|    info.extra_ = extra; | ||||
|    errors_.push_back( info ); | ||||
|    return false; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::recoverFromError( TokenType skipUntilToken ) | ||||
| { | ||||
|    int errorCount = int(errors_.size()); | ||||
|    Token skip; | ||||
|    while ( true ) | ||||
|    { | ||||
|       if ( !readToken(skip) ) | ||||
|          errors_.resize( errorCount ); // discard errors caused by recovery
 | ||||
|       if ( skip.type_ == skipUntilToken  ||  skip.type_ == tokenEndOfStream ) | ||||
|          break; | ||||
|    } | ||||
|    errors_.resize( errorCount ); | ||||
|    return false; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| Reader::addErrorAndRecover( const std::string &message,  | ||||
|                             Token &token, | ||||
|                             TokenType skipUntilToken ) | ||||
| { | ||||
|    addError( message, token ); | ||||
|    return recoverFromError( skipUntilToken ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value & | ||||
| Reader::currentValue() | ||||
| { | ||||
|    return *(nodes_.top()); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Reader::Char  | ||||
| Reader::getNextChar() | ||||
| { | ||||
|    if ( current_ == end_ ) | ||||
|       return 0; | ||||
|    return *current_++; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| Reader::getLocationLineAndColumn( Location location, | ||||
|                                   int &line, | ||||
|                                   int &column ) const | ||||
| { | ||||
|    Location current = begin_; | ||||
|    Location lastLineStart = current; | ||||
|    line = 0; | ||||
|    while ( current < location  &&  current != end_ ) | ||||
|    { | ||||
|       Char c = *current++; | ||||
|       if ( c == '\r' ) | ||||
|       { | ||||
|          if ( *current == '\n' ) | ||||
|             ++current; | ||||
|          lastLineStart = current; | ||||
|          ++line; | ||||
|       } | ||||
|       else if ( c == '\n' ) | ||||
|       { | ||||
|          lastLineStart = current; | ||||
|          ++line; | ||||
|       } | ||||
|    } | ||||
|    // column & line start at 1
 | ||||
|    column = int(location - lastLineStart) + 1; | ||||
|    ++line; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string | ||||
| Reader::getLocationLineAndColumn( Location location ) const | ||||
| { | ||||
|    int line, column; | ||||
|    getLocationLineAndColumn( location, line, column ); | ||||
|    char buffer[18+16+16+1]; | ||||
|    sprintf( buffer, "Line %d, Column %d", line, column ); | ||||
|    return buffer; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string  | ||||
| Reader::getFormatedErrorMessages() const | ||||
| { | ||||
|    std::string formattedMessage; | ||||
|    for ( Errors::const_iterator itError = errors_.begin(); | ||||
|          itError != errors_.end(); | ||||
|          ++itError ) | ||||
|    { | ||||
|       const ErrorInfo &error = *itError; | ||||
|       formattedMessage += "* " + getLocationLineAndColumn( error.token_.start_ ) + "\n"; | ||||
|       formattedMessage += "  " + error.message_ + "\n"; | ||||
|       if ( error.extra_ ) | ||||
|          formattedMessage += "See " + getLocationLineAndColumn( error.extra_ ) + " for detail.\n"; | ||||
|    } | ||||
|    return formattedMessage; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::istream& operator>>( std::istream &sin, Value &root ) | ||||
| { | ||||
|     Json::Reader reader; | ||||
|     bool ok = reader.parse(sin, root, true); | ||||
|     //JSON_ASSERT( ok );
 | ||||
|     if (!ok) throw std::runtime_error(reader.getFormatedErrorMessages()); | ||||
|     return sin; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| } // namespace Json
 | ||||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -1,292 +0,0 @@ | |||
| // included by json_value.cpp | ||||
| // everything is within Json namespace | ||||
| 
 | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class ValueIteratorBase | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| 
 | ||||
| ValueIteratorBase::ValueIteratorBase() | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    : current_() | ||||
|    , isNull_( true ) | ||||
| { | ||||
| } | ||||
| #else | ||||
|    : isArray_( true ) | ||||
|    , isNull_( true ) | ||||
| { | ||||
|    iterator_.array_ = ValueInternalArray::IteratorState(); | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| 
 | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
| ValueIteratorBase::ValueIteratorBase( const Value::ObjectValues::iterator ¤t ) | ||||
|    : current_( current ) | ||||
|    , isNull_( false ) | ||||
| { | ||||
| } | ||||
| #else | ||||
| ValueIteratorBase::ValueIteratorBase( const ValueInternalArray::IteratorState &state ) | ||||
|    : isArray_( true ) | ||||
| { | ||||
|    iterator_.array_ = state; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueIteratorBase::ValueIteratorBase( const ValueInternalMap::IteratorState &state ) | ||||
|    : isArray_( false ) | ||||
| { | ||||
|    iterator_.map_ = state; | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| Value & | ||||
| ValueIteratorBase::deref() const | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    return current_->second; | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       return ValueInternalArray::dereference( iterator_.array_ ); | ||||
|    return ValueInternalMap::value( iterator_.map_ ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueIteratorBase::increment() | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    ++current_; | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       ValueInternalArray::increment( iterator_.array_ ); | ||||
|    ValueInternalMap::increment( iterator_.map_ ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueIteratorBase::decrement() | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    --current_; | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       ValueInternalArray::decrement( iterator_.array_ ); | ||||
|    ValueInternalMap::decrement( iterator_.map_ ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| ValueIteratorBase::difference_type  | ||||
| ValueIteratorBase::computeDistance( const SelfType &other ) const | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
| # ifdef JSON_USE_CPPTL_SMALLMAP | ||||
|    return current_ - other.current_; | ||||
| # else | ||||
|    // Iterator for null value are initialized using the default | ||||
|    // constructor, which initialize current_ to the default | ||||
|    // std::map::iterator. As begin() and end() are two instance  | ||||
|    // of the default std::map::iterator, they can not be compared. | ||||
|    // To allow this, we handle this comparison specifically. | ||||
|    if ( isNull_  &&  other.isNull_ ) | ||||
|    { | ||||
|       return 0; | ||||
|    } | ||||
| 
 | ||||
| 
 | ||||
|    // Usage of std::distance is not portable (does not compile with Sun Studio 12 RogueWave STL, | ||||
|    // which is the one used by default). | ||||
|    // Using a portable hand-made version for non random iterator instead: | ||||
|    //   return difference_type( std::distance( current_, other.current_ ) ); | ||||
|    difference_type myDistance = 0; | ||||
|    for ( Value::ObjectValues::iterator it = current_; it != other.current_; ++it ) | ||||
|    { | ||||
|       ++myDistance; | ||||
|    } | ||||
|    return myDistance; | ||||
| # endif | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       return ValueInternalArray::distance( iterator_.array_, other.iterator_.array_ ); | ||||
|    return ValueInternalMap::distance( iterator_.map_, other.iterator_.map_ ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| ValueIteratorBase::isEqual( const SelfType &other ) const | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    if ( isNull_ ) | ||||
|    { | ||||
|       return other.isNull_; | ||||
|    } | ||||
|    return current_ == other.current_; | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       return ValueInternalArray::equals( iterator_.array_, other.iterator_.array_ ); | ||||
|    return ValueInternalMap::equals( iterator_.map_, other.iterator_.map_ ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| ValueIteratorBase::copy( const SelfType &other ) | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    current_ = other.current_; | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       iterator_.array_ = other.iterator_.array_; | ||||
|    iterator_.map_ = other.iterator_.map_; | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Value  | ||||
| ValueIteratorBase::key() const | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    const Value::CZString czstring = (*current_).first; | ||||
|    if ( czstring.c_str() ) | ||||
|    { | ||||
|       if ( czstring.isStaticString() ) | ||||
|          return Value( StaticString( czstring.c_str() ) ); | ||||
|       return Value( czstring.c_str() ); | ||||
|    } | ||||
|    return Value( czstring.index() ); | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       return Value( ValueInternalArray::indexOf( iterator_.array_ ) ); | ||||
|    bool isStatic; | ||||
|    const char *memberName = ValueInternalMap::key( iterator_.map_, isStatic ); | ||||
|    if ( isStatic ) | ||||
|       return Value( StaticString( memberName ) ); | ||||
|    return Value( memberName ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| UInt  | ||||
| ValueIteratorBase::index() const | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    const Value::CZString czstring = (*current_).first; | ||||
|    if ( !czstring.c_str() ) | ||||
|       return czstring.index(); | ||||
|    return Value::UInt( -1 ); | ||||
| #else | ||||
|    if ( isArray_ ) | ||||
|       return Value::UInt( ValueInternalArray::indexOf( iterator_.array_ ) ); | ||||
|    return Value::UInt( -1 ); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| const char * | ||||
| ValueIteratorBase::memberName() const | ||||
| { | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
|    const char *name = (*current_).first.c_str(); | ||||
|    return name ? name : ""; | ||||
| #else | ||||
|    if ( !isArray_ ) | ||||
|       return ValueInternalMap::key( iterator_.map_ ); | ||||
|    return ""; | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class ValueConstIterator | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| 
 | ||||
| ValueConstIterator::ValueConstIterator() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
| ValueConstIterator::ValueConstIterator( const Value::ObjectValues::iterator ¤t ) | ||||
|    : ValueIteratorBase( current ) | ||||
| { | ||||
| } | ||||
| #else | ||||
| ValueConstIterator::ValueConstIterator( const ValueInternalArray::IteratorState &state ) | ||||
|    : ValueIteratorBase( state ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| ValueConstIterator::ValueConstIterator( const ValueInternalMap::IteratorState &state ) | ||||
|    : ValueIteratorBase( state ) | ||||
| { | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| ValueConstIterator & | ||||
| ValueConstIterator::operator =( const ValueIteratorBase &other ) | ||||
| { | ||||
|    copy( other ); | ||||
|    return *this; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // class ValueIterator | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| // ////////////////////////////////////////////////////////////////// | ||||
| 
 | ||||
| ValueIterator::ValueIterator() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| #ifndef JSON_VALUE_USE_INTERNAL_MAP | ||||
| ValueIterator::ValueIterator( const Value::ObjectValues::iterator ¤t ) | ||||
|    : ValueIteratorBase( current ) | ||||
| { | ||||
| } | ||||
| #else | ||||
| ValueIterator::ValueIterator( const ValueInternalArray::IteratorState &state ) | ||||
|    : ValueIteratorBase( state ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| ValueIterator::ValueIterator( const ValueInternalMap::IteratorState &state ) | ||||
|    : ValueIteratorBase( state ) | ||||
| { | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| ValueIterator::ValueIterator( const ValueConstIterator &other ) | ||||
|    : ValueIteratorBase( other ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| ValueIterator::ValueIterator( const ValueIterator &other ) | ||||
|    : ValueIteratorBase( other ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| ValueIterator & | ||||
| ValueIterator::operator =( const SelfType &other ) | ||||
| { | ||||
|    copy( other ); | ||||
|    return *this; | ||||
| } | ||||
|  | @ -1,829 +0,0 @@ | |||
| #include "writer.h" | ||||
| #include <utility> | ||||
| #include <assert.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <iostream> | ||||
| #include <sstream> | ||||
| #include <iomanip> | ||||
| 
 | ||||
| #if _MSC_VER >= 1400 // VC++ 8.0
 | ||||
| #pragma warning( disable : 4996 )   // disable warning about strdup being deprecated.
 | ||||
| #endif | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
| static bool isControlCharacter(char ch) | ||||
| { | ||||
|    return ch > 0 && ch <= 0x1F; | ||||
| } | ||||
| 
 | ||||
| static bool containsControlCharacter( const char* str ) | ||||
| { | ||||
|    while ( *str )  | ||||
|    { | ||||
|       if ( isControlCharacter( *(str++) ) ) | ||||
|          return true; | ||||
|    } | ||||
|    return false; | ||||
| } | ||||
| static void uintToString( unsigned int value,  | ||||
|                           char *¤t ) | ||||
| { | ||||
|    *--current = 0; | ||||
|    do | ||||
|    { | ||||
|       *--current = (value % 10) + '0'; | ||||
|       value /= 10; | ||||
|    } | ||||
|    while ( value != 0 ); | ||||
| } | ||||
| 
 | ||||
| std::string valueToString( Int value ) | ||||
| { | ||||
|    char buffer[32]; | ||||
|    char *current = buffer + sizeof(buffer); | ||||
|    bool isNegative = value < 0; | ||||
|    if ( isNegative ) | ||||
|       value = -value; | ||||
|    uintToString( UInt(value), current ); | ||||
|    if ( isNegative ) | ||||
|       *--current = '-'; | ||||
|    assert( current >= buffer ); | ||||
|    return current; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string valueToString( UInt value ) | ||||
| { | ||||
|    char buffer[32]; | ||||
|    char *current = buffer + sizeof(buffer); | ||||
|    uintToString( value, current ); | ||||
|    assert( current >= buffer ); | ||||
|    return current; | ||||
| } | ||||
| 
 | ||||
| std::string valueToString( double value ) | ||||
| { | ||||
|    char buffer[32]; | ||||
| #if defined(_MSC_VER) && defined(__STDC_SECURE_LIB__) // Use secure version with visual studio 2005 to avoid warning. 
 | ||||
|    sprintf_s(buffer, sizeof(buffer), "%#.16g", value);  | ||||
| #else	 | ||||
|    sprintf(buffer, "%#.16g", value);  | ||||
| #endif | ||||
|    char* ch = buffer + strlen(buffer) - 1; | ||||
|    if (*ch != '0') return buffer; // nothing to truncate, so save time
 | ||||
|    while(ch > buffer && *ch == '0'){ | ||||
|      --ch; | ||||
|    } | ||||
|    char* last_nonzero = ch; | ||||
|    while(ch >= buffer){ | ||||
|      switch(*ch){ | ||||
|      case '0': | ||||
|      case '1': | ||||
|      case '2': | ||||
|      case '3': | ||||
|      case '4': | ||||
|      case '5': | ||||
|      case '6': | ||||
|      case '7': | ||||
|      case '8': | ||||
|      case '9': | ||||
|        --ch; | ||||
|        continue; | ||||
|      case '.': | ||||
|        // Truncate zeroes to save bytes in output, but keep one.
 | ||||
|        *(last_nonzero+2) = '\0'; | ||||
|        return buffer; | ||||
|      default: | ||||
|        return buffer; | ||||
|      } | ||||
|    } | ||||
|    return buffer; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string valueToString( bool value ) | ||||
| { | ||||
|    return value ? "true" : "false"; | ||||
| } | ||||
| 
 | ||||
| std::string valueToQuotedString( const char *value ) | ||||
| { | ||||
|    // Not sure how to handle unicode...
 | ||||
|    if (strpbrk(value, "\"\\\b\f\n\r\t") == NULL && !containsControlCharacter( value )) | ||||
|       return std::string("\"") + value + "\""; | ||||
|    // We have to walk value and escape any special characters.
 | ||||
|    // Appending to std::string is not efficient, but this should be rare.
 | ||||
|    // (Note: forward slashes are *not* rare, but I am not escaping them.)
 | ||||
|    unsigned maxsize = strlen(value)*2 + 3; // allescaped+quotes+NULL
 | ||||
|    std::string result; | ||||
|    result.reserve(maxsize); // to avoid lots of mallocs
 | ||||
|    result += "\""; | ||||
|    for (const char* c=value; *c != 0; ++c) | ||||
|    { | ||||
|       switch(*c) | ||||
|       { | ||||
|          case '\"': | ||||
|             result += "\\\""; | ||||
|             break; | ||||
|          case '\\': | ||||
|             result += "\\\\"; | ||||
|             break; | ||||
|          case '\b': | ||||
|             result += "\\b"; | ||||
|             break; | ||||
|          case '\f': | ||||
|             result += "\\f"; | ||||
|             break; | ||||
|          case '\n': | ||||
|             result += "\\n"; | ||||
|             break; | ||||
|          case '\r': | ||||
|             result += "\\r"; | ||||
|             break; | ||||
|          case '\t': | ||||
|             result += "\\t"; | ||||
|             break; | ||||
|          //case '/':
 | ||||
|             // Even though \/ is considered a legal escape in JSON, a bare
 | ||||
|             // slash is also legal, so I see no reason to escape it.
 | ||||
|             // (I hope I am not misunderstanding something.
 | ||||
|             // blep notes: actually escaping \/ may be useful in javascript to avoid </ 
 | ||||
|             // sequence.
 | ||||
|             // Should add a flag to allow this compatibility mode and prevent this 
 | ||||
|             // sequence from occurring.
 | ||||
|          default: | ||||
|             if ( isControlCharacter( *c ) ) | ||||
|             { | ||||
|                std::ostringstream oss; | ||||
|                oss << "\\u" << std::hex << std::uppercase << std::setfill('0') << std::setw(4) << static_cast<int>(*c); | ||||
|                result += oss.str(); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                result += *c; | ||||
|             } | ||||
|             break; | ||||
|       } | ||||
|    } | ||||
|    result += "\""; | ||||
|    return result; | ||||
| } | ||||
| 
 | ||||
| // Class Writer
 | ||||
| // //////////////////////////////////////////////////////////////////
 | ||||
| Writer::~Writer() | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Class FastWriter
 | ||||
| // //////////////////////////////////////////////////////////////////
 | ||||
| 
 | ||||
| FastWriter::FastWriter() | ||||
|    : yamlCompatiblityEnabled_( false ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| FastWriter::enableYAMLCompatibility() | ||||
| { | ||||
|    yamlCompatiblityEnabled_ = true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string  | ||||
| FastWriter::write( const Value &root ) | ||||
| { | ||||
|    document_ = ""; | ||||
|    writeValue( root ); | ||||
|    document_ += "\n"; | ||||
|    return document_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| FastWriter::writeValue( const Value &value ) | ||||
| { | ||||
|    switch ( value.type() ) | ||||
|    { | ||||
|    case nullValue: | ||||
|       document_ += "null"; | ||||
|       break; | ||||
|    case intValue: | ||||
|       document_ += valueToString( value.asInt() ); | ||||
|       break; | ||||
|    case uintValue: | ||||
|       document_ += valueToString( value.asUInt() ); | ||||
|       break; | ||||
|    case realValue: | ||||
|       document_ += valueToString( value.asDouble() ); | ||||
|       break; | ||||
|    case stringValue: | ||||
|       document_ += valueToQuotedString( value.asCString() ); | ||||
|       break; | ||||
|    case booleanValue: | ||||
|       document_ += valueToString( value.asBool() ); | ||||
|       break; | ||||
|    case arrayValue: | ||||
|       { | ||||
|          document_ += "["; | ||||
|          int size = value.size(); | ||||
|          for ( int index =0; index < size; ++index ) | ||||
|          { | ||||
|             if ( index > 0 ) | ||||
|                document_ += ","; | ||||
|             writeValue( value[index] ); | ||||
|          } | ||||
|          document_ += "]"; | ||||
|       } | ||||
|       break; | ||||
|    case objectValue: | ||||
|       { | ||||
|          Value::Members members( value.getMemberNames() ); | ||||
|          document_ += "{"; | ||||
|          for ( Value::Members::iterator it = members.begin();  | ||||
|                it != members.end();  | ||||
|                ++it ) | ||||
|          { | ||||
|             const std::string &name = *it; | ||||
|             if ( it != members.begin() ) | ||||
|                document_ += ","; | ||||
|             document_ += valueToQuotedString( name.c_str() ); | ||||
|             document_ += yamlCompatiblityEnabled_ ? ": "  | ||||
|                                                   : ":"; | ||||
|             writeValue( value[name] ); | ||||
|          } | ||||
|          document_ += "}"; | ||||
|       } | ||||
|       break; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Class StyledWriter
 | ||||
| // //////////////////////////////////////////////////////////////////
 | ||||
| 
 | ||||
| StyledWriter::StyledWriter() | ||||
|    : rightMargin_( 74 ) | ||||
|    , indentSize_( 3 ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string  | ||||
| StyledWriter::write( const Value &root ) | ||||
| { | ||||
|    document_ = ""; | ||||
|    addChildValues_ = false; | ||||
|    indentString_ = ""; | ||||
|    writeCommentBeforeValue( root ); | ||||
|    writeValue( root ); | ||||
|    writeCommentAfterValueOnSameLine( root ); | ||||
|    document_ += "\n"; | ||||
|    return document_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::writeValue( const Value &value ) | ||||
| { | ||||
|    switch ( value.type() ) | ||||
|    { | ||||
|    case nullValue: | ||||
|       pushValue( "null" ); | ||||
|       break; | ||||
|    case intValue: | ||||
|       pushValue( valueToString( value.asInt() ) ); | ||||
|       break; | ||||
|    case uintValue: | ||||
|       pushValue( valueToString( value.asUInt() ) ); | ||||
|       break; | ||||
|    case realValue: | ||||
|       pushValue( valueToString( value.asDouble() ) ); | ||||
|       break; | ||||
|    case stringValue: | ||||
|       pushValue( valueToQuotedString( value.asCString() ) ); | ||||
|       break; | ||||
|    case booleanValue: | ||||
|       pushValue( valueToString( value.asBool() ) ); | ||||
|       break; | ||||
|    case arrayValue: | ||||
|       writeArrayValue( value); | ||||
|       break; | ||||
|    case objectValue: | ||||
|       { | ||||
|          Value::Members members( value.getMemberNames() ); | ||||
|          if ( members.empty() ) | ||||
|             pushValue( "{}" ); | ||||
|          else | ||||
|          { | ||||
|             writeWithIndent( "{" ); | ||||
|             indent(); | ||||
|             Value::Members::iterator it = members.begin(); | ||||
|             while ( true ) | ||||
|             { | ||||
|                const std::string &name = *it; | ||||
|                const Value &childValue = value[name]; | ||||
|                writeCommentBeforeValue( childValue ); | ||||
|                writeWithIndent( valueToQuotedString( name.c_str() ) ); | ||||
|                document_ += " : "; | ||||
|                writeValue( childValue ); | ||||
|                if ( ++it == members.end() ) | ||||
|                { | ||||
|                   writeCommentAfterValueOnSameLine( childValue ); | ||||
|                   break; | ||||
|                } | ||||
|                document_ += ","; | ||||
|                writeCommentAfterValueOnSameLine( childValue ); | ||||
|             } | ||||
|             unindent(); | ||||
|             writeWithIndent( "}" ); | ||||
|          } | ||||
|       } | ||||
|       break; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::writeArrayValue( const Value &value ) | ||||
| { | ||||
|    unsigned size = value.size(); | ||||
|    if ( size == 0 ) | ||||
|       pushValue( "[]" ); | ||||
|    else | ||||
|    { | ||||
|       bool isArrayMultiLine = isMultineArray( value ); | ||||
|       if ( isArrayMultiLine ) | ||||
|       { | ||||
|          writeWithIndent( "[" ); | ||||
|          indent(); | ||||
|          bool hasChildValue = !childValues_.empty(); | ||||
|          unsigned index =0; | ||||
|          while ( true ) | ||||
|          { | ||||
|             const Value &childValue = value[index]; | ||||
|             writeCommentBeforeValue( childValue ); | ||||
|             if ( hasChildValue ) | ||||
|                writeWithIndent( childValues_[index] ); | ||||
|             else | ||||
|             { | ||||
|                writeIndent(); | ||||
|                writeValue( childValue ); | ||||
|             } | ||||
|             if ( ++index == size ) | ||||
|             { | ||||
|                writeCommentAfterValueOnSameLine( childValue ); | ||||
|                break; | ||||
|             } | ||||
|             document_ += ","; | ||||
|             writeCommentAfterValueOnSameLine( childValue ); | ||||
|          } | ||||
|          unindent(); | ||||
|          writeWithIndent( "]" ); | ||||
|       } | ||||
|       else // output on a single line
 | ||||
|       { | ||||
|          assert( childValues_.size() == size ); | ||||
|          document_ += "[ "; | ||||
|          for ( unsigned index =0; index < size; ++index ) | ||||
|          { | ||||
|             if ( index > 0 ) | ||||
|                document_ += ", "; | ||||
|             document_ += childValues_[index]; | ||||
|          } | ||||
|          document_ += " ]"; | ||||
|       } | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| StyledWriter::isMultineArray( const Value &value ) | ||||
| { | ||||
|    int size = value.size(); | ||||
|    bool isMultiLine = size*3 >= rightMargin_ ; | ||||
|    childValues_.clear(); | ||||
|    for ( int index =0; index < size  &&  !isMultiLine; ++index ) | ||||
|    { | ||||
|       const Value &childValue = value[index]; | ||||
|       isMultiLine = isMultiLine  || | ||||
|                      ( (childValue.isArray()  ||  childValue.isObject())  &&   | ||||
|                         childValue.size() > 0 ); | ||||
|    } | ||||
|    if ( !isMultiLine ) // check if line length > max line length
 | ||||
|    { | ||||
|       childValues_.reserve( size ); | ||||
|       addChildValues_ = true; | ||||
|       int lineLength = 4 + (size-1)*2; // '[ ' + ', '*n + ' ]'
 | ||||
|       for ( int index =0; index < size  &&  !isMultiLine; ++index ) | ||||
|       { | ||||
|          writeValue( value[index] ); | ||||
|          lineLength += int( childValues_[index].length() ); | ||||
|          isMultiLine = isMultiLine  &&  hasCommentForValue( value[index] ); | ||||
|       } | ||||
|       addChildValues_ = false; | ||||
|       isMultiLine = isMultiLine  ||  lineLength >= rightMargin_; | ||||
|    } | ||||
|    return isMultiLine; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::pushValue( const std::string &value ) | ||||
| { | ||||
|    if ( addChildValues_ ) | ||||
|       childValues_.push_back( value ); | ||||
|    else | ||||
|       document_ += value; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::writeIndent() | ||||
| { | ||||
|    if ( !document_.empty() ) | ||||
|    { | ||||
|       char last = document_[document_.length()-1]; | ||||
|       if ( last == ' ' )     // already indented
 | ||||
|          return; | ||||
|       if ( last != '\n' )    // Comments may add new-line
 | ||||
|          document_ += '\n'; | ||||
|    } | ||||
|    document_ += indentString_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::writeWithIndent( const std::string &value ) | ||||
| { | ||||
|    writeIndent(); | ||||
|    document_ += value; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::indent() | ||||
| { | ||||
|    indentString_ += std::string( indentSize_, ' ' ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::unindent() | ||||
| { | ||||
|    assert( int(indentString_.size()) >= indentSize_ ); | ||||
|    indentString_.resize( indentString_.size() - indentSize_ ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::writeCommentBeforeValue( const Value &root ) | ||||
| { | ||||
|    if ( !root.hasComment( commentBefore ) ) | ||||
|       return; | ||||
|    document_ += normalizeEOL( root.getComment( commentBefore ) ); | ||||
|    document_ += "\n"; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledWriter::writeCommentAfterValueOnSameLine( const Value &root ) | ||||
| { | ||||
|    if ( root.hasComment( commentAfterOnSameLine ) ) | ||||
|       document_ += " " + normalizeEOL( root.getComment( commentAfterOnSameLine ) ); | ||||
| 
 | ||||
|    if ( root.hasComment( commentAfter ) ) | ||||
|    { | ||||
|       document_ += "\n"; | ||||
|       document_ += normalizeEOL( root.getComment( commentAfter ) ); | ||||
|       document_ += "\n"; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| StyledWriter::hasCommentForValue( const Value &value ) | ||||
| { | ||||
|    return value.hasComment( commentBefore ) | ||||
|           ||  value.hasComment( commentAfterOnSameLine ) | ||||
|           ||  value.hasComment( commentAfter ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string  | ||||
| StyledWriter::normalizeEOL( const std::string &text ) | ||||
| { | ||||
|    std::string normalized; | ||||
|    normalized.reserve( text.length() ); | ||||
|    const char *begin = text.c_str(); | ||||
|    const char *end = begin + text.length(); | ||||
|    const char *current = begin; | ||||
|    while ( current != end ) | ||||
|    { | ||||
|       char c = *current++; | ||||
|       if ( c == '\r' ) // mac or dos EOL
 | ||||
|       { | ||||
|          if ( *current == '\n' ) // convert dos EOL
 | ||||
|             ++current; | ||||
|          normalized += '\n'; | ||||
|       } | ||||
|       else // handle unix EOL & other char
 | ||||
|          normalized += c; | ||||
|    } | ||||
|    return normalized; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Class StyledStreamWriter
 | ||||
| // //////////////////////////////////////////////////////////////////
 | ||||
| 
 | ||||
| StyledStreamWriter::StyledStreamWriter( std::string indentation ) | ||||
|    : document_(NULL) | ||||
|    , rightMargin_( 74 ) | ||||
|    , indentation_( indentation ) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void | ||||
| StyledStreamWriter::write( std::ostream &out, const Value &root ) | ||||
| { | ||||
|    document_ = &out; | ||||
|    addChildValues_ = false; | ||||
|    indentString_ = ""; | ||||
|    writeCommentBeforeValue( root ); | ||||
|    writeValue( root ); | ||||
|    writeCommentAfterValueOnSameLine( root ); | ||||
|    *document_ << "\n"; | ||||
|    document_ = NULL; // Forget the stream, for safety.
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::writeValue( const Value &value ) | ||||
| { | ||||
|    switch ( value.type() ) | ||||
|    { | ||||
|    case nullValue: | ||||
|       pushValue( "null" ); | ||||
|       break; | ||||
|    case intValue: | ||||
|       pushValue( valueToString( value.asInt() ) ); | ||||
|       break; | ||||
|    case uintValue: | ||||
|       pushValue( valueToString( value.asUInt() ) ); | ||||
|       break; | ||||
|    case realValue: | ||||
|       pushValue( valueToString( value.asDouble() ) ); | ||||
|       break; | ||||
|    case stringValue: | ||||
|       pushValue( valueToQuotedString( value.asCString() ) ); | ||||
|       break; | ||||
|    case booleanValue: | ||||
|       pushValue( valueToString( value.asBool() ) ); | ||||
|       break; | ||||
|    case arrayValue: | ||||
|       writeArrayValue( value); | ||||
|       break; | ||||
|    case objectValue: | ||||
|       { | ||||
|          Value::Members members( value.getMemberNames() ); | ||||
|          if ( members.empty() ) | ||||
|             pushValue( "{}" ); | ||||
|          else | ||||
|          { | ||||
|             writeWithIndent( "{" ); | ||||
|             indent(); | ||||
|             Value::Members::iterator it = members.begin(); | ||||
|             while ( true ) | ||||
|             { | ||||
|                const std::string &name = *it; | ||||
|                const Value &childValue = value[name]; | ||||
|                writeCommentBeforeValue( childValue ); | ||||
|                writeWithIndent( valueToQuotedString( name.c_str() ) ); | ||||
|                *document_ << " : "; | ||||
|                writeValue( childValue ); | ||||
|                if ( ++it == members.end() ) | ||||
|                { | ||||
|                   writeCommentAfterValueOnSameLine( childValue ); | ||||
|                   break; | ||||
|                } | ||||
|                *document_ << ","; | ||||
|                writeCommentAfterValueOnSameLine( childValue ); | ||||
|             } | ||||
|             unindent(); | ||||
|             writeWithIndent( "}" ); | ||||
|          } | ||||
|       } | ||||
|       break; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::writeArrayValue( const Value &value ) | ||||
| { | ||||
|    unsigned size = value.size(); | ||||
|    if ( size == 0 ) | ||||
|       pushValue( "[]" ); | ||||
|    else | ||||
|    { | ||||
|       bool isArrayMultiLine = isMultineArray( value ); | ||||
|       if ( isArrayMultiLine ) | ||||
|       { | ||||
|          writeWithIndent( "[" ); | ||||
|          indent(); | ||||
|          bool hasChildValue = !childValues_.empty(); | ||||
|          unsigned index =0; | ||||
|          while ( true ) | ||||
|          { | ||||
|             const Value &childValue = value[index]; | ||||
|             writeCommentBeforeValue( childValue ); | ||||
|             if ( hasChildValue ) | ||||
|                writeWithIndent( childValues_[index] ); | ||||
|             else | ||||
|             { | ||||
| 	       writeIndent(); | ||||
|                writeValue( childValue ); | ||||
|             } | ||||
|             if ( ++index == size ) | ||||
|             { | ||||
|                writeCommentAfterValueOnSameLine( childValue ); | ||||
|                break; | ||||
|             } | ||||
|             *document_ << ","; | ||||
|             writeCommentAfterValueOnSameLine( childValue ); | ||||
|          } | ||||
|          unindent(); | ||||
|          writeWithIndent( "]" ); | ||||
|       } | ||||
|       else // output on a single line
 | ||||
|       { | ||||
|          assert( childValues_.size() == size ); | ||||
|          *document_ << "[ "; | ||||
|          for ( unsigned index =0; index < size; ++index ) | ||||
|          { | ||||
|             if ( index > 0 ) | ||||
|                *document_ << ", "; | ||||
|             *document_ << childValues_[index]; | ||||
|          } | ||||
|          *document_ << " ]"; | ||||
|       } | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| StyledStreamWriter::isMultineArray( const Value &value ) | ||||
| { | ||||
|    int size = value.size(); | ||||
|    bool isMultiLine = size*3 >= rightMargin_ ; | ||||
|    childValues_.clear(); | ||||
|    for ( int index =0; index < size  &&  !isMultiLine; ++index ) | ||||
|    { | ||||
|       const Value &childValue = value[index]; | ||||
|       isMultiLine = isMultiLine  || | ||||
|                      ( (childValue.isArray()  ||  childValue.isObject())  &&   | ||||
|                         childValue.size() > 0 ); | ||||
|    } | ||||
|    if ( !isMultiLine ) // check if line length > max line length
 | ||||
|    { | ||||
|       childValues_.reserve( size ); | ||||
|       addChildValues_ = true; | ||||
|       int lineLength = 4 + (size-1)*2; // '[ ' + ', '*n + ' ]'
 | ||||
|       for ( int index =0; index < size  &&  !isMultiLine; ++index ) | ||||
|       { | ||||
|          writeValue( value[index] ); | ||||
|          lineLength += int( childValues_[index].length() ); | ||||
|          isMultiLine = isMultiLine  &&  hasCommentForValue( value[index] ); | ||||
|       } | ||||
|       addChildValues_ = false; | ||||
|       isMultiLine = isMultiLine  ||  lineLength >= rightMargin_; | ||||
|    } | ||||
|    return isMultiLine; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::pushValue( const std::string &value ) | ||||
| { | ||||
|    if ( addChildValues_ ) | ||||
|       childValues_.push_back( value ); | ||||
|    else | ||||
|       *document_ << value; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::writeIndent() | ||||
| { | ||||
|   /*
 | ||||
|     Some comments in this method would have been nice. ;-) | ||||
| 
 | ||||
|    if ( !document_.empty() ) | ||||
|    { | ||||
|       char last = document_[document_.length()-1]; | ||||
|       if ( last == ' ' )     // already indented
 | ||||
|          return; | ||||
|       if ( last != '\n' )    // Comments may add new-line
 | ||||
|          *document_ << '\n'; | ||||
|    } | ||||
|   */ | ||||
|    *document_ << '\n' << indentString_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::writeWithIndent( const std::string &value ) | ||||
| { | ||||
|    writeIndent(); | ||||
|    *document_ << value; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::indent() | ||||
| { | ||||
|    indentString_ += indentation_; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::unindent() | ||||
| { | ||||
|    assert( indentString_.size() >= indentation_.size() ); | ||||
|    indentString_.resize( indentString_.size() - indentation_.size() ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::writeCommentBeforeValue( const Value &root ) | ||||
| { | ||||
|    if ( !root.hasComment( commentBefore ) ) | ||||
|       return; | ||||
|    *document_ << normalizeEOL( root.getComment( commentBefore ) ); | ||||
|    *document_ << "\n"; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void  | ||||
| StyledStreamWriter::writeCommentAfterValueOnSameLine( const Value &root ) | ||||
| { | ||||
|    if ( root.hasComment( commentAfterOnSameLine ) ) | ||||
|       *document_ << " " + normalizeEOL( root.getComment( commentAfterOnSameLine ) ); | ||||
| 
 | ||||
|    if ( root.hasComment( commentAfter ) ) | ||||
|    { | ||||
|       *document_ << "\n"; | ||||
|       *document_ << normalizeEOL( root.getComment( commentAfter ) ); | ||||
|       *document_ << "\n"; | ||||
|    } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| bool  | ||||
| StyledStreamWriter::hasCommentForValue( const Value &value ) | ||||
| { | ||||
|    return value.hasComment( commentBefore ) | ||||
|           ||  value.hasComment( commentAfterOnSameLine ) | ||||
|           ||  value.hasComment( commentAfter ); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::string  | ||||
| StyledStreamWriter::normalizeEOL( const std::string &text ) | ||||
| { | ||||
|    std::string normalized; | ||||
|    normalized.reserve( text.length() ); | ||||
|    const char *begin = text.c_str(); | ||||
|    const char *end = begin + text.length(); | ||||
|    const char *current = begin; | ||||
|    while ( current != end ) | ||||
|    { | ||||
|       char c = *current++; | ||||
|       if ( c == '\r' ) // mac or dos EOL
 | ||||
|       { | ||||
|          if ( *current == '\n' ) // convert dos EOL
 | ||||
|             ++current; | ||||
|          normalized += '\n'; | ||||
|       } | ||||
|       else // handle unix EOL & other char
 | ||||
|          normalized += c; | ||||
|    } | ||||
|    return normalized; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| std::ostream& operator<<( std::ostream &sout, const Value &root ) | ||||
| { | ||||
|    Json::StyledStreamWriter writer; | ||||
|    writer.write(sout, root); | ||||
|    return sout; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| } // namespace Json
 | ||||
|  | @ -1,196 +0,0 @@ | |||
| #ifndef CPPTL_JSON_READER_H_INCLUDED | ||||
| # define CPPTL_JSON_READER_H_INCLUDED | ||||
| 
 | ||||
| # include "features.h" | ||||
| # include "value.h" | ||||
| # include <deque> | ||||
| # include <stack> | ||||
| # include <string> | ||||
| # include <iostream> | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
|    /** \brief Unserialize a <a HREF="http://www.json.org">JSON</a> document into a Value.
 | ||||
|     * | ||||
|     */ | ||||
|    class JSON_API Reader | ||||
|    { | ||||
|    public: | ||||
|       typedef char Char; | ||||
|       typedef const Char *Location; | ||||
| 
 | ||||
|       /** \brief Constructs a Reader allowing all features
 | ||||
|        * for parsing. | ||||
|        */ | ||||
|       Reader(); | ||||
| 
 | ||||
|       /** \brief Constructs a Reader allowing the specified feature set
 | ||||
|        * for parsing. | ||||
|        */ | ||||
|       Reader( const Features &features ); | ||||
| 
 | ||||
|       /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a> document.
 | ||||
|        * \param document UTF-8 encoded string containing the document to read. | ||||
|        * \param root [out] Contains the root value of the document if it was | ||||
|        *             successfully parsed. | ||||
|        * \param collectComments \c true to collect comment and allow writing them back during | ||||
|        *                        serialization, \c false to discard comments. | ||||
|        *                        This parameter is ignored if Features::allowComments_ | ||||
|        *                        is \c false. | ||||
|        * \return \c true if the document was successfully parsed, \c false if an error occurred. | ||||
|        */ | ||||
|       bool parse( const std::string &document,  | ||||
|                   Value &root, | ||||
|                   bool collectComments = true ); | ||||
| 
 | ||||
|       /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a> document.
 | ||||
|        * \param document UTF-8 encoded string containing the document to read. | ||||
|        * \param root [out] Contains the root value of the document if it was | ||||
|        *             successfully parsed. | ||||
|        * \param collectComments \c true to collect comment and allow writing them back during | ||||
|        *                        serialization, \c false to discard comments. | ||||
|        *                        This parameter is ignored if Features::allowComments_ | ||||
|        *                        is \c false. | ||||
|        * \return \c true if the document was successfully parsed, \c false if an error occurred. | ||||
|        */ | ||||
|       bool parse( const char *beginDoc, const char *endDoc,  | ||||
|                   Value &root, | ||||
|                   bool collectComments = true ); | ||||
| 
 | ||||
|       /// \brief Parse from input stream.
 | ||||
|       /// \see Json::operator>>(std::istream&, Json::Value&).
 | ||||
|       bool parse( std::istream &is, | ||||
|                   Value &root, | ||||
|                   bool collectComments = true ); | ||||
| 
 | ||||
|       /** \brief Returns a user friendly string that list errors in the parsed document.
 | ||||
|        * \return Formatted error message with the list of errors with their location in  | ||||
|        *         the parsed document. An empty string is returned if no error occurred | ||||
|        *         during parsing. | ||||
|        */ | ||||
|       std::string getFormatedErrorMessages() const; | ||||
| 
 | ||||
|    private: | ||||
|       enum TokenType | ||||
|       { | ||||
|          tokenEndOfStream = 0, | ||||
|          tokenObjectBegin, | ||||
|          tokenObjectEnd, | ||||
|          tokenArrayBegin, | ||||
|          tokenArrayEnd, | ||||
|          tokenString, | ||||
|          tokenNumber, | ||||
|          tokenTrue, | ||||
|          tokenFalse, | ||||
|          tokenNull, | ||||
|          tokenArraySeparator, | ||||
|          tokenMemberSeparator, | ||||
|          tokenComment, | ||||
|          tokenError | ||||
|       }; | ||||
| 
 | ||||
|       class Token | ||||
|       { | ||||
|       public: | ||||
|          TokenType type_; | ||||
|          Location start_; | ||||
|          Location end_; | ||||
|       }; | ||||
| 
 | ||||
|       class ErrorInfo | ||||
|       { | ||||
|       public: | ||||
|          Token token_; | ||||
|          std::string message_; | ||||
|          Location extra_; | ||||
|       }; | ||||
| 
 | ||||
|       typedef std::deque<ErrorInfo> Errors; | ||||
| 
 | ||||
|       bool expectToken( TokenType type, Token &token, const char *message ); | ||||
|       bool readToken( Token &token ); | ||||
|       void skipSpaces(); | ||||
|       bool match( Location pattern,  | ||||
|                   int patternLength ); | ||||
|       bool readComment(); | ||||
|       bool readCStyleComment(); | ||||
|       bool readCppStyleComment(); | ||||
|       bool readString(); | ||||
|       void readNumber(); | ||||
|       bool readValue(); | ||||
|       bool readObject( Token &token ); | ||||
|       bool readArray( Token &token ); | ||||
|       bool decodeNumber( Token &token ); | ||||
|       bool decodeString( Token &token ); | ||||
|       bool decodeString( Token &token, std::string &decoded ); | ||||
|       bool decodeDouble( Token &token ); | ||||
|       bool decodeUnicodeCodePoint( Token &token,  | ||||
|                                    Location ¤t,  | ||||
|                                    Location end,  | ||||
|                                    unsigned int &unicode ); | ||||
|       bool decodeUnicodeEscapeSequence( Token &token,  | ||||
|                                         Location ¤t,  | ||||
|                                         Location end,  | ||||
|                                         unsigned int &unicode ); | ||||
|       bool addError( const std::string &message,  | ||||
|                      Token &token, | ||||
|                      Location extra = 0 ); | ||||
|       bool recoverFromError( TokenType skipUntilToken ); | ||||
|       bool addErrorAndRecover( const std::string &message,  | ||||
|                                Token &token, | ||||
|                                TokenType skipUntilToken ); | ||||
|       void skipUntilSpace(); | ||||
|       Value ¤tValue(); | ||||
|       Char getNextChar(); | ||||
|       void getLocationLineAndColumn( Location location, | ||||
|                                      int &line, | ||||
|                                      int &column ) const; | ||||
|       std::string getLocationLineAndColumn( Location location ) const; | ||||
|       void addComment( Location begin,  | ||||
|                        Location end,  | ||||
|                        CommentPlacement placement ); | ||||
|       void skipCommentTokens( Token &token ); | ||||
|     | ||||
|       typedef std::stack<Value *> Nodes; | ||||
|       Nodes nodes_; | ||||
|       Errors errors_; | ||||
|       std::string document_; | ||||
|       Location begin_; | ||||
|       Location end_; | ||||
|       Location current_; | ||||
|       Location lastValueEnd_; | ||||
|       Value *lastValue_; | ||||
|       std::string commentsBefore_; | ||||
|       Features features_; | ||||
|       bool collectComments_; | ||||
|    }; | ||||
| 
 | ||||
|    /** \brief Read from 'sin' into 'root'.
 | ||||
| 
 | ||||
|     Always keep comments from the input JSON. | ||||
| 
 | ||||
|     This can be used to read a file into a particular sub-object. | ||||
|     For example: | ||||
|     \code | ||||
|     Json::Value root; | ||||
|     cin >> root["dir"]["file"]; | ||||
|     cout << root; | ||||
|     \endcode | ||||
|     Result: | ||||
|     \verbatim | ||||
|     { | ||||
| 	"dir": { | ||||
| 	    "file": { | ||||
| 		// The input stream JSON would be nested here.
 | ||||
| 	    } | ||||
| 	} | ||||
|     } | ||||
|     \endverbatim | ||||
|     \throw std::exception on parse error. | ||||
|     \see Json::operator<<() | ||||
|    */ | ||||
|    std::istream& operator>>( std::istream&, Value& ); | ||||
| 
 | ||||
| } // namespace Json
 | ||||
| 
 | ||||
| #endif // CPPTL_JSON_READER_H_INCLUDED
 | ||||
							
								
								
									
										1069
									
								
								util/json/value.h
									
										
									
									
									
								
							
							
						
						
									
										1069
									
								
								util/json/value.h
									
										
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -1,174 +0,0 @@ | |||
| #ifndef JSON_WRITER_H_INCLUDED | ||||
| # define JSON_WRITER_H_INCLUDED | ||||
| 
 | ||||
| # include "value.h" | ||||
| # include <vector> | ||||
| # include <string> | ||||
| # include <iostream> | ||||
| 
 | ||||
| namespace Json { | ||||
| 
 | ||||
|    class Value; | ||||
| 
 | ||||
|    /** \brief Abstract class for writers.
 | ||||
|     */ | ||||
|    class JSON_API Writer | ||||
|    { | ||||
|    public: | ||||
|       virtual ~Writer(); | ||||
| 
 | ||||
|       virtual std::string write( const Value &root ) = 0; | ||||
|    }; | ||||
| 
 | ||||
|    /** \brief Outputs a Value in <a HREF="http://www.json.org">JSON</a> format without formatting (not human friendly).
 | ||||
|     * | ||||
|     * The JSON document is written in a single line. It is not intended for 'human' consumption, | ||||
|     * but may be usefull to support feature such as RPC where bandwith is limited. | ||||
|     * \sa Reader, Value | ||||
|     */ | ||||
|    class JSON_API FastWriter : public Writer | ||||
|    { | ||||
|    public: | ||||
|       FastWriter(); | ||||
|       virtual ~FastWriter(){} | ||||
| 
 | ||||
|       void enableYAMLCompatibility(); | ||||
| 
 | ||||
|    public: // overridden from Writer
 | ||||
|       virtual std::string write( const Value &root ); | ||||
| 
 | ||||
|    private: | ||||
|       void writeValue( const Value &value ); | ||||
| 
 | ||||
|       std::string document_; | ||||
|       bool yamlCompatiblityEnabled_; | ||||
|    }; | ||||
| 
 | ||||
|    /** \brief Writes a Value in <a HREF="http://www.json.org">JSON</a> format in a human friendly way.
 | ||||
|     * | ||||
|     * The rules for line break and indent are as follow: | ||||
|     * - Object value: | ||||
|     *     - if empty then print {} without indent and line break | ||||
|     *     - if not empty the print '{', line break & indent, print one value per line | ||||
|     *       and then unindent and line break and print '}'. | ||||
|     * - Array value: | ||||
|     *     - if empty then print [] without indent and line break | ||||
|     *     - if the array contains no object value, empty array or some other value types, | ||||
|     *       and all the values fit on one lines, then print the array on a single line. | ||||
|     *     - otherwise, it the values do not fit on one line, or the array contains | ||||
|     *       object or non empty array, then print one value per line. | ||||
|     * | ||||
|     * If the Value have comments then they are outputed according to their #CommentPlacement. | ||||
|     * | ||||
|     * \sa Reader, Value, Value::setComment() | ||||
|     */ | ||||
|    class JSON_API StyledWriter: public Writer | ||||
|    { | ||||
|    public: | ||||
|       StyledWriter(); | ||||
|       virtual ~StyledWriter(){} | ||||
| 
 | ||||
|    public: // overridden from Writer
 | ||||
|       /** \brief Serialize a Value in <a HREF="http://www.json.org">JSON</a> format.
 | ||||
|        * \param root Value to serialize. | ||||
|        * \return String containing the JSON document that represents the root value. | ||||
|        */ | ||||
|       virtual std::string write( const Value &root ); | ||||
| 
 | ||||
|    private: | ||||
|       void writeValue( const Value &value ); | ||||
|       void writeArrayValue( const Value &value ); | ||||
|       bool isMultineArray( const Value &value ); | ||||
|       void pushValue( const std::string &value ); | ||||
|       void writeIndent(); | ||||
|       void writeWithIndent( const std::string &value ); | ||||
|       void indent(); | ||||
|       void unindent(); | ||||
|       void writeCommentBeforeValue( const Value &root ); | ||||
|       void writeCommentAfterValueOnSameLine( const Value &root ); | ||||
|       bool hasCommentForValue( const Value &value ); | ||||
|       static std::string normalizeEOL( const std::string &text ); | ||||
| 
 | ||||
|       typedef std::vector<std::string> ChildValues; | ||||
| 
 | ||||
|       ChildValues childValues_; | ||||
|       std::string document_; | ||||
|       std::string indentString_; | ||||
|       int rightMargin_; | ||||
|       int indentSize_; | ||||
|       bool addChildValues_; | ||||
|    }; | ||||
| 
 | ||||
|    /** \brief Writes a Value in <a HREF="http://www.json.org">JSON</a> format in a human friendly way,
 | ||||
|         to a stream rather than to a string. | ||||
|     * | ||||
|     * The rules for line break and indent are as follow: | ||||
|     * - Object value: | ||||
|     *     - if empty then print {} without indent and line break | ||||
|     *     - if not empty the print '{', line break & indent, print one value per line | ||||
|     *       and then unindent and line break and print '}'. | ||||
|     * - Array value: | ||||
|     *     - if empty then print [] without indent and line break | ||||
|     *     - if the array contains no object value, empty array or some other value types, | ||||
|     *       and all the values fit on one lines, then print the array on a single line. | ||||
|     *     - otherwise, it the values do not fit on one line, or the array contains | ||||
|     *       object or non empty array, then print one value per line. | ||||
|     * | ||||
|     * If the Value have comments then they are outputed according to their #CommentPlacement. | ||||
|     * | ||||
|     * \param indentation Each level will be indented by this amount extra. | ||||
|     * \sa Reader, Value, Value::setComment() | ||||
|     */ | ||||
|    class JSON_API StyledStreamWriter | ||||
|    { | ||||
|    public: | ||||
|       StyledStreamWriter( std::string indentation="\t" ); | ||||
|       ~StyledStreamWriter(){} | ||||
| 
 | ||||
|    public: | ||||
|       /** \brief Serialize a Value in <a HREF="http://www.json.org">JSON</a> format.
 | ||||
|        * \param out Stream to write to. (Can be ostringstream, e.g.) | ||||
|        * \param root Value to serialize. | ||||
|        * \note There is no point in deriving from Writer, since write() should not return a value. | ||||
|        */ | ||||
|       void write( std::ostream &out, const Value &root ); | ||||
| 
 | ||||
|    private: | ||||
|       void writeValue( const Value &value ); | ||||
|       void writeArrayValue( const Value &value ); | ||||
|       bool isMultineArray( const Value &value ); | ||||
|       void pushValue( const std::string &value ); | ||||
|       void writeIndent(); | ||||
|       void writeWithIndent( const std::string &value ); | ||||
|       void indent(); | ||||
|       void unindent(); | ||||
|       void writeCommentBeforeValue( const Value &root ); | ||||
|       void writeCommentAfterValueOnSameLine( const Value &root ); | ||||
|       bool hasCommentForValue( const Value &value ); | ||||
|       static std::string normalizeEOL( const std::string &text ); | ||||
| 
 | ||||
|       typedef std::vector<std::string> ChildValues; | ||||
| 
 | ||||
|       ChildValues childValues_; | ||||
|       std::ostream* document_; | ||||
|       std::string indentString_; | ||||
|       int rightMargin_; | ||||
|       std::string indentation_; | ||||
|       bool addChildValues_; | ||||
|    }; | ||||
| 
 | ||||
|    std::string JSON_API valueToString( Int value ); | ||||
|    std::string JSON_API valueToString( UInt value ); | ||||
|    std::string JSON_API valueToString( double value ); | ||||
|    std::string JSON_API valueToString( bool value ); | ||||
|    std::string JSON_API valueToQuotedString( const char *value ); | ||||
| 
 | ||||
|    /// \brief Output using the StyledStreamWriter.
 | ||||
|    /// \see Json::operator>>()
 | ||||
|    std::ostream& operator<<( std::ostream&, const Value &root ); | ||||
| 
 | ||||
| } // namespace Json
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| #endif // JSON_WRITER_H_INCLUDED
 | ||||
|  | @ -1,7 +1,7 @@ | |||
| /// \file util.cpp
 | ||||
| /// Contains generic functions for managing processes and configuration.
 | ||||
| /// \file procs.cpp
 | ||||
| /// Contains generic functions for managing processes.
 | ||||
| 
 | ||||
| #include "util.h" | ||||
| #include "procs.h" | ||||
| #include <string.h> | ||||
| #include <sys/types.h> | ||||
| #include <signal.h> | ||||
|  | @ -16,37 +16,12 @@ | |||
| #include <sys/types.h> | ||||
| #include <fcntl.h> | ||||
| #include <pwd.h> | ||||
| #include <getopt.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <fstream> | ||||
| 
 | ||||
| std::map<pid_t, std::string> Util::Procs::plist; | ||||
| bool Util::Procs::handler_set = false; | ||||
| 
 | ||||
| /// Sets the current process' running user
 | ||||
| void Util::setUser(std::string username){ | ||||
|   if (username != "root"){ | ||||
|     struct passwd * user_info = getpwnam(username.c_str()); | ||||
|     if (!user_info){ | ||||
|       #if DEBUG >= 1 | ||||
|       fprintf(stderr, "Error: could not setuid %s: could not get PID\n", username.c_str()); | ||||
|       #endif | ||||
|       return; | ||||
|     }else{ | ||||
|       if (setuid(user_info->pw_uid) != 0){ | ||||
|         #if DEBUG >= 1 | ||||
|         fprintf(stderr, "Error: could not setuid %s: not allowed\n", username.c_str()); | ||||
|         #endif | ||||
|       }else{ | ||||
|         #if DEBUG >= 3 | ||||
|         fprintf(stderr, "Changed user to %s\n", username.c_str()); | ||||
|         #endif | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// Used internally to capture child signals and update plist.
 | ||||
| void Util::Procs::childsig_handler(int signum){ | ||||
|   if (signum != SIGCHLD){return;} | ||||
|  | @ -263,108 +238,3 @@ std::string Util::Procs::getName(pid_t name){ | |||
|   } | ||||
|   return ""; | ||||
| } | ||||
| 
 | ||||
| /// Creates a new configuration manager.
 | ||||
| Util::Config::Config(){ | ||||
|   listen_port = 4242; | ||||
|   daemon_mode = true; | ||||
|   interface = "0.0.0.0"; | ||||
|   configfile = "/etc/ddvtech.conf"; | ||||
|   username = "root"; | ||||
|   ignore_daemon = false; | ||||
|   ignore_interface = false; | ||||
|   ignore_port = false; | ||||
|   ignore_user = false; | ||||
| } | ||||
| 
 | ||||
| /// Parses commandline arguments.
 | ||||
| /// Calls exit if an unknown option is encountered, printing a help message.
 | ||||
| /// confsection must be either already set or never be set at all when this function is called.
 | ||||
| /// In other words: do not change confsection after calling this function.
 | ||||
| void Util::Config::parseArgs(int argc, char ** argv){ | ||||
|   int opt = 0; | ||||
|   static const char *optString = "ndvp:i:u:c:h?"; | ||||
|   static const struct option longOpts[] = { | ||||
|     {"help",0,0,'h'}, | ||||
|     {"port",1,0,'p'}, | ||||
|     {"interface",1,0,'i'}, | ||||
|     {"username",1,0,'u'}, | ||||
|     {"no-daemon",0,0,'n'}, | ||||
|     {"daemon",0,0,'d'}, | ||||
|     {"configfile",1,0,'c'}, | ||||
|     {"version",0,0,'v'} | ||||
|   }; | ||||
|   while ((opt = getopt_long(argc, argv, optString, longOpts, 0)) != -1){ | ||||
|     switch (opt){ | ||||
|       case 'p': listen_port = atoi(optarg); ignore_port = true; break; | ||||
|       case 'i': interface = optarg; ignore_interface = true; break; | ||||
|       case 'n': daemon_mode = false; ignore_daemon = true; break; | ||||
|       case 'd': daemon_mode = true; ignore_daemon = true; break; | ||||
|       case 'c': configfile = optarg; break; | ||||
|       case 'u': username = optarg; ignore_user = true; break; | ||||
|       case 'v': | ||||
|         printf("%s\n", TOSTRING(VERSION)); | ||||
|         exit(1); | ||||
|         break; | ||||
|       case 'h': | ||||
|       case '?': | ||||
|         std::string doingdaemon = "true"; | ||||
|         if (!daemon_mode){doingdaemon = "false";} | ||||
|         if (confsection == ""){ | ||||
|           printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -u[sername] VAL\n"); | ||||
|           printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), username.c_str()); | ||||
|         }else{ | ||||
|           printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -c[onfigfile] VAL, -u[sername] VAL\n"); | ||||
|           printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  configfile: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), configfile.c_str(), username.c_str()); | ||||
|           printf("Username root means no change to UID, no matter what the UID is.\n"); | ||||
|           printf("If the configfile exists, it is always loaded first. Commandline settings then overwrite the config file.\n"); | ||||
|           printf("\nThis process takes it directives from the %s section of the configfile.\n", confsection.c_str()); | ||||
|         } | ||||
|         printf("This is %s version %s\n", argv[0], TOSTRING(VERSION)); | ||||
|         exit(1); | ||||
|         break; | ||||
|     } | ||||
|   }//commandline options parser
 | ||||
| } | ||||
| 
 | ||||
| /// Parses the configuration file at configfile, if it exists.
 | ||||
| /// Assumes confsection is set.
 | ||||
| void Util::Config::parseFile(){ | ||||
|   std::ifstream conf(configfile.c_str(), std::ifstream::in); | ||||
|   std::string tmpstr; | ||||
|   bool acc_comm = false; | ||||
|   size_t foundeq; | ||||
|   if (conf.fail()){ | ||||
|     #if DEBUG >= 3 | ||||
|     fprintf(stderr, "Configuration file %s not found - using build-in defaults...\n", configfile.c_str()); | ||||
|     #endif | ||||
|   }else{ | ||||
|     while (conf.good()){ | ||||
|       getline(conf, tmpstr); | ||||
|       if (tmpstr[0] == '['){//new section? check if we care.
 | ||||
|         if (tmpstr == confsection){acc_comm = true;}else{acc_comm = false;} | ||||
|       }else{ | ||||
|         if (!acc_comm){break;}//skip all lines in this section if we do not care about it
 | ||||
|         foundeq = tmpstr.find('='); | ||||
|         if (foundeq != std::string::npos){ | ||||
|           if ((tmpstr.substr(0, foundeq) == "port") && !ignore_port){listen_port = atoi(tmpstr.substr(foundeq+1).c_str());} | ||||
|           if ((tmpstr.substr(0, foundeq) == "interface") && !ignore_interface){interface = tmpstr.substr(foundeq+1);} | ||||
|           if ((tmpstr.substr(0, foundeq) == "username") && !ignore_user){username = tmpstr.substr(foundeq+1);} | ||||
|           if ((tmpstr.substr(0, foundeq) == "daemon") && !ignore_daemon){daemon_mode = true;} | ||||
|           if ((tmpstr.substr(0, foundeq) == "nodaemon") && !ignore_daemon){daemon_mode = false;} | ||||
|         }//found equals sign
 | ||||
|       }//section contents
 | ||||
|     }//configfile line loop
 | ||||
|   }//configuration
 | ||||
| } | ||||
| 
 | ||||
| /// Will turn the current process into a daemon.
 | ||||
| /// Works by calling daemon(1,0):
 | ||||
| /// Does not change directory to root.
 | ||||
| /// Does redirect output to /dev/null
 | ||||
| void Util::Daemonize(){ | ||||
|   #if DEBUG >= 3 | ||||
|   fprintf(stderr, "Going into background mode...\n"); | ||||
|   #endif | ||||
|   daemon(1, 0); | ||||
| } | ||||
|  | @ -1,13 +1,10 @@ | |||
| /// \file util.h
 | ||||
| /// Contains generic function headers for managing processes and configuration.
 | ||||
| /// \file procs.h
 | ||||
| /// Contains generic function headers for managing processes.
 | ||||
| 
 | ||||
| #include <unistd.h> | ||||
| #include <string> | ||||
| #include <map> | ||||
| 
 | ||||
| #define STRINGIFY(x) #x | ||||
| #define TOSTRING(x) STRINGIFY(x) | ||||
| 
 | ||||
| /// Contains utility code, not directly related to streaming media
 | ||||
| namespace Util{ | ||||
| 
 | ||||
|  | @ -31,29 +28,4 @@ namespace Util{ | |||
|       static std::string getName(pid_t name); | ||||
|   }; | ||||
| 
 | ||||
|   /// Will set the active user to the named username.
 | ||||
|   void setUser(std::string user); | ||||
| 
 | ||||
|   /// Deals with parsing configuration from files or commandline options.
 | ||||
|   class Config{ | ||||
|   private: | ||||
|     bool ignore_daemon; | ||||
|     bool ignore_interface; | ||||
|     bool ignore_port; | ||||
|     bool ignore_user; | ||||
|   public: | ||||
|     std::string confsection; | ||||
|     std::string configfile; | ||||
|     bool daemon_mode; | ||||
|     std::string interface; | ||||
|     int listen_port; | ||||
|     std::string username; | ||||
|     Config(); | ||||
|     void parseArgs(int argc, char ** argv); | ||||
|     void parseFile(); | ||||
|   }; | ||||
| 
 | ||||
|   /// Will turn the current process into a daemon.
 | ||||
|   void Daemonize(); | ||||
| 
 | ||||
| }; | ||||
|  | @ -22,7 +22,7 @@ | |||
| #endif | ||||
| 
 | ||||
| #include "socket.h" //Socket library
 | ||||
| #include "util.h" //utilities for process spawning and config management
 | ||||
| #include "config.h" //utilities for config management
 | ||||
| #include <signal.h> | ||||
| #include <sys/types.h> | ||||
| #include <pwd.h> | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Thulinma
						Thulinma