/* * TinyJson 1.3.0 * A Minimalistic JSON Reader Based On Boost.Spirit, Boost.Any, and Boost.Smart_Ptr. * * Copyright (c) 2008 Thomas Jansen (thomas@beef.de) * * Distributed under the Boost Software License, Version 1.0. (See accompanying * file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) * * See http://blog.beef.de/projects/tinyjson/ for documentation. * * (view source with tab-size = 3) * * 16 Mar 2009 - allow root of JSON to be array (Jim Porter) * 29 Mar 2008 - use strict_real_p for number parsing, small cleanup (Thomas Jansen) * 26 Mar 2008 - made json::grammar a template (Boris Schaeling) * 20 Mar 2008 - optimized by using boost::shared_ptr (Thomas Jansen) * 29 Jan 2008 - Small bugfixes (Thomas Jansen) * 04 Jan 2008 - Released to the public (Thomas Jansen) * 13 Nov 2007 - initial release (Thomas Jansen) * * * 29 Mar 2008 */ #ifndef TINYJSON_HPP #define TINYJSON_HPP #include <boost/shared_ptr.hpp> #include <boost/any.hpp> #include <boost/spirit/core.hpp> #include <boost/spirit/utility/loops.hpp> #include <boost/lexical_cast.hpp> #include <string> #include <stack> #include <utility> #include <deque> #include <map> namespace json { boost::spirit::int_parser<long long> const longlong_p = boost::spirit::int_parser<long long>(); // ========================================================================================================== // === U N I C O D E _ C O N V E R T === // ========================================================================================================== template< typename Char > struct unicodecvt { static std::basic_string< Char > convert(int iUnicode) { return std::basic_string< Char >(1, static_cast< Char >(iUnicode)); } }; // ---[ TEMPLATE SPECIALIZATION FOR CHAR ]-------------------------------------------------------------------- template<> struct unicodecvt< char > { static std::string convert(int iUnicode) { std::string strString; if(iUnicode < 0x0080) { // character 0x0000 - 0x007f... strString.push_back(0x00 | ((iUnicode & 0x007f) >> 0)); } else if(iUnicode < 0x0800) { // character 0x0080 - 0x07ff... strString.push_back(0xc0 | ((iUnicode & 0x07c0) >> 6)); strString.push_back(0x80 | ((iUnicode & 0x003f) >> 0)); } else { // character 0x0800 - 0xffff... strString.push_back(0xe0 | ((iUnicode & 0x00f000) >> 12)); strString.push_back(0x80 | ((iUnicode & 0x000fc0) >> 6)); strString.push_back(0x80 | ((iUnicode & 0x00003f) >> 0)); } return strString; } }; // ========================================================================================================== // === T H E J S O N G R A M M A R === // ========================================================================================================== template< typename Char > class grammar : public boost::spirit::grammar< grammar< Char > > { public: // ---[ TYPEDEFINITIONS ]--------------------------------------------------------------------------------- typedef boost::shared_ptr< boost::any > variant; // pointer to a shared variant typedef std::stack< variant > stack; // a stack of json variants typedef std::pair< std::basic_string< Char >, variant > pair; // a pair as it appears in json typedef std::deque< variant > array; // an array of json variants typedef std::map< std::basic_string< Char >, variant > object; // an object with json pairs protected: // ---[ SEMANTIC ACTION: PUSH A STRING ON THE STACK (AND ENCODE AS UTF-8) ]------------------------------- struct push_string { stack & m_stack; push_string(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator szStart, Iterator szEnd) const { // 1: skip the quotes... ++szStart; --szEnd; // 2: traverse through the original string and check for escape codes.. std::basic_string< typename Iterator::value_type > strString; while(szStart < szEnd) { // 2.1: if it's no escape code, just append to the resulting string... if(*szStart != static_cast< typename Iterator::value_type >('\\')) { // 2.1.1: append the character... strString.push_back(*szStart); } else { // 2.1.2: otherwise, check the escape code... ++szStart; switch(*szStart) { default: strString.push_back(*szStart); break; case 'b': strString.push_back(static_cast< typename Iterator::value_type >('\b')); break; case 'f': strString.push_back(static_cast< typename Iterator::value_type >('\f')); break; case 'n': strString.push_back(static_cast< typename Iterator::value_type >('\n')); break; case 'r': strString.push_back(static_cast< typename Iterator::value_type >('\r')); break; case 't': strString.push_back(static_cast< typename Iterator::value_type >('\t')); break; case 'u': { // 2.1.2.1: convert the following hex value into an int... int iUnicode; std::basic_istringstream< Char >(std::basic_string< typename Iterator::value_type >(&szStart[1], 4)) >> std::hex >> iUnicode; szStart += 4; // 2.1.2.2: append the unicode int... strString.append(unicodecvt< typename Iterator::value_type >::convert(iUnicode)); } } } // 2.2: go on with the next character... ++szStart; } // 3: finally, push the string on the stack... m_stack.push(variant(new boost::any(strString))); } }; // ---[ SEMANTIC ACTION: PUSH A REAL ON THE STACK ]------------------------------------------------------- struct push_double { stack & m_stack; push_double(stack & stack) : m_stack(stack) { } void operator() (double dValue) const { m_stack.push(variant(new boost::any(dValue))); } }; // ---[ SEMANTIC ACTION: PUSH AN INT ON THE STACK ]------------------------------------------------------- struct push_int { stack & m_stack; push_int(stack & stack) : m_stack(stack) { } void operator() (long long iValue) const { m_stack.push(variant(new boost::any(iValue))); } }; // ---[ SEMANTIC ACTION: PUSH A BOOLEAN ON THE STACK ]---------------------------------------------------- struct push_boolean { stack & m_stack; push_boolean(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator szStart, Iterator /* szEnd */ ) const { // 1: push a boolean that is "true" if the string starts with 't' and "false" otherwise... m_stack.push(variant(new boost::any(*szStart == static_cast< typename Iterator::value_type >('t')))); } }; // ---[ SEMANTIC ACTION: PUSH A NULL VALUE ON THE STACK ]------------------------------------------------- struct push_null { stack & m_stack; push_null(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator /* szStart */ , Iterator /* szEnd */ ) const { m_stack.push(variant(new boost::any())); } }; // ---[ SEMANTIC ACTION: CREATE A "JSON PAIR" ON THE STACK ]---------------------------------------------- struct create_pair { stack & m_stack; create_pair(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator /* szStart */, Iterator /* szEnd */ ) const { // 1: get the variant from the stack... variant var = m_stack.top(); m_stack.pop(); // 2: get the name from the stack... std::basic_string< typename Iterator::value_type > strName; try { strName = boost::any_cast< std::basic_string< typename Iterator::value_type > >(*m_stack.top()); } catch(boost::bad_any_cast &) { /* NOTHING */ } m_stack.pop(); // 3: push a pair of both on the stack... m_stack.push(variant(new boost::any(pair(strName, var)))); } }; // ---[ SEMANTIC ACTION: BEGIN AN ARRAY ]----------------------------------------------------------------- class array_delimiter { /* EMPTY CLASS */ }; struct begin_array { stack & m_stack; begin_array(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator /* cCharacter */) const { m_stack.push(variant(new boost::any(array_delimiter()))); } }; // ---[ SEMANTIC ACTION: CREATE AN ARRAY FROM THE VALUES ON THE STACK ]----------------------------------- struct end_array { stack & m_stack; end_array(stack & stack) : m_stack(stack) { } // - -[ functional operator ] - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - template <typename Iterator> void operator() (Iterator /* cCharacter */) const { // 1: create an array object and push everything in it, that's on the stack... variant varArray(new boost::any(array())); while(!m_stack.empty()) { // 1.1: get the top most variant of the stack... variant var = m_stack.top(); m_stack.pop(); // 1.2: is it the end of the array? if yes => break the loop... if(boost::any_cast< array_delimiter >(var.get()) != NULL) { break; } // 1.3: otherwise, add to the array... boost::any_cast< array >(varArray.get())->push_front(var); } // 2: finally, push the array at the end of the stack... m_stack.push(varArray); } }; // ---[ SEMANTIC ACTION: BEGIN AN OBJECT ]---------------------------------------------------------------- class object_delimiter { /* EMPTY CLASS */ }; struct begin_object { stack & m_stack; begin_object(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator /* cCharacter */) const { m_stack.push(variant(new boost::any(object_delimiter()))); } }; // ---[ SEMANTIC ACTION: CREATE AN OBJECT FROM THE VALUES ON THE STACK ]---------------------------------- struct end_object { stack & m_stack; end_object(stack & stack) : m_stack(stack) { } template <typename Iterator> void operator() (Iterator /* cCharacter */) const { // 1: create an array object and push everything in it, that's on the stack... variant varObject(new boost::any(object())); while(!m_stack.empty()) { // 1.1: get the top most variant of the stack... variant var = m_stack.top(); m_stack.pop(); // 1.2: is it the end of the array? if yes => break the loop... if(boost::any_cast< object_delimiter >(var.get()) != NULL) { break; } // 1.3: if this is not a pair, we have a problem... pair * pPair = boost::any_cast< pair >(var.get()); if(!pPair) { /* BIG PROBLEM!! */ continue; } // 1.4: set the child of this object... boost::any_cast< object >(varObject.get())->insert(std::make_pair(pPair->first, pPair->second)); } // 2: finally, push the array at the end of the stack... m_stack.push(varObject); } }; public: stack & m_stack; grammar(stack & stack) : m_stack(stack) { } // ---[ THE ACTUAL GRAMMAR DEFINITION ]------------------------------------------------------------------- template <typename SCANNER> class definition { boost::spirit::rule< SCANNER > m_start; boost::spirit::rule< SCANNER > m_object; boost::spirit::rule< SCANNER > m_array; boost::spirit::rule< SCANNER > m_pair; boost::spirit::rule< SCANNER > m_value; boost::spirit::rule< SCANNER > m_string; boost::spirit::rule< SCANNER > m_number; boost::spirit::rule< SCANNER > m_boolean; boost::spirit::rule< SCANNER > m_null; public: boost::spirit::rule< SCANNER > const & start() const { return m_start; } // - -[ create the definition ] - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - definition(grammar const & self) { using namespace boost::spirit; // 0: JSON can either be an object or an array m_start = m_object | m_array; // 1: an object is an unordered set of pairs (seperated by commas)... m_object = ch_p('{') [ begin_object(self.m_stack) ] >> !(m_pair >> *(ch_p(',') >> m_pair)) >> ch_p('}') [ end_object (self.m_stack) ]; // 2: an array is an ordered collection of values (seperated by commas)... m_array = ch_p('[') [ begin_array(self.m_stack) ] >> !(m_value >> *(ch_p(',') >> m_value)) >> ch_p(']') [ end_array (self.m_stack) ]; // 3: a pair is given by a name and a value... m_pair = ( m_string >> ch_p(':') >> m_value ) [ create_pair(self.m_stack) ] ; // 4: a value can be a string in double quotes, a number, a boolean, an object or an array. m_value = m_string | m_number | m_object | m_array | m_boolean | m_null ; // 5: a string is a collection of zero or more unicode characters, wrapped in double quotes... m_string = lexeme_d [ ( ch_p('"') >> *( ( (anychar_p - (ch_p('"') | ch_p('\\'))) | ch_p('\\') >> ( ch_p('\"') | ch_p('\\') | ch_p('/') | ch_p('b') | ch_p('f') | ch_p('n') | ch_p('r') | ch_p('t') | (ch_p('u') >> repeat_p(4)[ xdigit_p ]) ) )) >> ch_p('"') ) [ push_string(self.m_stack) ] ] ; // 6: a number is very much like a C or java number... m_number = strict_real_p [ push_double(self.m_stack) ] | longlong_p [ push_int (self.m_stack) ] ; // 7: a boolean can be "true" or "false"... m_boolean = ( str_p("true") | str_p("false") ) [ push_boolean(self.m_stack) ] ; // 8: finally, a value also can be a 'null', i.e. an empty item... m_null = str_p("null") [ push_null(self.m_stack) ] ; } }; }; // ========================================================================================================== // === T H E F I N A L P A R S I N G R O U T I N E === // ========================================================================================================== template <typename Iterator> typename json::grammar< typename Iterator::value_type >::variant parse(Iterator const & szFirst, Iterator const & szEnd) { // 1: parse the input... json::grammar< typename Iterator::value_type >::stack st; json::grammar< typename Iterator::value_type > gr(st); boost::spirit::parse_info<Iterator> pi = boost::spirit::parse(szFirst, szEnd, gr, boost::spirit::space_p); // 2: skip any spaces at the end of the parsed section... while((pi.stop != szEnd) && (*pi.stop == static_cast< typename Iterator::value_type >(' '))) { ++pi.stop; } // 3: if the input's end wasn't reached or if there is more than one object on the stack => cancel... if((pi.stop != szEnd) || (st.size() != 1)) { return json::grammar< typename Iterator::value_type >::variant(new boost::any()); } // 4: otherwise, return the result... return st.top(); } }; #endif // TINYJSON_HPP