| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192 | 
							- /// Json-cpp amalgated source (http://jsoncpp.sourceforge.net/).
 
- /// It is intended to be used with #include "json/json.h"
 
- // //////////////////////////////////////////////////////////////////////
 
- // Beginning of content of file: LICENSE
 
- // //////////////////////////////////////////////////////////////////////
 
- /*
 
- The JsonCpp library's source code, including accompanying documentation, 
 
- tests and demonstration applications, are licensed under the following
 
- conditions...
 
- The author (Baptiste Lepilleur) explicitly disclaims copyright in all 
 
- jurisdictions which recognize such a disclaimer. In such jurisdictions, 
 
- this software is released into the Public Domain.
 
- In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
 
- 2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur, and is
 
- released under the terms of the MIT License (see below).
 
- In jurisdictions which recognize Public Domain property, the user of this 
 
- software may choose to accept it either as 1) Public Domain, 2) under the 
 
- conditions of the MIT License (see below), or 3) under the terms of dual 
 
- Public Domain/MIT License conditions described here, as they choose.
 
- The MIT License is about as close to Public Domain as a license can get, and is
 
- described in clear, concise terms at:
 
-    http://en.wikipedia.org/wiki/MIT_License
 
-    
 
- The full text of the MIT License follows:
 
- ========================================================================
 
- Copyright (c) 2007-2010 Baptiste Lepilleur
 
- Permission is hereby granted, free of charge, to any person
 
- obtaining a copy of this software and associated documentation
 
- files (the "Software"), to deal in the Software without
 
- restriction, including without limitation the rights to use, copy,
 
- modify, merge, publish, distribute, sublicense, and/or sell copies
 
- of the Software, and to permit persons to whom the Software is
 
- furnished to do so, subject to the following conditions:
 
- The above copyright notice and this permission notice shall be
 
- included in all copies or substantial portions of the Software.
 
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 
- EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 
- MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 
- NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
 
- BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
 
- ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 
- CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 
- SOFTWARE.
 
- ========================================================================
 
- (END LICENSE TEXT)
 
- The MIT license is compatible with both the GPL and commercial
 
- software, affording one all of the rights of Public Domain with the
 
- minor nuisance of being required to keep the above copyright notice
 
- and license text in the source code. Note also that by accepting the
 
- Public Domain "license" you can re-license your copy using whatever
 
- license you like.
 
- */
 
- // //////////////////////////////////////////////////////////////////////
 
- // End of content of file: LICENSE
 
- // //////////////////////////////////////////////////////////////////////
 
- #include "third_party/jsoncpp/json.h"
 
- #ifndef JSON_IS_AMALGAMATION
 
- #error "Compile with -I PATH_TO_JSON_DIRECTORY"
 
- #endif
 
- // //////////////////////////////////////////////////////////////////////
 
- // Beginning of content of file: src/lib_json/json_tool.h
 
- // //////////////////////////////////////////////////////////////////////
 
- // Copyright 2007-2010 Baptiste Lepilleur
 
- // Distributed under MIT license, or public domain if desired and
 
- // recognized in your jurisdiction.
 
- // See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
 
- #ifndef LIB_JSONCPP_JSON_TOOL_H_INCLUDED
 
- #define LIB_JSONCPP_JSON_TOOL_H_INCLUDED
 
- /* This header provides common string manipulation support, such as UTF-8,
 
-  * portable conversion from/to string...
 
-  *
 
-  * It is an internal header that must not be exposed.
 
-  */
 
- namespace Json {
 
- /// Converts a unicode code-point to UTF-8.
 
- static inline 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] = static_cast<char>(0x80 | (0x3f & (cp >> 6)));
 
-     result[0] = static_cast<char>(0xE0 | (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;
 
- }
 
- /// Returns true if ch is a control character (in range [1,31]).
 
- static inline bool isControlCharacter(char ch) { return ch > 0 && ch <= 0x1F; }
 
- enum {
 
-   /// Constant that specify the size of the buffer that must be passed to
 
-   /// uintToString.
 
-   uintToStringBufferSize = 3 * sizeof(LargestUInt) + 1
 
- };
 
- // Defines a char buffer for use with uintToString().
 
- typedef char UIntToStringBuffer[uintToStringBufferSize];
 
- /** Converts an unsigned integer to string.
 
-  * @param value Unsigned interger to convert to string
 
-  * @param current Input/Output string buffer.
 
-  *        Must have at least uintToStringBufferSize chars free.
 
-  */
 
- static inline void uintToString(LargestUInt value, char*& current) {
 
-   *--current = 0;
 
-   do {
 
-     *--current = static_cast<signed char>(value % 10U + static_cast<unsigned>('0'));
 
-     value /= 10;
 
-   } while (value != 0);
 
- }
 
- /** Change ',' to '.' everywhere in buffer.
 
-  *
 
-  * We had a sophisticated way, but it did not work in WinCE.
 
-  * @see https://github.com/open-source-parsers/jsoncpp/pull/9
 
-  */
 
- static inline void fixNumericLocale(char* begin, char* end) {
 
-   while (begin < end) {
 
-     if (*begin == ',') {
 
-       *begin = '.';
 
-     }
 
-     ++begin;
 
-   }
 
- }
 
- } // namespace Json {
 
- #endif // LIB_JSONCPP_JSON_TOOL_H_INCLUDED
 
- // //////////////////////////////////////////////////////////////////////
 
- // End of content of file: src/lib_json/json_tool.h
 
- // //////////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////////
 
- // Beginning of content of file: src/lib_json/json_reader.cpp
 
- // //////////////////////////////////////////////////////////////////////
 
- // Copyright 2007-2011 Baptiste Lepilleur
 
- // Distributed under MIT license, or public domain if desired and
 
- // recognized in your jurisdiction.
 
- // See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
 
- #if !defined(JSON_IS_AMALGAMATION)
 
- #include <json/assertions.h>
 
- #include <json/reader.h>
 
- #include <json/value.h>
 
- #include "json_tool.h"
 
- #endif // if !defined(JSON_IS_AMALGAMATION)
 
- #include <utility>
 
- #include <cstdio>
 
- #include <cassert>
 
- #include <cstring>
 
- #include <istream>
 
- #include <sstream>
 
- #include <memory>
 
- #include <set>
 
- #include <limits>
 
- #if defined(_MSC_VER)
 
- #if !defined(WINCE) && defined(__STDC_SECURE_LIB__) && _MSC_VER >= 1500 // VC++ 9.0 and above 
 
- #define snprintf sprintf_s
 
- #elif _MSC_VER >= 1900 // VC++ 14.0 and above
 
- #define snprintf std::snprintf
 
- #else
 
- #define snprintf _snprintf
 
- #endif
 
- #elif defined(__ANDROID__) || defined(__QNXNTO__)
 
- #define snprintf snprintf
 
- #elif __cplusplus >= 201103L
 
- #define snprintf std::snprintf
 
- #endif
 
- #if defined(__QNXNTO__)
 
- #define sscanf std::sscanf
 
- #endif
 
- #if defined(_MSC_VER) && _MSC_VER >= 1400 // VC++ 8.0
 
- // Disable warning about strdup being deprecated.
 
- #pragma warning(disable : 4996)
 
- #endif
 
- static int const stackLimit_g = 1000;
 
- static int       stackDepth_g = 0;  // see readValue()
 
- namespace Json {
 
- #if __cplusplus >= 201103L || (defined(_CPPLIB_VER) && _CPPLIB_VER >= 520)
 
- typedef std::unique_ptr<CharReader> CharReaderPtr;
 
- #else
 
- typedef std::auto_ptr<CharReader>   CharReaderPtr;
 
- #endif
 
- // Implementation of class Features
 
- // ////////////////////////////////
 
- Features::Features()
 
-     : allowComments_(true), strictRoot_(false),
 
-       allowDroppedNullPlaceholders_(false), allowNumericKeys_(false) {}
 
- Features Features::all() { return Features(); }
 
- Features Features::strictMode() {
 
-   Features features;
 
-   features.allowComments_ = false;
 
-   features.strictRoot_ = true;
 
-   features.allowDroppedNullPlaceholders_ = false;
 
-   features.allowNumericKeys_ = false;
 
-   return features;
 
- }
 
- // Implementation of class Reader
 
- // ////////////////////////////////
 
- static bool containsNewLine(Reader::Location begin, Reader::Location end) {
 
-   for (; begin < end; ++begin)
 
-     if (*begin == '\n' || *begin == '\r')
 
-       return true;
 
-   return false;
 
- }
 
- // Class Reader
 
- // //////////////////////////////////////////////////////////////////
 
- Reader::Reader()
 
-     : errors_(), document_(), begin_(), end_(), current_(), lastValueEnd_(),
 
-       lastValue_(), commentsBefore_(), features_(Features::all()),
 
-       collectComments_() {}
 
- Reader::Reader(const Features& features)
 
-     : errors_(), document_(), begin_(), end_(), current_(), lastValueEnd_(),
 
-       lastValue_(), commentsBefore_(), features_(features), collectComments_() {
 
- }
 
- 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);
 
-   stackDepth_g = 0;  // Yes, this is bad coding, but options are limited.
 
-   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() {
 
-   // This is a non-reentrant way to support a stackLimit. Terrible!
 
-   // But this deprecated class has a security problem: Bad input can
 
-   // cause a seg-fault. This seems like a fair, binary-compatible way
 
-   // to prevent the problem.
 
-   if (stackDepth_g >= stackLimit_g) throwRuntimeError("Exceeded stackLimit in readValue().");
 
-   ++stackDepth_g;
 
-   Token token;
 
-   skipCommentTokens(token);
 
-   bool successful = true;
 
-   if (collectComments_ && !commentsBefore_.empty()) {
 
-     currentValue().setComment(commentsBefore_, commentBefore);
 
-     commentsBefore_ = "";
 
-   }
 
-   switch (token.type_) {
 
-   case tokenObjectBegin:
 
-     successful = readObject(token);
 
-     currentValue().setOffsetLimit(current_ - begin_);
 
-     break;
 
-   case tokenArrayBegin:
 
-     successful = readArray(token);
 
-     currentValue().setOffsetLimit(current_ - begin_);
 
-     break;
 
-   case tokenNumber:
 
-     successful = decodeNumber(token);
 
-     break;
 
-   case tokenString:
 
-     successful = decodeString(token);
 
-     break;
 
-   case tokenTrue:
 
-     {
 
-     Value v(true);
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenFalse:
 
-     {
 
-     Value v(false);
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenNull:
 
-     {
 
-     Value v;
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenArraySeparator:
 
-   case tokenObjectEnd:
 
-   case tokenArrayEnd:
 
-     if (features_.allowDroppedNullPlaceholders_) {
 
-       // "Un-read" the current token and mark the current value as a null
 
-       // token.
 
-       current_--;
 
-       Value v;
 
-       currentValue().swapPayload(v);
 
-       currentValue().setOffsetStart(current_ - begin_ - 1);
 
-       currentValue().setOffsetLimit(current_ - begin_);
 
-       break;
 
-     } // Else, fall through...
 
-   default:
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     return addError("Syntax error: value, object or array expected.", token);
 
-   }
 
-   if (collectComments_) {
 
-     lastValueEnd_ = current_;
 
-     lastValue_ = ¤tValue();
 
-   }
 
-   --stackDepth_g;
 
-   return successful;
 
- }
 
- void Reader::skipCommentTokens(Token& token) {
 
-   if (features_.allowComments_) {
 
-     do {
 
-       readToken(token);
 
-     } while (token.type_ == tokenComment);
 
-   } else {
 
-     readToken(token);
 
-   }
 
- }
 
- 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;
 
- }
 
- static std::string normalizeEOL(Reader::Location begin, Reader::Location end) {
 
-   std::string normalized;
 
-   normalized.reserve(end - begin);
 
-   Reader::Location current = begin;
 
-   while (current != end) {
 
-     char c = *current++;
 
-     if (c == '\r') {
 
-       if (current != end && *current == '\n')
 
-          // convert dos EOL
 
-          ++current;
 
-       // convert Mac EOL
 
-       normalized += '\n';
 
-     } else {
 
-       normalized += c;
 
-     }
 
-   }
 
-   return normalized;
 
- }
 
- void
 
- Reader::addComment(Location begin, Location end, CommentPlacement placement) {
 
-   assert(collectComments_);
 
-   const std::string& normalized = normalizeEOL(begin, end);
 
-   if (placement == commentAfterOnSameLine) {
 
-     assert(lastValue_ != 0);
 
-     lastValue_->setComment(normalized, placement);
 
-   } else {
 
-     commentsBefore_ += normalized;
 
-   }
 
- }
 
- 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 == '\n')
 
-       break;
 
-     if (c == '\r') {
 
-       // Consume DOS EOL. It will be normalized in addComment.
 
-       if (current_ != end_ && *current_ == '\n')
 
-         getNextChar();
 
-       // Break on Moc OS 9 EOL.
 
-       break;
 
-     }
 
-   }
 
-   return true;
 
- }
 
- void Reader::readNumber() {
 
-   const char *p = current_;
 
-   char c = '0'; // stopgap for already consumed character
 
-   // integral part
 
-   while (c >= '0' && c <= '9')
 
-     c = (current_ = p) < end_ ? *p++ : 0;
 
-   // fractional part
 
-   if (c == '.') {
 
-     c = (current_ = p) < end_ ? *p++ : 0;
 
-     while (c >= '0' && c <= '9')
 
-       c = (current_ = p) < end_ ? *p++ : 0;
 
-   }
 
-   // exponential part
 
-   if (c == 'e' || c == 'E') {
 
-     c = (current_ = p) < end_ ? *p++ : 0;
 
-     if (c == '+' || c == '-')
 
-       c = (current_ = p) < end_ ? *p++ : 0;
 
-     while (c >= '0' && c <= '9')
 
-       c = (current_ = p) < end_ ? *p++ : 0;
 
-   }
 
- }
 
- 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;
 
-   Value init(objectValue);
 
-   currentValue().swapPayload(init);
 
-   currentValue().setOffsetStart(tokenStart.start_ - begin_);
 
-   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;
 
-     name = "";
 
-     if (tokenName.type_ == tokenString) {
 
-       if (!decodeString(tokenName, name))
 
-         return recoverFromError(tokenObjectEnd);
 
-     } else if (tokenName.type_ == tokenNumber && features_.allowNumericKeys_) {
 
-       Value numberName;
 
-       if (!decodeNumber(tokenName, numberName))
 
-         return recoverFromError(tokenObjectEnd);
 
-       name = numberName.asString();
 
-     } else {
 
-       break;
 
-     }
 
-     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) {
 
-   Value init(arrayValue);
 
-   currentValue().swapPayload(init);
 
-   currentValue().setOffsetStart(tokenStart.start_ - begin_);
 
-   skipSpaces();
 
-   if (*current_ == ']') // empty array
 
-   {
 
-     Token endArray;
 
-     readToken(endArray);
 
-     return true;
 
-   }
 
-   int index = 0;
 
-   for (;;) {
 
-     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) {
 
-   Value decoded;
 
-   if (!decodeNumber(token, decoded))
 
-     return false;
 
-   currentValue().swapPayload(decoded);
 
-   currentValue().setOffsetStart(token.start_ - begin_);
 
-   currentValue().setOffsetLimit(token.end_ - begin_);
 
-   return true;
 
- }
 
- bool Reader::decodeNumber(Token& token, Value& decoded) {
 
-   // Attempts to parse the number as an integer. If the number is
 
-   // larger than the maximum supported value of an integer then
 
-   // we decode the number as a double.
 
-   Location current = token.start_;
 
-   bool isNegative = *current == '-';
 
-   if (isNegative)
 
-     ++current;
 
-   // TODO: Help the compiler do the div and mod at compile time or get rid of them.
 
-   Value::LargestUInt maxIntegerValue =
 
-       isNegative ? Value::LargestUInt(Value::maxLargestInt) + 1
 
-                  : Value::maxLargestUInt;
 
-   Value::LargestUInt threshold = maxIntegerValue / 10;
 
-   Value::LargestUInt value = 0;
 
-   while (current < token.end_) {
 
-     Char c = *current++;
 
-     if (c < '0' || c > '9')
 
-       return decodeDouble(token, decoded);
 
-     Value::UInt digit(c - '0');
 
-     if (value >= threshold) {
 
-       // We've hit or exceeded the max value divided by 10 (rounded down). If
 
-       // a) we've only just touched the limit, b) this is the last digit, and
 
-       // c) it's small enough to fit in that rounding delta, we're okay.
 
-       // Otherwise treat this number as a double to avoid overflow.
 
-       if (value > threshold || current != token.end_ ||
 
-           digit > maxIntegerValue % 10) {
 
-         return decodeDouble(token, decoded);
 
-       }
 
-     }
 
-     value = value * 10 + digit;
 
-   }
 
-   if (isNegative && value == maxIntegerValue)
 
-     decoded = Value::minLargestInt;
 
-   else if (isNegative)
 
-     decoded = -Value::LargestInt(value);
 
-   else if (value <= Value::LargestUInt(Value::maxInt))
 
-     decoded = Value::LargestInt(value);
 
-   else
 
-     decoded = value;
 
-   return true;
 
- }
 
- bool Reader::decodeDouble(Token& token) {
 
-   Value decoded;
 
-   if (!decodeDouble(token, decoded))
 
-     return false;
 
-   currentValue().swapPayload(decoded);
 
-   currentValue().setOffsetStart(token.start_ - begin_);
 
-   currentValue().setOffsetLimit(token.end_ - begin_);
 
-   return true;
 
- }
 
- bool Reader::decodeDouble(Token& token, Value& decoded) {
 
-   double value = 0;
 
-   std::string buffer(token.start_, token.end_);
 
-   std::istringstream is(buffer);
 
-   if (!(is >> value))
 
-     return addError("'" + std::string(token.start_, token.end_) +
 
-                         "' is not a number.",
 
-                     token);
 
-   decoded = value;
 
-   return true;
 
- }
 
- bool Reader::decodeString(Token& token) {
 
-   std::string decoded_string;
 
-   if (!decodeString(token, decoded_string))
 
-     return false;
 
-   Value decoded(decoded_string);
 
-   currentValue().swapPayload(decoded);
 
-   currentValue().setOffsetStart(token.start_ - begin_);
 
-   currentValue().setOffsetLimit(token.end_ - begin_);
 
-   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& current,
 
-                                     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& current,
 
-                                          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;
 
-   for (;;) {
 
-     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];
 
-   snprintf(buffer, sizeof(buffer), "Line %d, Column %d", line, column);
 
-   return buffer;
 
- }
 
- // Deprecated. Preserved for backward compatibility
 
- std::string Reader::getFormatedErrorMessages() const {
 
-   return getFormattedErrorMessages();
 
- }
 
- std::string Reader::getFormattedErrorMessages() 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::vector<Reader::StructuredError> Reader::getStructuredErrors() const {
 
-   std::vector<Reader::StructuredError> allErrors;
 
-   for (Errors::const_iterator itError = errors_.begin();
 
-        itError != errors_.end();
 
-        ++itError) {
 
-     const ErrorInfo& error = *itError;
 
-     Reader::StructuredError structured;
 
-     structured.offset_start = error.token_.start_ - begin_;
 
-     structured.offset_limit = error.token_.end_ - begin_;
 
-     structured.message = error.message_;
 
-     allErrors.push_back(structured);
 
-   }
 
-   return allErrors;
 
- }
 
- bool Reader::pushError(const Value& value, const std::string& message) {
 
-   size_t length = end_ - begin_;
 
-   if(value.getOffsetStart() > length
 
-     || value.getOffsetLimit() > length)
 
-     return false;
 
-   Token token;
 
-   token.type_ = tokenError;
 
-   token.start_ = begin_ + value.getOffsetStart();
 
-   token.end_ = end_ + value.getOffsetLimit();
 
-   ErrorInfo info;
 
-   info.token_ = token;
 
-   info.message_ = message;
 
-   info.extra_ = 0;
 
-   errors_.push_back(info);
 
-   return true;
 
- }
 
- bool Reader::pushError(const Value& value, const std::string& message, const Value& extra) {
 
-   size_t length = end_ - begin_;
 
-   if(value.getOffsetStart() > length
 
-     || value.getOffsetLimit() > length
 
-     || extra.getOffsetLimit() > length)
 
-     return false;
 
-   Token token;
 
-   token.type_ = tokenError;
 
-   token.start_ = begin_ + value.getOffsetStart();
 
-   token.end_ = begin_ + value.getOffsetLimit();
 
-   ErrorInfo info;
 
-   info.token_ = token;
 
-   info.message_ = message;
 
-   info.extra_ = begin_ + extra.getOffsetStart();
 
-   errors_.push_back(info);
 
-   return true;
 
- }
 
- bool Reader::good() const {
 
-   return !errors_.size();
 
- }
 
- // exact copy of Features
 
- class OurFeatures {
 
- public:
 
-   static OurFeatures all();
 
-   bool allowComments_;
 
-   bool strictRoot_;
 
-   bool allowDroppedNullPlaceholders_;
 
-   bool allowNumericKeys_;
 
-   bool allowSingleQuotes_;
 
-   bool failIfExtra_;
 
-   bool rejectDupKeys_;
 
-   bool allowSpecialFloats_;
 
-   int stackLimit_;
 
- };  // OurFeatures
 
- // exact copy of Implementation of class Features
 
- // ////////////////////////////////
 
- OurFeatures OurFeatures::all() { return OurFeatures(); }
 
- // Implementation of class Reader
 
- // ////////////////////////////////
 
- // exact copy of Reader, renamed to OurReader
 
- class OurReader {
 
- public:
 
-   typedef char Char;
 
-   typedef const Char* Location;
 
-   struct StructuredError {
 
-     size_t offset_start;
 
-     size_t offset_limit;
 
-     std::string message;
 
-   };
 
-   OurReader(OurFeatures const& features);
 
-   bool parse(const char* beginDoc,
 
-              const char* endDoc,
 
-              Value& root,
 
-              bool collectComments = true);
 
-   std::string getFormattedErrorMessages() const;
 
-   std::vector<StructuredError> getStructuredErrors() const;
 
-   bool pushError(const Value& value, const std::string& message);
 
-   bool pushError(const Value& value, const std::string& message, const Value& extra);
 
-   bool good() const;
 
- private:
 
-   OurReader(OurReader const&);  // no impl
 
-   void operator=(OurReader const&);  // no impl
 
-   enum TokenType {
 
-     tokenEndOfStream = 0,
 
-     tokenObjectBegin,
 
-     tokenObjectEnd,
 
-     tokenArrayBegin,
 
-     tokenArrayEnd,
 
-     tokenString,
 
-     tokenNumber,
 
-     tokenTrue,
 
-     tokenFalse,
 
-     tokenNull,
 
-     tokenNaN,
 
-     tokenPosInf,
 
-     tokenNegInf,
 
-     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 readToken(Token& token);
 
-   void skipSpaces();
 
-   bool match(Location pattern, int patternLength);
 
-   bool readComment();
 
-   bool readCStyleComment();
 
-   bool readCppStyleComment();
 
-   bool readString();
 
-   bool readStringSingleQuote();
 
-   bool readNumber(bool checkInf);
 
-   bool readValue();
 
-   bool readObject(Token& token);
 
-   bool readArray(Token& token);
 
-   bool decodeNumber(Token& token);
 
-   bool decodeNumber(Token& token, Value& decoded);
 
-   bool decodeString(Token& token);
 
-   bool decodeString(Token& token, std::string& decoded);
 
-   bool decodeDouble(Token& token);
 
-   bool decodeDouble(Token& token, Value& decoded);
 
-   bool decodeUnicodeCodePoint(Token& token,
 
-                               Location& current,
 
-                               Location end,
 
-                               unsigned int& unicode);
 
-   bool decodeUnicodeEscapeSequence(Token& token,
 
-                                    Location& current,
 
-                                    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& currentValue();
 
-   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_;
 
-   int stackDepth_;
 
-   OurFeatures const features_;
 
-   bool collectComments_;
 
- };  // OurReader
 
- // complete copy of Read impl, for OurReader
 
- OurReader::OurReader(OurFeatures const& features)
 
-     : errors_(), document_(), begin_(), end_(), current_(), lastValueEnd_(),
 
-       lastValue_(), commentsBefore_(),
 
-       stackDepth_(0),
 
-       features_(features), collectComments_() {
 
- }
 
- bool OurReader::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);
 
-   stackDepth_ = 0;
 
-   bool successful = readValue();
 
-   Token token;
 
-   skipCommentTokens(token);
 
-   if (features_.failIfExtra_) {
 
-     if (token.type_ != tokenError && token.type_ != tokenEndOfStream) {
 
-       addError("Extra non-whitespace after JSON value.", token);
 
-       return false;
 
-     }
 
-   }
 
-   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 OurReader::readValue() {
 
-   if (stackDepth_ >= features_.stackLimit_) throwRuntimeError("Exceeded stackLimit in readValue().");
 
-   ++stackDepth_;
 
-   Token token;
 
-   skipCommentTokens(token);
 
-   bool successful = true;
 
-   if (collectComments_ && !commentsBefore_.empty()) {
 
-     currentValue().setComment(commentsBefore_, commentBefore);
 
-     commentsBefore_ = "";
 
-   }
 
-   switch (token.type_) {
 
-   case tokenObjectBegin:
 
-     successful = readObject(token);
 
-     currentValue().setOffsetLimit(current_ - begin_);
 
-     break;
 
-   case tokenArrayBegin:
 
-     successful = readArray(token);
 
-     currentValue().setOffsetLimit(current_ - begin_);
 
-     break;
 
-   case tokenNumber:
 
-     successful = decodeNumber(token);
 
-     break;
 
-   case tokenString:
 
-     successful = decodeString(token);
 
-     break;
 
-   case tokenTrue:
 
-     {
 
-     Value v(true);
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenFalse:
 
-     {
 
-     Value v(false);
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenNull:
 
-     {
 
-     Value v;
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenNaN:
 
-     {
 
-     Value v(std::numeric_limits<double>::quiet_NaN());
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenPosInf:
 
-     {
 
-     Value v(std::numeric_limits<double>::infinity());
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenNegInf:
 
-     {
 
-     Value v(-std::numeric_limits<double>::infinity());
 
-     currentValue().swapPayload(v);
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     }
 
-     break;
 
-   case tokenArraySeparator:
 
-   case tokenObjectEnd:
 
-   case tokenArrayEnd:
 
-     if (features_.allowDroppedNullPlaceholders_) {
 
-       // "Un-read" the current token and mark the current value as a null
 
-       // token.
 
-       current_--;
 
-       Value v;
 
-       currentValue().swapPayload(v);
 
-       currentValue().setOffsetStart(current_ - begin_ - 1);
 
-       currentValue().setOffsetLimit(current_ - begin_);
 
-       break;
 
-     } // else, fall through ...
 
-   default:
 
-     currentValue().setOffsetStart(token.start_ - begin_);
 
-     currentValue().setOffsetLimit(token.end_ - begin_);
 
-     return addError("Syntax error: value, object or array expected.", token);
 
-   }
 
-   if (collectComments_) {
 
-     lastValueEnd_ = current_;
 
-     lastValue_ = ¤tValue();
 
-   }
 
-   --stackDepth_;
 
-   return successful;
 
- }
 
- void OurReader::skipCommentTokens(Token& token) {
 
-   if (features_.allowComments_) {
 
-     do {
 
-       readToken(token);
 
-     } while (token.type_ == tokenComment);
 
-   } else {
 
-     readToken(token);
 
-   }
 
- }
 
- bool OurReader::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 '\'':
 
-     if (features_.allowSingleQuotes_) {
 
-     token.type_ = tokenString;
 
-     ok = readStringSingleQuote();
 
-     break;
 
-     } // else continue
 
-   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':
 
-     token.type_ = tokenNumber;
 
-     readNumber(false);
 
-     break;
 
-   case '-':
 
-     if (readNumber(true)) {
 
-       token.type_ = tokenNumber;
 
-     } else {
 
-       token.type_ = tokenNegInf;
 
-       ok = features_.allowSpecialFloats_ && match("nfinity", 7);
 
-     }
 
-     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 'N':
 
-     if (features_.allowSpecialFloats_) {
 
-       token.type_ = tokenNaN;
 
-       ok = match("aN", 2);
 
-     } else {
 
-       ok = false;
 
-     }
 
-     break;
 
-   case 'I':
 
-     if (features_.allowSpecialFloats_) {
 
-       token.type_ = tokenPosInf;
 
-       ok = match("nfinity", 7);
 
-     } else {
 
-       ok = false;
 
-     }
 
-     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 OurReader::skipSpaces() {
 
-   while (current_ != end_) {
 
-     Char c = *current_;
 
-     if (c == ' ' || c == '\t' || c == '\r' || c == '\n')
 
-       ++current_;
 
-     else
 
-       break;
 
-   }
 
- }
 
- bool OurReader::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 OurReader::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
 
- OurReader::addComment(Location begin, Location end, CommentPlacement placement) {
 
-   assert(collectComments_);
 
-   const std::string& normalized = normalizeEOL(begin, end);
 
-   if (placement == commentAfterOnSameLine) {
 
-     assert(lastValue_ != 0);
 
-     lastValue_->setComment(normalized, placement);
 
-   } else {
 
-     commentsBefore_ += normalized;
 
-   }
 
- }
 
- bool OurReader::readCStyleComment() {
 
-   while (current_ != end_) {
 
-     Char c = getNextChar();
 
-     if (c == '*' && *current_ == '/')
 
-       break;
 
-   }
 
-   return getNextChar() == '/';
 
- }
 
- bool OurReader::readCppStyleComment() {
 
-   while (current_ != end_) {
 
-     Char c = getNextChar();
 
-     if (c == '\n')
 
-       break;
 
-     if (c == '\r') {
 
-       // Consume DOS EOL. It will be normalized in addComment.
 
-       if (current_ != end_ && *current_ == '\n')
 
-         getNextChar();
 
-       // Break on Moc OS 9 EOL.
 
-       break;
 
-     }
 
-   }
 
-   return true;
 
- }
 
- bool OurReader::readNumber(bool checkInf) {
 
-   const char *p = current_;
 
-   if (checkInf && p != end_ && *p == 'I') {
 
-     current_ = ++p;
 
-     return false;
 
-   }
 
-   char c = '0'; // stopgap for already consumed character
 
-   // integral part
 
-   while (c >= '0' && c <= '9')
 
-     c = (current_ = p) < end_ ? *p++ : 0;
 
-   // fractional part
 
-   if (c == '.') {
 
-     c = (current_ = p) < end_ ? *p++ : 0;
 
-     while (c >= '0' && c <= '9')
 
-       c = (current_ = p) < end_ ? *p++ : 0;
 
-   }
 
-   // exponential part
 
-   if (c == 'e' || c == 'E') {
 
-     c = (current_ = p) < end_ ? *p++ : 0;
 
-     if (c == '+' || c == '-')
 
-       c = (current_ = p) < end_ ? *p++ : 0;
 
-     while (c >= '0' && c <= '9')
 
-       c = (current_ = p) < end_ ? *p++ : 0;
 
-   }
 
-   return true;
 
- }
 
- bool OurReader::readString() {
 
-   Char c = 0;
 
-   while (current_ != end_) {
 
-     c = getNextChar();
 
-     if (c == '\\')
 
-       getNextChar();
 
-     else if (c == '"')
 
-       break;
 
-   }
 
-   return c == '"';
 
- }
 
- bool OurReader::readStringSingleQuote() {
 
-   Char c = 0;
 
-   while (current_ != end_) {
 
-     c = getNextChar();
 
-     if (c == '\\')
 
-       getNextChar();
 
-     else if (c == '\'')
 
-       break;
 
-   }
 
-   return c == '\'';
 
- }
 
- bool OurReader::readObject(Token& tokenStart) {
 
-   Token tokenName;
 
-   std::string name;
 
-   Value init(objectValue);
 
-   currentValue().swapPayload(init);
 
-   currentValue().setOffsetStart(tokenStart.start_ - begin_);
 
-   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;
 
-     name = "";
 
-     if (tokenName.type_ == tokenString) {
 
-       if (!decodeString(tokenName, name))
 
-         return recoverFromError(tokenObjectEnd);
 
-     } else if (tokenName.type_ == tokenNumber && features_.allowNumericKeys_) {
 
-       Value numberName;
 
-       if (!decodeNumber(tokenName, numberName))
 
-         return recoverFromError(tokenObjectEnd);
 
-       name = numberName.asString();
 
-     } else {
 
-       break;
 
-     }
 
-     Token colon;
 
-     if (!readToken(colon) || colon.type_ != tokenMemberSeparator) {
 
-       return addErrorAndRecover(
 
-           "Missing ':' after object member name", colon, tokenObjectEnd);
 
-     }
 
-     if (name.length() >= (1U<<30)) throwRuntimeError("keylength >= 2^30");
 
-     if (features_.rejectDupKeys_ && currentValue().isMember(name)) {
 
-       std::string msg = "Duplicate key: '" + name + "'";
 
-       return addErrorAndRecover(
 
-           msg, tokenName, 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 OurReader::readArray(Token& tokenStart) {
 
-   Value init(arrayValue);
 
-   currentValue().swapPayload(init);
 
-   currentValue().setOffsetStart(tokenStart.start_ - begin_);
 
-   skipSpaces();
 
-   if (*current_ == ']') // empty array
 
-   {
 
-     Token endArray;
 
-     readToken(endArray);
 
-     return true;
 
-   }
 
-   int index = 0;
 
-   for (;;) {
 
-     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 OurReader::decodeNumber(Token& token) {
 
-   Value decoded;
 
-   if (!decodeNumber(token, decoded))
 
-     return false;
 
-   currentValue().swapPayload(decoded);
 
-   currentValue().setOffsetStart(token.start_ - begin_);
 
-   currentValue().setOffsetLimit(token.end_ - begin_);
 
-   return true;
 
- }
 
- bool OurReader::decodeNumber(Token& token, Value& decoded) {
 
-   // Attempts to parse the number as an integer. If the number is
 
-   // larger than the maximum supported value of an integer then
 
-   // we decode the number as a double.
 
-   Location current = token.start_;
 
-   bool isNegative = *current == '-';
 
-   if (isNegative)
 
-     ++current;
 
-   // TODO: Help the compiler do the div and mod at compile time or get rid of them.
 
-   Value::LargestUInt maxIntegerValue =
 
-       isNegative ? Value::LargestUInt(-Value::minLargestInt)
 
-                  : Value::maxLargestUInt;
 
-   Value::LargestUInt threshold = maxIntegerValue / 10;
 
-   Value::LargestUInt value = 0;
 
-   while (current < token.end_) {
 
-     Char c = *current++;
 
-     if (c < '0' || c > '9')
 
-       return decodeDouble(token, decoded);
 
-     Value::UInt digit(c - '0');
 
-     if (value >= threshold) {
 
-       // We've hit or exceeded the max value divided by 10 (rounded down). If
 
-       // a) we've only just touched the limit, b) this is the last digit, and
 
-       // c) it's small enough to fit in that rounding delta, we're okay.
 
-       // Otherwise treat this number as a double to avoid overflow.
 
-       if (value > threshold || current != token.end_ ||
 
-           digit > maxIntegerValue % 10) {
 
-         return decodeDouble(token, decoded);
 
-       }
 
-     }
 
-     value = value * 10 + digit;
 
-   }
 
-   if (isNegative)
 
-     decoded = -Value::LargestInt(value);
 
-   else if (value <= Value::LargestUInt(Value::maxInt))
 
-     decoded = Value::LargestInt(value);
 
-   else
 
-     decoded = value;
 
-   return true;
 
- }
 
- bool OurReader::decodeDouble(Token& token) {
 
-   Value decoded;
 
-   if (!decodeDouble(token, decoded))
 
-     return false;
 
-   currentValue().swapPayload(decoded);
 
-   currentValue().setOffsetStart(token.start_ - begin_);
 
-   currentValue().setOffsetLimit(token.end_ - begin_);
 
-   return true;
 
- }
 
- bool OurReader::decodeDouble(Token& token, Value& decoded) {
 
-   double value = 0;
 
-   const int bufferSize = 32;
 
-   int count;
 
-   int length = int(token.end_ - token.start_);
 
-   // Sanity check to avoid buffer overflow exploits.
 
-   if (length < 0) {
 
-     return addError("Unable to parse token length", token);
 
-   }
 
-   // Avoid using a string constant for the format control string given to
 
-   // sscanf, as this can cause hard to debug crashes on OS X. See here for more
 
-   // info:
 
-   //
 
-   //     http://developer.apple.com/library/mac/#DOCUMENTATION/DeveloperTools/gcc-4.0.1/gcc/Incompatibilities.html
 
-   char format[] = "%lf";
 
-   if (length <= bufferSize) {
 
-     Char buffer[bufferSize + 1];
 
-     memcpy(buffer, token.start_, length);
 
-     buffer[length] = 0;
 
-     count = sscanf(buffer, format, &value);
 
-   } else {
 
-     std::string buffer(token.start_, token.end_);
 
-     count = sscanf(buffer.c_str(), format, &value);
 
-   }
 
-   if (count != 1)
 
-     return addError("'" + std::string(token.start_, token.end_) +
 
-                         "' is not a number.",
 
-                     token);
 
-   decoded = value;
 
-   return true;
 
- }
 
- bool OurReader::decodeString(Token& token) {
 
-   std::string decoded_string;
 
-   if (!decodeString(token, decoded_string))
 
-     return false;
 
-   Value decoded(decoded_string);
 
-   currentValue().swapPayload(decoded);
 
-   currentValue().setOffsetStart(token.start_ - begin_);
 
-   currentValue().setOffsetLimit(token.end_ - begin_);
 
-   return true;
 
- }
 
- bool OurReader::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 OurReader::decodeUnicodeCodePoint(Token& token,
 
-                                     Location& current,
 
-                                     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 OurReader::decodeUnicodeEscapeSequence(Token& token,
 
-                                          Location& current,
 
-                                          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
 
- OurReader::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 OurReader::recoverFromError(TokenType skipUntilToken) {
 
-   int errorCount = int(errors_.size());
 
-   Token skip;
 
-   for (;;) {
 
-     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 OurReader::addErrorAndRecover(const std::string& message,
 
-                                 Token& token,
 
-                                 TokenType skipUntilToken) {
 
-   addError(message, token);
 
-   return recoverFromError(skipUntilToken);
 
- }
 
- Value& OurReader::currentValue() { return *(nodes_.top()); }
 
- OurReader::Char OurReader::getNextChar() {
 
-   if (current_ == end_)
 
-     return 0;
 
-   return *current_++;
 
- }
 
- void OurReader::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 OurReader::getLocationLineAndColumn(Location location) const {
 
-   int line, column;
 
-   getLocationLineAndColumn(location, line, column);
 
-   char buffer[18 + 16 + 16 + 1];
 
-   snprintf(buffer, sizeof(buffer), "Line %d, Column %d", line, column);
 
-   return buffer;
 
- }
 
- std::string OurReader::getFormattedErrorMessages() 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::vector<OurReader::StructuredError> OurReader::getStructuredErrors() const {
 
-   std::vector<OurReader::StructuredError> allErrors;
 
-   for (Errors::const_iterator itError = errors_.begin();
 
-        itError != errors_.end();
 
-        ++itError) {
 
-     const ErrorInfo& error = *itError;
 
-     OurReader::StructuredError structured;
 
-     structured.offset_start = error.token_.start_ - begin_;
 
-     structured.offset_limit = error.token_.end_ - begin_;
 
-     structured.message = error.message_;
 
-     allErrors.push_back(structured);
 
-   }
 
-   return allErrors;
 
- }
 
- bool OurReader::pushError(const Value& value, const std::string& message) {
 
-   size_t length = end_ - begin_;
 
-   if(value.getOffsetStart() > length
 
-     || value.getOffsetLimit() > length)
 
-     return false;
 
-   Token token;
 
-   token.type_ = tokenError;
 
-   token.start_ = begin_ + value.getOffsetStart();
 
-   token.end_ = end_ + value.getOffsetLimit();
 
-   ErrorInfo info;
 
-   info.token_ = token;
 
-   info.message_ = message;
 
-   info.extra_ = 0;
 
-   errors_.push_back(info);
 
-   return true;
 
- }
 
- bool OurReader::pushError(const Value& value, const std::string& message, const Value& extra) {
 
-   size_t length = end_ - begin_;
 
-   if(value.getOffsetStart() > length
 
-     || value.getOffsetLimit() > length
 
-     || extra.getOffsetLimit() > length)
 
-     return false;
 
-   Token token;
 
-   token.type_ = tokenError;
 
-   token.start_ = begin_ + value.getOffsetStart();
 
-   token.end_ = begin_ + value.getOffsetLimit();
 
-   ErrorInfo info;
 
-   info.token_ = token;
 
-   info.message_ = message;
 
-   info.extra_ = begin_ + extra.getOffsetStart();
 
-   errors_.push_back(info);
 
-   return true;
 
- }
 
- bool OurReader::good() const {
 
-   return !errors_.size();
 
- }
 
- class OurCharReader : public CharReader {
 
-   bool const collectComments_;
 
-   OurReader reader_;
 
- public:
 
-   OurCharReader(
 
-     bool collectComments,
 
-     OurFeatures const& features)
 
-   : collectComments_(collectComments)
 
-   , reader_(features)
 
-   {}
 
-   bool parse(
 
-       char const* beginDoc, char const* endDoc,
 
-       Value* root, std::string* errs) override {
 
-     bool ok = reader_.parse(beginDoc, endDoc, *root, collectComments_);
 
-     if (errs) {
 
-       *errs = reader_.getFormattedErrorMessages();
 
-     }
 
-     return ok;
 
-   }
 
- };
 
- CharReaderBuilder::CharReaderBuilder()
 
- {
 
-   setDefaults(&settings_);
 
- }
 
- CharReaderBuilder::~CharReaderBuilder()
 
- {}
 
- CharReader* CharReaderBuilder::newCharReader() const
 
- {
 
-   bool collectComments = settings_["collectComments"].asBool();
 
-   OurFeatures features = OurFeatures::all();
 
-   features.allowComments_ = settings_["allowComments"].asBool();
 
-   features.strictRoot_ = settings_["strictRoot"].asBool();
 
-   features.allowDroppedNullPlaceholders_ = settings_["allowDroppedNullPlaceholders"].asBool();
 
-   features.allowNumericKeys_ = settings_["allowNumericKeys"].asBool();
 
-   features.allowSingleQuotes_ = settings_["allowSingleQuotes"].asBool();
 
-   features.stackLimit_ = settings_["stackLimit"].asInt();
 
-   features.failIfExtra_ = settings_["failIfExtra"].asBool();
 
-   features.rejectDupKeys_ = settings_["rejectDupKeys"].asBool();
 
-   features.allowSpecialFloats_ = settings_["allowSpecialFloats"].asBool();
 
-   return new OurCharReader(collectComments, features);
 
- }
 
- static void getValidReaderKeys(std::set<std::string>* valid_keys)
 
- {
 
-   valid_keys->clear();
 
-   valid_keys->insert("collectComments");
 
-   valid_keys->insert("allowComments");
 
-   valid_keys->insert("strictRoot");
 
-   valid_keys->insert("allowDroppedNullPlaceholders");
 
-   valid_keys->insert("allowNumericKeys");
 
-   valid_keys->insert("allowSingleQuotes");
 
-   valid_keys->insert("stackLimit");
 
-   valid_keys->insert("failIfExtra");
 
-   valid_keys->insert("rejectDupKeys");
 
-   valid_keys->insert("allowSpecialFloats");
 
- }
 
- bool CharReaderBuilder::validate(Json::Value* invalid) const
 
- {
 
-   Json::Value my_invalid;
 
-   if (!invalid) invalid = &my_invalid;  // so we do not need to test for NULL
 
-   Json::Value& inv = *invalid;
 
-   std::set<std::string> valid_keys;
 
-   getValidReaderKeys(&valid_keys);
 
-   Value::Members keys = settings_.getMemberNames();
 
-   size_t n = keys.size();
 
-   for (size_t i = 0; i < n; ++i) {
 
-     std::string const& key = keys[i];
 
-     if (valid_keys.find(key) == valid_keys.end()) {
 
-       inv[key] = settings_[key];
 
-     }
 
-   }
 
-   return 0u == inv.size();
 
- }
 
- Value& CharReaderBuilder::operator[](std::string key)
 
- {
 
-   return settings_[key];
 
- }
 
- // static
 
- void CharReaderBuilder::strictMode(Json::Value* settings)
 
- {
 
- //! [CharReaderBuilderStrictMode]
 
-   (*settings)["allowComments"] = false;
 
-   (*settings)["strictRoot"] = true;
 
-   (*settings)["allowDroppedNullPlaceholders"] = false;
 
-   (*settings)["allowNumericKeys"] = false;
 
-   (*settings)["allowSingleQuotes"] = false;
 
-   (*settings)["stackLimit"] = 1000;
 
-   (*settings)["failIfExtra"] = true;
 
-   (*settings)["rejectDupKeys"] = true;
 
-   (*settings)["allowSpecialFloats"] = false;
 
- //! [CharReaderBuilderStrictMode]
 
- }
 
- // static
 
- void CharReaderBuilder::setDefaults(Json::Value* settings)
 
- {
 
- //! [CharReaderBuilderDefaults]
 
-   (*settings)["collectComments"] = true;
 
-   (*settings)["allowComments"] = true;
 
-   (*settings)["strictRoot"] = false;
 
-   (*settings)["allowDroppedNullPlaceholders"] = false;
 
-   (*settings)["allowNumericKeys"] = false;
 
-   (*settings)["allowSingleQuotes"] = false;
 
-   (*settings)["stackLimit"] = 1000;
 
-   (*settings)["failIfExtra"] = false;
 
-   (*settings)["rejectDupKeys"] = false;
 
-   (*settings)["allowSpecialFloats"] = false;
 
- //! [CharReaderBuilderDefaults]
 
- }
 
- //////////////////////////////////
 
- // global functions
 
- bool parseFromStream(
 
-     CharReader::Factory const& fact, std::istream& sin,
 
-     Value* root, std::string* errs)
 
- {
 
-   std::ostringstream ssin;
 
-   ssin << sin.rdbuf();
 
-   std::string doc = ssin.str();
 
-   char const* begin = doc.data();
 
-   char const* end = begin + doc.size();
 
-   // Note that we do not actually need a null-terminator.
 
-   CharReaderPtr const reader(fact.newCharReader());
 
-   return reader->parse(begin, end, root, errs);
 
- }
 
- std::istream& operator>>(std::istream& sin, Value& root) {
 
-   CharReaderBuilder b;
 
-   std::string errs;
 
-   bool ok = parseFromStream(b, sin, &root, &errs);
 
-   if (!ok) {
 
-     fprintf(stderr,
 
-             "Error from reader: %s",
 
-             errs.c_str());
 
-     throwRuntimeError(errs);
 
-   }
 
-   return sin;
 
- }
 
- } // namespace Json
 
- // //////////////////////////////////////////////////////////////////////
 
- // End of content of file: src/lib_json/json_reader.cpp
 
- // //////////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////////
 
- // Beginning of content of file: src/lib_json/json_valueiterator.inl
 
- // //////////////////////////////////////////////////////////////////////
 
- // Copyright 2007-2010 Baptiste Lepilleur
 
- // Distributed under MIT license, or public domain if desired and
 
- // recognized in your jurisdiction.
 
- // See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
 
- // included by json_value.cpp
 
- namespace Json {
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // class ValueIteratorBase
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- ValueIteratorBase::ValueIteratorBase()
 
-     : current_(), isNull_(true) {
 
- }
 
- ValueIteratorBase::ValueIteratorBase(
 
-     const Value::ObjectValues::iterator& current)
 
-     : current_(current), isNull_(false) {}
 
- Value& ValueIteratorBase::deref() const {
 
-   return current_->second;
 
- }
 
- void ValueIteratorBase::increment() {
 
-   ++current_;
 
- }
 
- void ValueIteratorBase::decrement() {
 
-   --current_;
 
- }
 
- ValueIteratorBase::difference_type
 
- ValueIteratorBase::computeDistance(const SelfType& other) const {
 
- #ifdef JSON_USE_CPPTL_SMALLMAP
 
-   return other.current_ - 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
 
- }
 
- bool ValueIteratorBase::isEqual(const SelfType& other) const {
 
-   if (isNull_) {
 
-     return other.isNull_;
 
-   }
 
-   return current_ == other.current_;
 
- }
 
- void ValueIteratorBase::copy(const SelfType& other) {
 
-   current_ = other.current_;
 
-   isNull_ = other.isNull_;
 
- }
 
- Value ValueIteratorBase::key() const {
 
-   const Value::CZString czstring = (*current_).first;
 
-   if (czstring.data()) {
 
-     if (czstring.isStaticString())
 
-       return Value(StaticString(czstring.data()));
 
-     return Value(czstring.data(), czstring.data() + czstring.length());
 
-   }
 
-   return Value(czstring.index());
 
- }
 
- UInt ValueIteratorBase::index() const {
 
-   const Value::CZString czstring = (*current_).first;
 
-   if (!czstring.data())
 
-     return czstring.index();
 
-   return Value::UInt(-1);
 
- }
 
- std::string ValueIteratorBase::name() const {
 
-   char const* keey;
 
-   char const* end;
 
-   keey = memberName(&end);
 
-   if (!keey) return std::string();
 
-   return std::string(keey, end);
 
- }
 
- char const* ValueIteratorBase::memberName() const {
 
-   const char* cname = (*current_).first.data();
 
-   return cname ? cname : "";
 
- }
 
- char const* ValueIteratorBase::memberName(char const** end) const {
 
-   const char* cname = (*current_).first.data();
 
-   if (!cname) {
 
-     *end = NULL;
 
-     return NULL;
 
-   }
 
-   *end = cname + (*current_).first.length();
 
-   return cname;
 
- }
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // class ValueConstIterator
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- ValueConstIterator::ValueConstIterator() {}
 
- ValueConstIterator::ValueConstIterator(
 
-     const Value::ObjectValues::iterator& current)
 
-     : ValueIteratorBase(current) {}
 
- ValueConstIterator::ValueConstIterator(ValueIterator const& other)
 
-     : ValueIteratorBase(other) {}
 
- ValueConstIterator& ValueConstIterator::
 
- operator=(const ValueIteratorBase& other) {
 
-   copy(other);
 
-   return *this;
 
- }
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // class ValueIterator
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- ValueIterator::ValueIterator() {}
 
- ValueIterator::ValueIterator(const Value::ObjectValues::iterator& current)
 
-     : ValueIteratorBase(current) {}
 
- ValueIterator::ValueIterator(const ValueConstIterator& other)
 
-     : ValueIteratorBase(other) {
 
-   throwRuntimeError("ConstIterator to Iterator should never be allowed.");
 
- }
 
- ValueIterator::ValueIterator(const ValueIterator& other)
 
-     : ValueIteratorBase(other) {}
 
- ValueIterator& ValueIterator::operator=(const SelfType& other) {
 
-   copy(other);
 
-   return *this;
 
- }
 
- } // namespace Json
 
- // //////////////////////////////////////////////////////////////////////
 
- // End of content of file: src/lib_json/json_valueiterator.inl
 
- // //////////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////////
 
- // Beginning of content of file: src/lib_json/json_value.cpp
 
- // //////////////////////////////////////////////////////////////////////
 
- // Copyright 2011 Baptiste Lepilleur
 
- // Distributed under MIT license, or public domain if desired and
 
- // recognized in your jurisdiction.
 
- // See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
 
- #if !defined(JSON_IS_AMALGAMATION)
 
- #include <json/assertions.h>
 
- #include <json/value.h>
 
- #include <json/writer.h>
 
- #endif // if !defined(JSON_IS_AMALGAMATION)
 
- #include <math.h>
 
- #include <sstream>
 
- #include <utility>
 
- #include <cstring>
 
- #include <cassert>
 
- #ifdef JSON_USE_CPPTL
 
- #include <cpptl/conststring.h>
 
- #endif
 
- #include <cstddef> // size_t
 
- #include <algorithm> // min()
 
- #define JSON_ASSERT_UNREACHABLE assert(false)
 
- namespace Json {
 
- // This is a walkaround to avoid the static initialization of Value::null.
 
- // kNull must be word-aligned to avoid crashing on ARM.  We use an alignment of
 
- // 8 (instead of 4) as a bit of future-proofing.
 
- #if defined(__ARMEL__)
 
- #define ALIGNAS(byte_alignment) __attribute__((aligned(byte_alignment)))
 
- #else
 
- #define ALIGNAS(byte_alignment)
 
- #endif
 
- static const unsigned char ALIGNAS(8) kNull[sizeof(Value)] = { 0 };
 
- const unsigned char& kNullRef = kNull[0];
 
- const Value& Value::null = reinterpret_cast<const Value&>(kNullRef);
 
- const Value& Value::nullRef = null;
 
- const Int Value::minInt = Int(~(UInt(-1) / 2));
 
- const Int Value::maxInt = Int(UInt(-1) / 2);
 
- const UInt Value::maxUInt = UInt(-1);
 
- #if defined(JSON_HAS_INT64)
 
- const Int64 Value::minInt64 = Int64(~(UInt64(-1) / 2));
 
- const Int64 Value::maxInt64 = Int64(UInt64(-1) / 2);
 
- const UInt64 Value::maxUInt64 = UInt64(-1);
 
- // The constant is hard-coded because some compiler have trouble
 
- // converting Value::maxUInt64 to a double correctly (AIX/xlC).
 
- // Assumes that UInt64 is a 64 bits integer.
 
- static const double maxUInt64AsDouble = 18446744073709551615.0;
 
- #endif // defined(JSON_HAS_INT64)
 
- const LargestInt Value::minLargestInt = LargestInt(~(LargestUInt(-1) / 2));
 
- const LargestInt Value::maxLargestInt = LargestInt(LargestUInt(-1) / 2);
 
- const LargestUInt Value::maxLargestUInt = LargestUInt(-1);
 
- #if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
- template <typename T, typename U>
 
- static inline bool InRange(double d, T min, U max) {
 
-   return d >= min && d <= max;
 
- }
 
- #else  // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
- static inline double integerToDouble(Json::UInt64 value) {
 
-   return static_cast<double>(Int64(value / 2)) * 2.0 + Int64(value & 1);
 
- }
 
- template <typename T> static inline double integerToDouble(T value) {
 
-   return static_cast<double>(value);
 
- }
 
- template <typename T, typename U>
 
- static inline bool InRange(double d, T min, U max) {
 
-   return d >= integerToDouble(min) && d <= integerToDouble(max);
 
- }
 
- #endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
- /** Duplicates the specified string value.
 
-  * @param value Pointer to the string to duplicate. Must be zero-terminated if
 
-  *              length is "unknown".
 
-  * @param length Length of the value. if equals to unknown, then it will be
 
-  *               computed using strlen(value).
 
-  * @return Pointer on the duplicate instance of string.
 
-  */
 
- static inline char* duplicateStringValue(const char* value,
 
-                                          size_t length) {
 
-   // Avoid an integer overflow in the call to malloc below by limiting length
 
-   // to a sane value.
 
-   if (length >= (size_t)Value::maxInt)
 
-     length = Value::maxInt - 1;
 
-   char* newString = static_cast<char*>(malloc(length + 1));
 
-   if (newString == NULL) {
 
-     throwRuntimeError(
 
-         "in Json::Value::duplicateStringValue(): "
 
-         "Failed to allocate string value buffer");
 
-   }
 
-   memcpy(newString, value, length);
 
-   newString[length] = 0;
 
-   return newString;
 
- }
 
- /* Record the length as a prefix.
 
-  */
 
- static inline char* duplicateAndPrefixStringValue(
 
-     const char* value,
 
-     unsigned int length)
 
- {
 
-   // Avoid an integer overflow in the call to malloc below by limiting length
 
-   // to a sane value.
 
-   JSON_ASSERT_MESSAGE(length <= (unsigned)Value::maxInt - sizeof(unsigned) - 1U,
 
-                       "in Json::Value::duplicateAndPrefixStringValue(): "
 
-                       "length too big for prefixing");
 
-   unsigned actualLength = length + static_cast<unsigned>(sizeof(unsigned)) + 1U;
 
-   char* newString = static_cast<char*>(malloc(actualLength));
 
-   if (newString == 0) {
 
-     throwRuntimeError(
 
-         "in Json::Value::duplicateAndPrefixStringValue(): "
 
-         "Failed to allocate string value buffer");
 
-   }
 
-   *reinterpret_cast<unsigned*>(newString) = length;
 
-   memcpy(newString + sizeof(unsigned), value, length);
 
-   newString[actualLength - 1U] = 0; // to avoid buffer over-run accidents by users later
 
-   return newString;
 
- }
 
- inline static void decodePrefixedString(
 
-     bool isPrefixed, char const* prefixed,
 
-     unsigned* length, char const** value)
 
- {
 
-   if (!isPrefixed) {
 
-     *length = static_cast<unsigned>(strlen(prefixed));
 
-     *value = prefixed;
 
-   } else {
 
-     *length = *reinterpret_cast<unsigned const*>(prefixed);
 
-     *value = prefixed + sizeof(unsigned);
 
-   }
 
- }
 
- /** Free the string duplicated by duplicateStringValue()/duplicateAndPrefixStringValue().
 
-  */
 
- static inline void releaseStringValue(char* value) { free(value); }
 
- } // namespace Json
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // ValueInternals...
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- #if !defined(JSON_IS_AMALGAMATION)
 
- #include "json_valueiterator.inl"
 
- #endif // if !defined(JSON_IS_AMALGAMATION)
 
- namespace Json {
 
- Exception::Exception(std::string const& msg)
 
-   : msg_(msg)
 
- {}
 
- Exception::~Exception() throw()
 
- {}
 
- char const* Exception::what() const throw()
 
- {
 
-   return msg_.c_str();
 
- }
 
- RuntimeError::RuntimeError(std::string const& msg)
 
-   : Exception(msg)
 
- {}
 
- LogicError::LogicError(std::string const& msg)
 
-   : Exception(msg)
 
- {}
 
- void throwRuntimeError(std::string const& msg)
 
- {
 
-   throw RuntimeError(msg);
 
- }
 
- void throwLogicError(std::string const& msg)
 
- {
 
-   throw LogicError(msg);
 
- }
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // class Value::CommentInfo
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- Value::CommentInfo::CommentInfo() : comment_(0) {}
 
- Value::CommentInfo::~CommentInfo() {
 
-   if (comment_)
 
-     releaseStringValue(comment_);
 
- }
 
- void Value::CommentInfo::setComment(const char* text, size_t len) {
 
-   if (comment_) {
 
-     releaseStringValue(comment_);
 
-     comment_ = 0;
 
-   }
 
-   JSON_ASSERT(text != 0);
 
-   JSON_ASSERT_MESSAGE(
 
-       text[0] == '\0' || text[0] == '/',
 
-       "in Json::Value::setComment(): Comments must start with /");
 
-   // It seems that /**/ style comments are acceptable as well.
 
-   comment_ = duplicateStringValue(text, len);
 
- }
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // class Value::CZString
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // Notes: policy_ indicates if the string was allocated when
 
- // a string is stored.
 
- Value::CZString::CZString(ArrayIndex aindex) : cstr_(0), index_(aindex) {}
 
- Value::CZString::CZString(char const* str, unsigned ulength, DuplicationPolicy allocate)
 
-     : cstr_(str) {
 
-   // allocate != duplicate
 
-   storage_.policy_ = allocate & 0x3;
 
-   storage_.length_ = ulength & 0x3FFFFFFF;
 
- }
 
- Value::CZString::CZString(const CZString& other)
 
-     : cstr_(other.storage_.policy_ != noDuplication && other.cstr_ != 0
 
-                 ? duplicateStringValue(other.cstr_, other.storage_.length_)
 
-                 : other.cstr_) {
 
-   storage_.policy_ = (other.cstr_
 
-                  ? (static_cast<DuplicationPolicy>(other.storage_.policy_) == noDuplication
 
-                      ? noDuplication : duplicate)
 
-                  : static_cast<DuplicationPolicy>(other.storage_.policy_));
 
-   storage_.length_ = other.storage_.length_;
 
- }
 
- #if JSON_HAS_RVALUE_REFERENCES
 
- Value::CZString::CZString(CZString&& other)
 
-   : cstr_(other.cstr_), index_(other.index_) {
 
-   other.cstr_ = nullptr;
 
- }
 
- #endif
 
- Value::CZString::~CZString() {
 
-   if (cstr_ && storage_.policy_ == duplicate)
 
-     releaseStringValue(const_cast<char*>(cstr_));
 
- }
 
- void Value::CZString::swap(CZString& other) {
 
-   std::swap(cstr_, other.cstr_);
 
-   std::swap(index_, other.index_);
 
- }
 
- Value::CZString& Value::CZString::operator=(CZString other) {
 
-   swap(other);
 
-   return *this;
 
- }
 
- bool Value::CZString::operator<(const CZString& other) const {
 
-   if (!cstr_) return index_ < other.index_;
 
-   //return strcmp(cstr_, other.cstr_) < 0;
 
-   // Assume both are strings.
 
-   unsigned this_len = this->storage_.length_;
 
-   unsigned other_len = other.storage_.length_;
 
-   unsigned min_len = std::min(this_len, other_len);
 
-   int comp = memcmp(this->cstr_, other.cstr_, min_len);
 
-   if (comp < 0) return true;
 
-   if (comp > 0) return false;
 
-   return (this_len < other_len);
 
- }
 
- bool Value::CZString::operator==(const CZString& other) const {
 
-   if (!cstr_) return index_ == other.index_;
 
-   //return strcmp(cstr_, other.cstr_) == 0;
 
-   // Assume both are strings.
 
-   unsigned this_len = this->storage_.length_;
 
-   unsigned other_len = other.storage_.length_;
 
-   if (this_len != other_len) return false;
 
-   int comp = memcmp(this->cstr_, other.cstr_, this_len);
 
-   return comp == 0;
 
- }
 
- ArrayIndex Value::CZString::index() const { return index_; }
 
- //const char* Value::CZString::c_str() const { return cstr_; }
 
- const char* Value::CZString::data() const { return cstr_; }
 
- unsigned Value::CZString::length() const { return storage_.length_; }
 
- bool Value::CZString::isStaticString() const { return storage_.policy_ == noDuplication; }
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // class Value::Value
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////
 
- /*! \internal Default constructor initialization must be equivalent to:
 
-  * memset( this, 0, sizeof(Value) )
 
-  * This optimization is used in ValueInternalMap fast allocator.
 
-  */
 
- Value::Value(ValueType vtype) {
 
-   initBasic(vtype);
 
-   switch (vtype) {
 
-   case nullValue:
 
-     break;
 
-   case intValue:
 
-   case uintValue:
 
-     value_.int_ = 0;
 
-     break;
 
-   case realValue:
 
-     value_.real_ = 0.0;
 
-     break;
 
-   case stringValue:
 
-     value_.string_ = 0;
 
-     break;
 
-   case arrayValue:
 
-   case objectValue:
 
-     value_.map_ = new ObjectValues();
 
-     break;
 
-   case booleanValue:
 
-     value_.bool_ = false;
 
-     break;
 
-   default:
 
-     JSON_ASSERT_UNREACHABLE;
 
-   }
 
- }
 
- Value::Value(Int value) {
 
-   initBasic(intValue);
 
-   value_.int_ = value;
 
- }
 
- Value::Value(UInt value) {
 
-   initBasic(uintValue);
 
-   value_.uint_ = value;
 
- }
 
- #if defined(JSON_HAS_INT64)
 
- Value::Value(Int64 value) {
 
-   initBasic(intValue);
 
-   value_.int_ = value;
 
- }
 
- Value::Value(UInt64 value) {
 
-   initBasic(uintValue);
 
-   value_.uint_ = value;
 
- }
 
- #endif // defined(JSON_HAS_INT64)
 
- Value::Value(double value) {
 
-   initBasic(realValue);
 
-   value_.real_ = value;
 
- }
 
- Value::Value(const char* value) {
 
-   initBasic(stringValue, true);
 
-   value_.string_ = duplicateAndPrefixStringValue(value, static_cast<unsigned>(strlen(value)));
 
- }
 
- Value::Value(const char* beginValue, const char* endValue) {
 
-   initBasic(stringValue, true);
 
-   value_.string_ =
 
-       duplicateAndPrefixStringValue(beginValue, static_cast<unsigned>(endValue - beginValue));
 
- }
 
- Value::Value(const std::string& value) {
 
-   initBasic(stringValue, true);
 
-   value_.string_ =
 
-       duplicateAndPrefixStringValue(value.data(), static_cast<unsigned>(value.length()));
 
- }
 
- Value::Value(const StaticString& value) {
 
-   initBasic(stringValue);
 
-   value_.string_ = const_cast<char*>(value.c_str());
 
- }
 
- #ifdef JSON_USE_CPPTL
 
- Value::Value(const CppTL::ConstString& value) {
 
-   initBasic(stringValue, true);
 
-   value_.string_ = duplicateAndPrefixStringValue(value, static_cast<unsigned>(value.length()));
 
- }
 
- #endif
 
- Value::Value(bool value) {
 
-   initBasic(booleanValue);
 
-   value_.bool_ = value;
 
- }
 
- Value::Value(Value const& other)
 
-     : type_(other.type_), allocated_(false)
 
-       ,
 
-       comments_(0), start_(other.start_), limit_(other.limit_)
 
- {
 
-   switch (type_) {
 
-   case nullValue:
 
-   case intValue:
 
-   case uintValue:
 
-   case realValue:
 
-   case booleanValue:
 
-     value_ = other.value_;
 
-     break;
 
-   case stringValue:
 
-     if (other.value_.string_ && other.allocated_) {
 
-       unsigned len;
 
-       char const* str;
 
-       decodePrefixedString(other.allocated_, other.value_.string_,
 
-           &len, &str);
 
-       value_.string_ = duplicateAndPrefixStringValue(str, len);
 
-       allocated_ = true;
 
-     } else {
 
-       value_.string_ = other.value_.string_;
 
-       allocated_ = false;
 
-     }
 
-     break;
 
-   case arrayValue:
 
-   case objectValue:
 
-     value_.map_ = new ObjectValues(*other.value_.map_);
 
-     break;
 
-   default:
 
-     JSON_ASSERT_UNREACHABLE;
 
-   }
 
-   if (other.comments_) {
 
-     comments_ = new CommentInfo[numberOfCommentPlacement];
 
-     for (int comment = 0; comment < numberOfCommentPlacement; ++comment) {
 
-       const CommentInfo& otherComment = other.comments_[comment];
 
-       if (otherComment.comment_)
 
-         comments_[comment].setComment(
 
-             otherComment.comment_, strlen(otherComment.comment_));
 
-     }
 
-   }
 
- }
 
- #if JSON_HAS_RVALUE_REFERENCES
 
- // Move constructor
 
- Value::Value(Value&& other) {
 
-   initBasic(nullValue);
 
-   swap(other);
 
- }
 
- #endif
 
- Value::~Value() {
 
-   switch (type_) {
 
-   case nullValue:
 
-   case intValue:
 
-   case uintValue:
 
-   case realValue:
 
-   case booleanValue:
 
-     break;
 
-   case stringValue:
 
-     if (allocated_)
 
-       releaseStringValue(value_.string_);
 
-     break;
 
-   case arrayValue:
 
-   case objectValue:
 
-     delete value_.map_;
 
-     break;
 
-   default:
 
-     JSON_ASSERT_UNREACHABLE;
 
-   }
 
-   if (comments_)
 
-     delete[] comments_;
 
- }
 
- Value& Value::operator=(Value other) {
 
-   swap(other);
 
-   return *this;
 
- }
 
- void Value::swapPayload(Value& other) {
 
-   ValueType temp = type_;
 
-   type_ = other.type_;
 
-   other.type_ = temp;
 
-   std::swap(value_, other.value_);
 
-   int temp2 = allocated_;
 
-   allocated_ = other.allocated_;
 
-   other.allocated_ = temp2 & 0x1;
 
- }
 
- void Value::swap(Value& other) {
 
-   swapPayload(other);
 
-   std::swap(comments_, other.comments_);
 
-   std::swap(start_, other.start_);
 
-   std::swap(limit_, other.limit_);
 
- }
 
- ValueType Value::type() const { return type_; }
 
- int Value::compare(const Value& other) const {
 
-   if (*this < other)
 
-     return -1;
 
-   if (*this > other)
 
-     return 1;
 
-   return 0;
 
- }
 
- bool Value::operator<(const Value& other) const {
 
-   int typeDelta = type_ - other.type_;
 
-   if (typeDelta)
 
-     return typeDelta < 0 ? true : false;
 
-   switch (type_) {
 
-   case nullValue:
 
-     return false;
 
-   case intValue:
 
-     return value_.int_ < other.value_.int_;
 
-   case uintValue:
 
-     return value_.uint_ < other.value_.uint_;
 
-   case realValue:
 
-     return value_.real_ < other.value_.real_;
 
-   case booleanValue:
 
-     return value_.bool_ < other.value_.bool_;
 
-   case stringValue:
 
-   {
 
-     if ((value_.string_ == 0) || (other.value_.string_ == 0)) {
 
-       if (other.value_.string_) return true;
 
-       else return false;
 
-     }
 
-     unsigned this_len;
 
-     unsigned other_len;
 
-     char const* this_str;
 
-     char const* other_str;
 
-     decodePrefixedString(this->allocated_, this->value_.string_, &this_len, &this_str);
 
-     decodePrefixedString(other.allocated_, other.value_.string_, &other_len, &other_str);
 
-     unsigned min_len = std::min(this_len, other_len);
 
-     int comp = memcmp(this_str, other_str, min_len);
 
-     if (comp < 0) return true;
 
-     if (comp > 0) return false;
 
-     return (this_len < other_len);
 
-   }
 
-   case arrayValue:
 
-   case objectValue: {
 
-     int delta = int(value_.map_->size() - other.value_.map_->size());
 
-     if (delta)
 
-       return delta < 0;
 
-     return (*value_.map_) < (*other.value_.map_);
 
-   }
 
-   default:
 
-     JSON_ASSERT_UNREACHABLE;
 
-   }
 
-   return false; // unreachable
 
- }
 
- bool Value::operator<=(const Value& other) const { return !(other < *this); }
 
- bool Value::operator>=(const Value& other) const { return !(*this < other); }
 
- bool Value::operator>(const Value& other) const { return other < *this; }
 
- bool Value::operator==(const Value& other) const {
 
-   // if ( type_ != other.type_ )
 
-   // GCC 2.95.3 says:
 
-   // attempt to take address of bit-field structure member `Json::Value::type_'
 
-   // Beats me, but a temp solves the problem.
 
-   int temp = other.type_;
 
-   if (type_ != temp)
 
-     return false;
 
-   switch (type_) {
 
-   case nullValue:
 
-     return true;
 
-   case intValue:
 
-     return value_.int_ == other.value_.int_;
 
-   case uintValue:
 
-     return value_.uint_ == other.value_.uint_;
 
-   case realValue:
 
-     return value_.real_ == other.value_.real_;
 
-   case booleanValue:
 
-     return value_.bool_ == other.value_.bool_;
 
-   case stringValue:
 
-   {
 
-     if ((value_.string_ == 0) || (other.value_.string_ == 0)) {
 
-       return (value_.string_ == other.value_.string_);
 
-     }
 
-     unsigned this_len;
 
-     unsigned other_len;
 
-     char const* this_str;
 
-     char const* other_str;
 
-     decodePrefixedString(this->allocated_, this->value_.string_, &this_len, &this_str);
 
-     decodePrefixedString(other.allocated_, other.value_.string_, &other_len, &other_str);
 
-     if (this_len != other_len) return false;
 
-     int comp = memcmp(this_str, other_str, this_len);
 
-     return comp == 0;
 
-   }
 
-   case arrayValue:
 
-   case objectValue:
 
-     return value_.map_->size() == other.value_.map_->size() &&
 
-            (*value_.map_) == (*other.value_.map_);
 
-   default:
 
-     JSON_ASSERT_UNREACHABLE;
 
-   }
 
-   return false; // unreachable
 
- }
 
- bool Value::operator!=(const Value& other) const { return !(*this == other); }
 
- const char* Value::asCString() const {
 
-   JSON_ASSERT_MESSAGE(type_ == stringValue,
 
-                       "in Json::Value::asCString(): requires stringValue");
 
-   if (value_.string_ == 0) return 0;
 
-   unsigned this_len;
 
-   char const* this_str;
 
-   decodePrefixedString(this->allocated_, this->value_.string_, &this_len, &this_str);
 
-   return this_str;
 
- }
 
- bool Value::getString(char const** str, char const** cend) const {
 
-   if (type_ != stringValue) return false;
 
-   if (value_.string_ == 0) return false;
 
-   unsigned length;
 
-   decodePrefixedString(this->allocated_, this->value_.string_, &length, str);
 
-   *cend = *str + length;
 
-   return true;
 
- }
 
- std::string Value::asString() const {
 
-   switch (type_) {
 
-   case nullValue:
 
-     return "";
 
-   case stringValue:
 
-   {
 
-     if (value_.string_ == 0) return "";
 
-     unsigned this_len;
 
-     char const* this_str;
 
-     decodePrefixedString(this->allocated_, this->value_.string_, &this_len, &this_str);
 
-     return std::string(this_str, this_len);
 
-   }
 
-   case booleanValue:
 
-     return value_.bool_ ? "true" : "false";
 
-   case intValue:
 
-     return valueToString(value_.int_);
 
-   case uintValue:
 
-     return valueToString(value_.uint_);
 
-   case realValue:
 
-     return valueToString(value_.real_);
 
-   default:
 
-     JSON_FAIL_MESSAGE("Type is not convertible to string");
 
-   }
 
- }
 
- #ifdef JSON_USE_CPPTL
 
- CppTL::ConstString Value::asConstString() const {
 
-   unsigned len;
 
-   char const* str;
 
-   decodePrefixedString(allocated_, value_.string_,
 
-       &len, &str);
 
-   return CppTL::ConstString(str, len);
 
- }
 
- #endif
 
- Value::Int Value::asInt() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     JSON_ASSERT_MESSAGE(isInt(), "LargestInt out of Int range");
 
-     return Int(value_.int_);
 
-   case uintValue:
 
-     JSON_ASSERT_MESSAGE(isInt(), "LargestUInt out of Int range");
 
-     return Int(value_.uint_);
 
-   case realValue:
 
-     JSON_ASSERT_MESSAGE(InRange(value_.real_, minInt, maxInt),
 
-                         "double out of Int range");
 
-     return Int(value_.real_);
 
-   case nullValue:
 
-     return 0;
 
-   case booleanValue:
 
-     return value_.bool_ ? 1 : 0;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to Int.");
 
- }
 
- Value::UInt Value::asUInt() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     JSON_ASSERT_MESSAGE(isUInt(), "LargestInt out of UInt range");
 
-     return UInt(value_.int_);
 
-   case uintValue:
 
-     JSON_ASSERT_MESSAGE(isUInt(), "LargestUInt out of UInt range");
 
-     return UInt(value_.uint_);
 
-   case realValue:
 
-     JSON_ASSERT_MESSAGE(InRange(value_.real_, 0, maxUInt),
 
-                         "double out of UInt range");
 
-     return UInt(value_.real_);
 
-   case nullValue:
 
-     return 0;
 
-   case booleanValue:
 
-     return value_.bool_ ? 1 : 0;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to UInt.");
 
- }
 
- #if defined(JSON_HAS_INT64)
 
- Value::Int64 Value::asInt64() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     return Int64(value_.int_);
 
-   case uintValue:
 
-     JSON_ASSERT_MESSAGE(isInt64(), "LargestUInt out of Int64 range");
 
-     return Int64(value_.uint_);
 
-   case realValue:
 
-     JSON_ASSERT_MESSAGE(InRange(value_.real_, minInt64, maxInt64),
 
-                         "double out of Int64 range");
 
-     return Int64(value_.real_);
 
-   case nullValue:
 
-     return 0;
 
-   case booleanValue:
 
-     return value_.bool_ ? 1 : 0;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to Int64.");
 
- }
 
- Value::UInt64 Value::asUInt64() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     JSON_ASSERT_MESSAGE(isUInt64(), "LargestInt out of UInt64 range");
 
-     return UInt64(value_.int_);
 
-   case uintValue:
 
-     return UInt64(value_.uint_);
 
-   case realValue:
 
-     JSON_ASSERT_MESSAGE(InRange(value_.real_, 0, maxUInt64),
 
-                         "double out of UInt64 range");
 
-     return UInt64(value_.real_);
 
-   case nullValue:
 
-     return 0;
 
-   case booleanValue:
 
-     return value_.bool_ ? 1 : 0;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to UInt64.");
 
- }
 
- #endif // if defined(JSON_HAS_INT64)
 
- LargestInt Value::asLargestInt() const {
 
- #if defined(JSON_NO_INT64)
 
-   return asInt();
 
- #else
 
-   return asInt64();
 
- #endif
 
- }
 
- LargestUInt Value::asLargestUInt() const {
 
- #if defined(JSON_NO_INT64)
 
-   return asUInt();
 
- #else
 
-   return asUInt64();
 
- #endif
 
- }
 
- double Value::asDouble() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     return static_cast<double>(value_.int_);
 
-   case uintValue:
 
- #if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
-     return static_cast<double>(value_.uint_);
 
- #else  // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
-     return integerToDouble(value_.uint_);
 
- #endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
-   case realValue:
 
-     return value_.real_;
 
-   case nullValue:
 
-     return 0.0;
 
-   case booleanValue:
 
-     return value_.bool_ ? 1.0 : 0.0;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to double.");
 
- }
 
- float Value::asFloat() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     return static_cast<float>(value_.int_);
 
-   case uintValue:
 
- #if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
-     return static_cast<float>(value_.uint_);
 
- #else  // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
-     return integerToDouble(value_.uint_);
 
- #endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
 
-   case realValue:
 
-     return static_cast<float>(value_.real_);
 
-   case nullValue:
 
-     return 0.0;
 
-   case booleanValue:
 
-     return value_.bool_ ? 1.0f : 0.0f;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to float.");
 
- }
 
- bool Value::asBool() const {
 
-   switch (type_) {
 
-   case booleanValue:
 
-     return value_.bool_;
 
-   case nullValue:
 
-     return false;
 
-   case intValue:
 
-     return value_.int_ ? true : false;
 
-   case uintValue:
 
-     return value_.uint_ ? true : false;
 
-   case realValue:
 
-     // This is kind of strange. Not recommended.
 
-     return (value_.real_ != 0.0) ? true : false;
 
-   default:
 
-     break;
 
-   }
 
-   JSON_FAIL_MESSAGE("Value is not convertible to bool.");
 
- }
 
- bool Value::isConvertibleTo(ValueType other) const {
 
-   switch (other) {
 
-   case nullValue:
 
-     return (isNumeric() && asDouble() == 0.0) ||
 
-            (type_ == booleanValue && value_.bool_ == false) ||
 
-            (type_ == stringValue && asString() == "") ||
 
-            (type_ == arrayValue && value_.map_->size() == 0) ||
 
-            (type_ == objectValue && value_.map_->size() == 0) ||
 
-            type_ == nullValue;
 
-   case intValue:
 
-     return isInt() ||
 
-            (type_ == realValue && InRange(value_.real_, minInt, maxInt)) ||
 
-            type_ == booleanValue || type_ == nullValue;
 
-   case uintValue:
 
-     return isUInt() ||
 
-            (type_ == realValue && InRange(value_.real_, 0, maxUInt)) ||
 
-            type_ == booleanValue || type_ == nullValue;
 
-   case realValue:
 
-     return isNumeric() || type_ == booleanValue || type_ == nullValue;
 
-   case booleanValue:
 
-     return isNumeric() || type_ == booleanValue || type_ == nullValue;
 
-   case stringValue:
 
-     return isNumeric() || type_ == booleanValue || type_ == stringValue ||
 
-            type_ == nullValue;
 
-   case arrayValue:
 
-     return type_ == arrayValue || type_ == nullValue;
 
-   case objectValue:
 
-     return type_ == objectValue || type_ == nullValue;
 
-   }
 
-   JSON_ASSERT_UNREACHABLE;
 
-   return false;
 
- }
 
- /// Number of values in array or object
 
- ArrayIndex Value::size() const {
 
-   switch (type_) {
 
-   case nullValue:
 
-   case intValue:
 
-   case uintValue:
 
-   case realValue:
 
-   case booleanValue:
 
-   case stringValue:
 
-     return 0;
 
-   case arrayValue: // size of the array is highest index + 1
 
-     if (!value_.map_->empty()) {
 
-       ObjectValues::const_iterator itLast = value_.map_->end();
 
-       --itLast;
 
-       return (*itLast).first.index() + 1;
 
-     }
 
-     return 0;
 
-   case objectValue:
 
-     return ArrayIndex(value_.map_->size());
 
-   }
 
-   JSON_ASSERT_UNREACHABLE;
 
-   return 0; // unreachable;
 
- }
 
- bool Value::empty() const {
 
-   if (isNull() || isArray() || isObject())
 
-     return size() == 0u;
 
-   else
 
-     return false;
 
- }
 
- bool Value::operator!() const { return isNull(); }
 
- void Value::clear() {
 
-   JSON_ASSERT_MESSAGE(type_ == nullValue || type_ == arrayValue ||
 
-                           type_ == objectValue,
 
-                       "in Json::Value::clear(): requires complex value");
 
-   start_ = 0;
 
-   limit_ = 0;
 
-   switch (type_) {
 
-   case arrayValue:
 
-   case objectValue:
 
-     value_.map_->clear();
 
-     break;
 
-   default:
 
-     break;
 
-   }
 
- }
 
- void Value::resize(ArrayIndex newSize) {
 
-   JSON_ASSERT_MESSAGE(type_ == nullValue || type_ == arrayValue,
 
-                       "in Json::Value::resize(): requires arrayValue");
 
-   if (type_ == nullValue)
 
-     *this = Value(arrayValue);
 
-   ArrayIndex oldSize = size();
 
-   if (newSize == 0)
 
-     clear();
 
-   else if (newSize > oldSize)
 
-     (*this)[newSize - 1];
 
-   else {
 
-     for (ArrayIndex index = newSize; index < oldSize; ++index) {
 
-       value_.map_->erase(index);
 
-     }
 
-     assert(size() == newSize);
 
-   }
 
- }
 
- Value& Value::operator[](ArrayIndex index) {
 
-   JSON_ASSERT_MESSAGE(
 
-       type_ == nullValue || type_ == arrayValue,
 
-       "in Json::Value::operator[](ArrayIndex): requires arrayValue");
 
-   if (type_ == nullValue)
 
-     *this = Value(arrayValue);
 
-   CZString key(index);
 
-   ObjectValues::iterator it = value_.map_->lower_bound(key);
 
-   if (it != value_.map_->end() && (*it).first == key)
 
-     return (*it).second;
 
-   ObjectValues::value_type defaultValue(key, nullRef);
 
-   it = value_.map_->insert(it, defaultValue);
 
-   return (*it).second;
 
- }
 
- Value& Value::operator[](int index) {
 
-   JSON_ASSERT_MESSAGE(
 
-       index >= 0,
 
-       "in Json::Value::operator[](int index): index cannot be negative");
 
-   return (*this)[ArrayIndex(index)];
 
- }
 
- const Value& Value::operator[](ArrayIndex index) const {
 
-   JSON_ASSERT_MESSAGE(
 
-       type_ == nullValue || type_ == arrayValue,
 
-       "in Json::Value::operator[](ArrayIndex)const: requires arrayValue");
 
-   if (type_ == nullValue)
 
-     return nullRef;
 
-   CZString key(index);
 
-   ObjectValues::const_iterator it = value_.map_->find(key);
 
-   if (it == value_.map_->end())
 
-     return nullRef;
 
-   return (*it).second;
 
- }
 
- const Value& Value::operator[](int index) const {
 
-   JSON_ASSERT_MESSAGE(
 
-       index >= 0,
 
-       "in Json::Value::operator[](int index) const: index cannot be negative");
 
-   return (*this)[ArrayIndex(index)];
 
- }
 
- void Value::initBasic(ValueType vtype, bool allocated) {
 
-   type_ = vtype;
 
-   allocated_ = allocated;
 
-   comments_ = 0;
 
-   start_ = 0;
 
-   limit_ = 0;
 
- }
 
- // Access an object value by name, create a null member if it does not exist.
 
- // @pre Type of '*this' is object or null.
 
- // @param key is null-terminated.
 
- Value& Value::resolveReference(const char* key) {
 
-   JSON_ASSERT_MESSAGE(
 
-       type_ == nullValue || type_ == objectValue,
 
-       "in Json::Value::resolveReference(): requires objectValue");
 
-   if (type_ == nullValue)
 
-     *this = Value(objectValue);
 
-   CZString actualKey(
 
-       key, static_cast<unsigned>(strlen(key)), CZString::noDuplication); // NOTE!
 
-   ObjectValues::iterator it = value_.map_->lower_bound(actualKey);
 
-   if (it != value_.map_->end() && (*it).first == actualKey)
 
-     return (*it).second;
 
-   ObjectValues::value_type defaultValue(actualKey, nullRef);
 
-   it = value_.map_->insert(it, defaultValue);
 
-   Value& value = (*it).second;
 
-   return value;
 
- }
 
- // @param key is not null-terminated.
 
- Value& Value::resolveReference(char const* key, char const* cend)
 
- {
 
-   JSON_ASSERT_MESSAGE(
 
-       type_ == nullValue || type_ == objectValue,
 
-       "in Json::Value::resolveReference(key, end): requires objectValue");
 
-   if (type_ == nullValue)
 
-     *this = Value(objectValue);
 
-   CZString actualKey(
 
-       key, static_cast<unsigned>(cend-key), CZString::duplicateOnCopy);
 
-   ObjectValues::iterator it = value_.map_->lower_bound(actualKey);
 
-   if (it != value_.map_->end() && (*it).first == actualKey)
 
-     return (*it).second;
 
-   ObjectValues::value_type defaultValue(actualKey, nullRef);
 
-   it = value_.map_->insert(it, defaultValue);
 
-   Value& value = (*it).second;
 
-   return value;
 
- }
 
- Value Value::get(ArrayIndex index, const Value& defaultValue) const {
 
-   const Value* value = &((*this)[index]);
 
-   return value == &nullRef ? defaultValue : *value;
 
- }
 
- bool Value::isValidIndex(ArrayIndex index) const { return index < size(); }
 
- Value const* Value::find(char const* key, char const* cend) const
 
- {
 
-   JSON_ASSERT_MESSAGE(
 
-       type_ == nullValue || type_ == objectValue,
 
-       "in Json::Value::find(key, end, found): requires objectValue or nullValue");
 
-   if (type_ == nullValue) return NULL;
 
-   CZString actualKey(key, static_cast<unsigned>(cend-key), CZString::noDuplication);
 
-   ObjectValues::const_iterator it = value_.map_->find(actualKey);
 
-   if (it == value_.map_->end()) return NULL;
 
-   return &(*it).second;
 
- }
 
- const Value& Value::operator[](const char* key) const
 
- {
 
-   Value const* found = find(key, key + strlen(key));
 
-   if (!found) return nullRef;
 
-   return *found;
 
- }
 
- Value const& Value::operator[](std::string const& key) const
 
- {
 
-   Value const* found = find(key.data(), key.data() + key.length());
 
-   if (!found) return nullRef;
 
-   return *found;
 
- }
 
- Value& Value::operator[](const char* key) {
 
-   return resolveReference(key, key + strlen(key));
 
- }
 
- Value& Value::operator[](const std::string& key) {
 
-   return resolveReference(key.data(), key.data() + key.length());
 
- }
 
- Value& Value::operator[](const StaticString& key) {
 
-   return resolveReference(key.c_str());
 
- }
 
- #ifdef JSON_USE_CPPTL
 
- Value& Value::operator[](const CppTL::ConstString& key) {
 
-   return resolveReference(key.c_str(), key.end_c_str());
 
- }
 
- Value const& Value::operator[](CppTL::ConstString const& key) const
 
- {
 
-   Value const* found = find(key.c_str(), key.end_c_str());
 
-   if (!found) return nullRef;
 
-   return *found;
 
- }
 
- #endif
 
- Value& Value::append(const Value& value) { return (*this)[size()] = value; }
 
- Value Value::get(char const* key, char const* cend, Value const& defaultValue) const
 
- {
 
-   Value const* found = find(key, cend);
 
-   return !found ? defaultValue : *found;
 
- }
 
- Value Value::get(char const* key, Value const& defaultValue) const
 
- {
 
-   return get(key, key + strlen(key), defaultValue);
 
- }
 
- Value Value::get(std::string const& key, Value const& defaultValue) const
 
- {
 
-   return get(key.data(), key.data() + key.length(), defaultValue);
 
- }
 
- bool Value::removeMember(const char* key, const char* cend, Value* removed)
 
- {
 
-   if (type_ != objectValue) {
 
-     return false;
 
-   }
 
-   CZString actualKey(key, static_cast<unsigned>(cend-key), CZString::noDuplication);
 
-   ObjectValues::iterator it = value_.map_->find(actualKey);
 
-   if (it == value_.map_->end())
 
-     return false;
 
-   *removed = it->second;
 
-   value_.map_->erase(it);
 
-   return true;
 
- }
 
- bool Value::removeMember(const char* key, Value* removed)
 
- {
 
-   return removeMember(key, key + strlen(key), removed);
 
- }
 
- bool Value::removeMember(std::string const& key, Value* removed)
 
- {
 
-   return removeMember(key.data(), key.data() + key.length(), removed);
 
- }
 
- Value Value::removeMember(const char* key)
 
- {
 
-   JSON_ASSERT_MESSAGE(type_ == nullValue || type_ == objectValue,
 
-                       "in Json::Value::removeMember(): requires objectValue");
 
-   if (type_ == nullValue)
 
-     return nullRef;
 
-   Value removed;  // null
 
-   removeMember(key, key + strlen(key), &removed);
 
-   return removed; // still null if removeMember() did nothing
 
- }
 
- Value Value::removeMember(const std::string& key)
 
- {
 
-   return removeMember(key.c_str());
 
- }
 
- bool Value::removeIndex(ArrayIndex index, Value* removed) {
 
-   if (type_ != arrayValue) {
 
-     return false;
 
-   }
 
-   CZString key(index);
 
-   ObjectValues::iterator it = value_.map_->find(key);
 
-   if (it == value_.map_->end()) {
 
-     return false;
 
-   }
 
-   *removed = it->second;
 
-   ArrayIndex oldSize = size();
 
-   // shift left all items left, into the place of the "removed"
 
-   for (ArrayIndex i = index; i < (oldSize - 1); ++i){
 
-     CZString keey(i);
 
-     (*value_.map_)[keey] = (*this)[i + 1];
 
-   }
 
-   // erase the last one ("leftover")
 
-   CZString keyLast(oldSize - 1);
 
-   ObjectValues::iterator itLast = value_.map_->find(keyLast);
 
-   value_.map_->erase(itLast);
 
-   return true;
 
- }
 
- #ifdef JSON_USE_CPPTL
 
- Value Value::get(const CppTL::ConstString& key,
 
-                  const Value& defaultValue) const {
 
-   return get(key.c_str(), key.end_c_str(), defaultValue);
 
- }
 
- #endif
 
- bool Value::isMember(char const* key, char const* cend) const
 
- {
 
-   Value const* value = find(key, cend);
 
-   return NULL != value;
 
- }
 
- bool Value::isMember(char const* key) const
 
- {
 
-   return isMember(key, key + strlen(key));
 
- }
 
- bool Value::isMember(std::string const& key) const
 
- {
 
-   return isMember(key.data(), key.data() + key.length());
 
- }
 
- #ifdef JSON_USE_CPPTL
 
- bool Value::isMember(const CppTL::ConstString& key) const {
 
-   return isMember(key.c_str(), key.end_c_str());
 
- }
 
- #endif
 
- Value::Members Value::getMemberNames() const {
 
-   JSON_ASSERT_MESSAGE(
 
-       type_ == nullValue || type_ == objectValue,
 
-       "in Json::Value::getMemberNames(), value must be objectValue");
 
-   if (type_ == nullValue)
 
-     return Value::Members();
 
-   Members members;
 
-   members.reserve(value_.map_->size());
 
-   ObjectValues::const_iterator it = value_.map_->begin();
 
-   ObjectValues::const_iterator itEnd = value_.map_->end();
 
-   for (; it != itEnd; ++it) {
 
-     members.push_back(std::string((*it).first.data(),
 
-                                   (*it).first.length()));
 
-   }
 
-   return members;
 
- }
 
- //
 
- //# ifdef JSON_USE_CPPTL
 
- // EnumMemberNames
 
- // Value::enumMemberNames() const
 
- //{
 
- //   if ( type_ == objectValue )
 
- //   {
 
- //      return CppTL::Enum::any(  CppTL::Enum::transform(
 
- //         CppTL::Enum::keys( *(value_.map_), CppTL::Type<const CZString &>() ),
 
- //         MemberNamesTransform() ) );
 
- //   }
 
- //   return EnumMemberNames();
 
- //}
 
- //
 
- //
 
- // EnumValues
 
- // Value::enumValues() const
 
- //{
 
- //   if ( type_ == objectValue  ||  type_ == arrayValue )
 
- //      return CppTL::Enum::anyValues( *(value_.map_),
 
- //                                     CppTL::Type<const Value &>() );
 
- //   return EnumValues();
 
- //}
 
- //
 
- //# endif
 
- static bool IsIntegral(double d) {
 
-   double integral_part;
 
-   return modf(d, &integral_part) == 0.0;
 
- }
 
- bool Value::isNull() const { return type_ == nullValue; }
 
- bool Value::isBool() const { return type_ == booleanValue; }
 
- bool Value::isInt() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     return value_.int_ >= minInt && value_.int_ <= maxInt;
 
-   case uintValue:
 
-     return value_.uint_ <= UInt(maxInt);
 
-   case realValue:
 
-     return value_.real_ >= minInt && value_.real_ <= maxInt &&
 
-            IsIntegral(value_.real_);
 
-   default:
 
-     break;
 
-   }
 
-   return false;
 
- }
 
- bool Value::isUInt() const {
 
-   switch (type_) {
 
-   case intValue:
 
-     return value_.int_ >= 0 && LargestUInt(value_.int_) <= LargestUInt(maxUInt);
 
-   case uintValue:
 
-     return value_.uint_ <= maxUInt;
 
-   case realValue:
 
-     return value_.real_ >= 0 && value_.real_ <= maxUInt &&
 
-            IsIntegral(value_.real_);
 
-   default:
 
-     break;
 
-   }
 
-   return false;
 
- }
 
- bool Value::isInt64() const {
 
- #if defined(JSON_HAS_INT64)
 
-   switch (type_) {
 
-   case intValue:
 
-     return true;
 
-   case uintValue:
 
-     return value_.uint_ <= UInt64(maxInt64);
 
-   case realValue:
 
-     // Note that maxInt64 (= 2^63 - 1) is not exactly representable as a
 
-     // double, so double(maxInt64) will be rounded up to 2^63. Therefore we
 
-     // require the value to be strictly less than the limit.
 
-     return value_.real_ >= double(minInt64) &&
 
-            value_.real_ < double(maxInt64) && IsIntegral(value_.real_);
 
-   default:
 
-     break;
 
-   }
 
- #endif // JSON_HAS_INT64
 
-   return false;
 
- }
 
- bool Value::isUInt64() const {
 
- #if defined(JSON_HAS_INT64)
 
-   switch (type_) {
 
-   case intValue:
 
-     return value_.int_ >= 0;
 
-   case uintValue:
 
-     return true;
 
-   case realValue:
 
-     // Note that maxUInt64 (= 2^64 - 1) is not exactly representable as a
 
-     // double, so double(maxUInt64) will be rounded up to 2^64. Therefore we
 
-     // require the value to be strictly less than the limit.
 
-     return value_.real_ >= 0 && value_.real_ < maxUInt64AsDouble &&
 
-            IsIntegral(value_.real_);
 
-   default:
 
-     break;
 
-   }
 
- #endif // JSON_HAS_INT64
 
-   return false;
 
- }
 
- bool Value::isIntegral() const {
 
- #if defined(JSON_HAS_INT64)
 
-   return isInt64() || isUInt64();
 
- #else
 
-   return isInt() || isUInt();
 
- #endif
 
- }
 
- bool Value::isDouble() const { return type_ == realValue || isIntegral(); }
 
- bool Value::isNumeric() const { return isIntegral() || isDouble(); }
 
- bool Value::isString() const { return type_ == stringValue; }
 
- bool Value::isArray() const { return type_ == arrayValue; }
 
- bool Value::isObject() const { return type_ == objectValue; }
 
- void Value::setComment(const char* comment, size_t len, CommentPlacement placement) {
 
-   if (!comments_)
 
-     comments_ = new CommentInfo[numberOfCommentPlacement];
 
-   if ((len > 0) && (comment[len-1] == '\n')) {
 
-     // Always discard trailing newline, to aid indentation.
 
-     len -= 1;
 
-   }
 
-   comments_[placement].setComment(comment, len);
 
- }
 
- void Value::setComment(const char* comment, CommentPlacement placement) {
 
-   setComment(comment, strlen(comment), placement);
 
- }
 
- void Value::setComment(const std::string& comment, CommentPlacement placement) {
 
-   setComment(comment.c_str(), comment.length(), placement);
 
- }
 
- bool Value::hasComment(CommentPlacement placement) const {
 
-   return comments_ != 0 && comments_[placement].comment_ != 0;
 
- }
 
- std::string Value::getComment(CommentPlacement placement) const {
 
-   if (hasComment(placement))
 
-     return comments_[placement].comment_;
 
-   return "";
 
- }
 
- void Value::setOffsetStart(size_t start) { start_ = start; }
 
- void Value::setOffsetLimit(size_t limit) { limit_ = limit; }
 
- size_t Value::getOffsetStart() const { return start_; }
 
- size_t Value::getOffsetLimit() const { return limit_; }
 
- std::string Value::toStyledString() const {
 
-   StyledWriter writer;
 
-   return writer.write(*this);
 
- }
 
- Value::const_iterator Value::begin() const {
 
-   switch (type_) {
 
-   case arrayValue:
 
-   case objectValue:
 
-     if (value_.map_)
 
-       return const_iterator(value_.map_->begin());
 
-     break;
 
-   default:
 
-     break;
 
-   }
 
-   return const_iterator();
 
- }
 
- Value::const_iterator Value::end() const {
 
-   switch (type_) {
 
-   case arrayValue:
 
-   case objectValue:
 
-     if (value_.map_)
 
-       return const_iterator(value_.map_->end());
 
-     break;
 
-   default:
 
-     break;
 
-   }
 
-   return const_iterator();
 
- }
 
- Value::iterator Value::begin() {
 
-   switch (type_) {
 
-   case arrayValue:
 
-   case objectValue:
 
-     if (value_.map_)
 
-       return iterator(value_.map_->begin());
 
-     break;
 
-   default:
 
-     break;
 
-   }
 
-   return iterator();
 
- }
 
- Value::iterator Value::end() {
 
-   switch (type_) {
 
-   case arrayValue:
 
-   case objectValue:
 
-     if (value_.map_)
 
-       return iterator(value_.map_->end());
 
-     break;
 
-   default:
 
-     break;
 
-   }
 
-   return iterator();
 
- }
 
- // class PathArgument
 
- // //////////////////////////////////////////////////////////////////
 
- PathArgument::PathArgument() : key_(), index_(), kind_(kindNone) {}
 
- PathArgument::PathArgument(ArrayIndex index)
 
-     : key_(), index_(index), kind_(kindIndex) {}
 
- PathArgument::PathArgument(const char* key)
 
-     : key_(key), index_(), kind_(kindKey) {}
 
- PathArgument::PathArgument(const std::string& key)
 
-     : key_(key.c_str()), index_(), kind_(kindKey) {}
 
- // class Path
 
- // //////////////////////////////////////////////////////////////////
 
- Path::Path(const std::string& path,
 
-            const PathArgument& a1,
 
-            const PathArgument& a2,
 
-            const PathArgument& a3,
 
-            const PathArgument& a4,
 
-            const PathArgument& a5) {
 
-   InArgs in;
 
-   in.push_back(&a1);
 
-   in.push_back(&a2);
 
-   in.push_back(&a3);
 
-   in.push_back(&a4);
 
-   in.push_back(&a5);
 
-   makePath(path, in);
 
- }
 
- void Path::makePath(const std::string& path, const InArgs& in) {
 
-   const char* current = path.c_str();
 
-   const char* end = current + path.length();
 
-   InArgs::const_iterator itInArg = in.begin();
 
-   while (current != end) {
 
-     if (*current == '[') {
 
-       ++current;
 
-       if (*current == '%')
 
-         addPathInArg(path, in, itInArg, PathArgument::kindIndex);
 
-       else {
 
-         ArrayIndex index = 0;
 
-         for (; current != end && *current >= '0' && *current <= '9'; ++current)
 
-           index = index * 10 + ArrayIndex(*current - '0');
 
-         args_.push_back(index);
 
-       }
 
-       if (current == end || *current++ != ']')
 
-         invalidPath(path, int(current - path.c_str()));
 
-     } else if (*current == '%') {
 
-       addPathInArg(path, in, itInArg, PathArgument::kindKey);
 
-       ++current;
 
-     } else if (*current == '.') {
 
-       ++current;
 
-     } else {
 
-       const char* beginName = current;
 
-       while (current != end && !strchr("[.", *current))
 
-         ++current;
 
-       args_.push_back(std::string(beginName, current));
 
-     }
 
-   }
 
- }
 
- void Path::addPathInArg(const std::string& /*path*/,
 
-                         const InArgs& in,
 
-                         InArgs::const_iterator& itInArg,
 
-                         PathArgument::Kind kind) {
 
-   if (itInArg == in.end()) {
 
-     // Error: missing argument %d
 
-   } else if ((*itInArg)->kind_ != kind) {
 
-     // Error: bad argument type
 
-   } else {
 
-     args_.push_back(**itInArg);
 
-   }
 
- }
 
- void Path::invalidPath(const std::string& /*path*/, int /*location*/) {
 
-   // Error: invalid path.
 
- }
 
- const Value& Path::resolve(const Value& root) const {
 
-   const Value* node = &root;
 
-   for (Args::const_iterator it = args_.begin(); it != args_.end(); ++it) {
 
-     const PathArgument& arg = *it;
 
-     if (arg.kind_ == PathArgument::kindIndex) {
 
-       if (!node->isArray() || !node->isValidIndex(arg.index_)) {
 
-         // Error: unable to resolve path (array value expected at position...
 
-       }
 
-       node = &((*node)[arg.index_]);
 
-     } else if (arg.kind_ == PathArgument::kindKey) {
 
-       if (!node->isObject()) {
 
-         // Error: unable to resolve path (object value expected at position...)
 
-       }
 
-       node = &((*node)[arg.key_]);
 
-       if (node == &Value::nullRef) {
 
-         // Error: unable to resolve path (object has no member named '' at
 
-         // position...)
 
-       }
 
-     }
 
-   }
 
-   return *node;
 
- }
 
- Value Path::resolve(const Value& root, const Value& defaultValue) const {
 
-   const Value* node = &root;
 
-   for (Args::const_iterator it = args_.begin(); it != args_.end(); ++it) {
 
-     const PathArgument& arg = *it;
 
-     if (arg.kind_ == PathArgument::kindIndex) {
 
-       if (!node->isArray() || !node->isValidIndex(arg.index_))
 
-         return defaultValue;
 
-       node = &((*node)[arg.index_]);
 
-     } else if (arg.kind_ == PathArgument::kindKey) {
 
-       if (!node->isObject())
 
-         return defaultValue;
 
-       node = &((*node)[arg.key_]);
 
-       if (node == &Value::nullRef)
 
-         return defaultValue;
 
-     }
 
-   }
 
-   return *node;
 
- }
 
- Value& Path::make(Value& root) const {
 
-   Value* node = &root;
 
-   for (Args::const_iterator it = args_.begin(); it != args_.end(); ++it) {
 
-     const PathArgument& arg = *it;
 
-     if (arg.kind_ == PathArgument::kindIndex) {
 
-       if (!node->isArray()) {
 
-         // Error: node is not an array at position ...
 
-       }
 
-       node = &((*node)[arg.index_]);
 
-     } else if (arg.kind_ == PathArgument::kindKey) {
 
-       if (!node->isObject()) {
 
-         // Error: node is not an object at position...
 
-       }
 
-       node = &((*node)[arg.key_]);
 
-     }
 
-   }
 
-   return *node;
 
- }
 
- } // namespace Json
 
- // //////////////////////////////////////////////////////////////////////
 
- // End of content of file: src/lib_json/json_value.cpp
 
- // //////////////////////////////////////////////////////////////////////
 
- // //////////////////////////////////////////////////////////////////////
 
- // Beginning of content of file: src/lib_json/json_writer.cpp
 
- // //////////////////////////////////////////////////////////////////////
 
- // Copyright 2011 Baptiste Lepilleur
 
- // Distributed under MIT license, or public domain if desired and
 
- // recognized in your jurisdiction.
 
- // See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
 
- #if !defined(JSON_IS_AMALGAMATION)
 
- #include <json/writer.h>
 
- #include "json_tool.h"
 
- #endif // if !defined(JSON_IS_AMALGAMATION)
 
- #include <iomanip>
 
- #include <memory>
 
- #include <sstream>
 
- #include <utility>
 
- #include <set>
 
- #include <cassert>
 
- #include <cstring>
 
- #include <cstdio>
 
- #if defined(_MSC_VER) && _MSC_VER >= 1200 && _MSC_VER < 1800 // Between VC++ 6.0 and VC++ 11.0
 
- #include <float.h>
 
- #define isfinite _finite
 
- #elif defined(__sun) && defined(__SVR4) //Solaris
 
- #if !defined(isfinite)
 
- #include <ieeefp.h>
 
- #define isfinite finite
 
- #endif
 
- #elif defined(_AIX)
 
- #if !defined(isfinite)
 
- #include <math.h>
 
- #define isfinite finite
 
- #endif
 
- #elif defined(__hpux)
 
- #if !defined(isfinite)
 
- #if defined(__ia64) && !defined(finite)
 
- #define isfinite(x) ((sizeof(x) == sizeof(float) ? \
 
-                      _Isfinitef(x) : _IsFinite(x)))
 
- #else
 
- #include <math.h>
 
- #define isfinite finite
 
- #endif
 
- #endif
 
- #else
 
- #include <cmath>
 
- #if !(defined(__QNXNTO__)) // QNX already defines isfinite
 
- #define isfinite std::isfinite
 
- #endif
 
- #endif
 
- #if defined(_MSC_VER)
 
- #if !defined(WINCE) && defined(__STDC_SECURE_LIB__) && _MSC_VER >= 1500 // VC++ 9.0 and above
 
- #define snprintf sprintf_s
 
- #elif _MSC_VER >= 1900 // VC++ 14.0 and above
 
- #define snprintf std::snprintf
 
- #else
 
- #define snprintf _snprintf
 
- #endif
 
- #elif defined(__ANDROID__) || defined(__QNXNTO__)
 
- #define snprintf snprintf
 
- #elif __cplusplus >= 201103L
 
- #define snprintf std::snprintf
 
- #endif
 
- #if defined(__BORLANDC__)  
 
- #include <float.h>
 
- #define isfinite _finite
 
- #define snprintf _snprintf
 
- #endif
 
- #if defined(_MSC_VER) && _MSC_VER >= 1400 // VC++ 8.0
 
- // Disable warning about strdup being deprecated.
 
- #pragma warning(disable : 4996)
 
- #endif
 
- namespace Json {
 
- #if __cplusplus >= 201103L || (defined(_CPPLIB_VER) && _CPPLIB_VER >= 520)
 
- typedef std::unique_ptr<StreamWriter> StreamWriterPtr;
 
- #else
 
- typedef std::auto_ptr<StreamWriter>   StreamWriterPtr;
 
- #endif
 
- static bool containsControlCharacter(const char* str) {
 
-   while (*str) {
 
-     if (isControlCharacter(*(str++)))
 
-       return true;
 
-   }
 
-   return false;
 
- }
 
- static bool containsControlCharacter0(const char* str, unsigned len) {
 
-   char const* end = str + len;
 
-   while (end != str) {
 
-     if (isControlCharacter(*str) || 0==*str)
 
-       return true;
 
-     ++str;
 
-   }
 
-   return false;
 
- }
 
- std::string valueToString(LargestInt value) {
 
-   UIntToStringBuffer buffer;
 
-   char* current = buffer + sizeof(buffer);
 
-   if (value == Value::minLargestInt) {
 
-     uintToString(LargestUInt(Value::maxLargestInt) + 1, current);
 
-     *--current = '-';
 
-   } else if (value < 0) {
 
-     uintToString(LargestUInt(-value), current);
 
-     *--current = '-';
 
-   } else {
 
-     uintToString(LargestUInt(value), current);
 
-   }
 
-   assert(current >= buffer);
 
-   return current;
 
- }
 
- std::string valueToString(LargestUInt value) {
 
-   UIntToStringBuffer buffer;
 
-   char* current = buffer + sizeof(buffer);
 
-   uintToString(value, current);
 
-   assert(current >= buffer);
 
-   return current;
 
- }
 
- #if defined(JSON_HAS_INT64)
 
- std::string valueToString(Int value) {
 
-   return valueToString(LargestInt(value));
 
- }
 
- std::string valueToString(UInt value) {
 
-   return valueToString(LargestUInt(value));
 
- }
 
- #endif // # if defined(JSON_HAS_INT64)
 
- std::string valueToString(double value, bool useSpecialFloats, unsigned int precision) {
 
-   // Allocate a buffer that is more than large enough to store the 16 digits of
 
-   // precision requested below.
 
-   char buffer[32];
 
-   int len = -1;
 
-   char formatString[6];
 
-   sprintf(formatString, "%%.%dg", precision);
 
-   // Print into the buffer. We need not request the alternative representation
 
-   // that always has a decimal point because JSON doesn't distingish the
 
-   // concepts of reals and integers.
 
-   if (isfinite(value)) {
 
-     len = snprintf(buffer, sizeof(buffer), formatString, value);
 
-   } else {
 
-     // IEEE standard states that NaN values will not compare to themselves
 
-     if (value != value) {
 
-       len = snprintf(buffer, sizeof(buffer), useSpecialFloats ? "NaN" : "null");
 
-     } else if (value < 0) {
 
-       len = snprintf(buffer, sizeof(buffer), useSpecialFloats ? "-Infinity" : "-1e+9999");
 
-     } else {
 
-       len = snprintf(buffer, sizeof(buffer), useSpecialFloats ? "Infinity" : "1e+9999");
 
-     }
 
-     // For those, we do not need to call fixNumLoc, but it is fast.
 
-   }
 
-   assert(len >= 0);
 
-   fixNumericLocale(buffer, buffer + len);
 
-   return buffer;
 
- }
 
- std::string valueToString(double value) { return valueToString(value, false, 17); }
 
- std::string valueToString(bool value) { return value ? "true" : "false"; }
 
- std::string valueToQuotedString(const char* value) {
 
-   if (value == NULL)
 
-     return "";
 
-   // 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.)
 
-   std::string::size_type 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;
 
- }
 
- // https://github.com/upcaste/upcaste/blob/master/src/upcore/src/cstring/strnpbrk.cpp
 
- static char const* strnpbrk(char const* s, char const* accept, size_t n) {
 
-   assert((s || !n) && accept);
 
-   char const* const end = s + n;
 
-   for (char const* cur = s; cur < end; ++cur) {
 
-     int const c = *cur;
 
-     for (char const* a = accept; *a; ++a) {
 
-       if (*a == c) {
 
-         return cur;
 
-       }
 
-     }
 
-   }
 
-   return NULL;
 
- }
 
- static std::string valueToQuotedStringN(const char* value, unsigned length) {
 
-   if (value == NULL)
 
-     return "";
 
-   // Not sure how to handle unicode...
 
-   if (strnpbrk(value, "\"\\\b\f\n\r\t", length) == NULL &&
 
-       !containsControlCharacter0(value, length))
 
-     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.)
 
-   std::string::size_type maxsize =
 
-       length * 2 + 3; // allescaped+quotes+NULL
 
-   std::string result;
 
-   result.reserve(maxsize); // to avoid lots of mallocs
 
-   result += "\"";
 
-   char const* end = value + length;
 
-   for (const char* c = value; c != end; ++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)) || (*c == 0)) {
 
-         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), dropNullPlaceholders_(false),
 
-       omitEndingLineFeed_(false) {}
 
- void FastWriter::enableYAMLCompatibility() { yamlCompatiblityEnabled_ = true; }
 
- void FastWriter::dropNullPlaceholders() { dropNullPlaceholders_ = true; }
 
- void FastWriter::omitEndingLineFeed() { omitEndingLineFeed_ = true; }
 
- std::string FastWriter::write(const Value& root) {
 
-   document_ = "";
 
-   writeValue(root);
 
-   if (!omitEndingLineFeed_)
 
-     document_ += "\n";
 
-   return document_;
 
- }
 
- void FastWriter::writeValue(const Value& value) {
 
-   switch (value.type()) {
 
-   case nullValue:
 
-     if (!dropNullPlaceholders_)
 
-       document_ += "null";
 
-     break;
 
-   case intValue:
 
-     document_ += valueToString(value.asLargestInt());
 
-     break;
 
-   case uintValue:
 
-     document_ += valueToString(value.asLargestUInt());
 
-     break;
 
-   case realValue:
 
-     document_ += valueToString(value.asDouble());
 
-     break;
 
-   case stringValue:
 
-   {
 
-     // Is NULL possible for value.string_?
 
-     char const* str;
 
-     char const* end;
 
-     bool ok = value.getString(&str, &end);
 
-     if (ok) document_ += valueToQuotedStringN(str, static_cast<unsigned>(end-str));
 
-     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_ += valueToQuotedStringN(name.data(), static_cast<unsigned>(name.length()));
 
-       document_ += yamlCompatiblityEnabled_ ? ": " : ":";
 
-       writeValue(value[name]);
 
-     }
 
-     document_ += '}';
 
-   } break;
 
-   }
 
- }
 
- // Class StyledWriter
 
- // //////////////////////////////////////////////////////////////////
 
- StyledWriter::StyledWriter()
 
-     : rightMargin_(74), indentSize_(3), addChildValues_() {}
 
- 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.asLargestInt()));
 
-     break;
 
-   case uintValue:
 
-     pushValue(valueToString(value.asLargestUInt()));
 
-     break;
 
-   case realValue:
 
-     pushValue(valueToString(value.asDouble()));
 
-     break;
 
-   case stringValue:
 
-   {
 
-     // Is NULL possible for value.string_?
 
-     char const* str;
 
-     char const* end;
 
-     bool ok = value.getString(&str, &end);
 
-     if (ok) pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end-str)));
 
-     else pushValue("");
 
-     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();
 
-       for (;;) {
 
-         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;
 
-       for (;;) {
 
-         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 = ((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; ++index) {
 
-       if (hasCommentForValue(value[index])) {
 
-         isMultiLine = true;
 
-       }
 
-       writeValue(value[index]);
 
-       lineLength += int(childValues_[index].length());
 
-     }
 
-     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_ += "\n";
 
-   writeIndent();
 
-   const std::string& comment = root.getComment(commentBefore);
 
-   std::string::const_iterator iter = comment.begin();
 
-   while (iter != comment.end()) {
 
-     document_ += *iter;
 
-     if (*iter == '\n' &&
 
-        (iter != comment.end() && *(iter + 1) == '/'))
 
-       writeIndent();
 
-     ++iter;
 
-   }
 
-   // Comments are stripped of trailing newlines, so add one here
 
-   document_ += "\n";
 
- }
 
- void StyledWriter::writeCommentAfterValueOnSameLine(const Value& root) {
 
-   if (root.hasComment(commentAfterOnSameLine))
 
-     document_ += " " + root.getComment(commentAfterOnSameLine);
 
-   if (root.hasComment(commentAfter)) {
 
-     document_ += "\n";
 
-     document_ += root.getComment(commentAfter);
 
-     document_ += "\n";
 
-   }
 
- }
 
- bool StyledWriter::hasCommentForValue(const Value& value) {
 
-   return value.hasComment(commentBefore) ||
 
-          value.hasComment(commentAfterOnSameLine) ||
 
-          value.hasComment(commentAfter);
 
- }
 
- // Class StyledStreamWriter
 
- // //////////////////////////////////////////////////////////////////
 
- StyledStreamWriter::StyledStreamWriter(std::string indentation)
 
-     : document_(NULL), rightMargin_(74), indentation_(indentation),
 
-       addChildValues_() {}
 
- void StyledStreamWriter::write(std::ostream& out, const Value& root) {
 
-   document_ = &out;
 
-   addChildValues_ = false;
 
-   indentString_ = "";
 
-   indented_ = true;
 
-   writeCommentBeforeValue(root);
 
-   if (!indented_) writeIndent();
 
-   indented_ = true;
 
-   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.asLargestInt()));
 
-     break;
 
-   case uintValue:
 
-     pushValue(valueToString(value.asLargestUInt()));
 
-     break;
 
-   case realValue:
 
-     pushValue(valueToString(value.asDouble()));
 
-     break;
 
-   case stringValue:
 
-   {
 
-     // Is NULL possible for value.string_?
 
-     char const* str;
 
-     char const* end;
 
-     bool ok = value.getString(&str, &end);
 
-     if (ok) pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end-str)));
 
-     else pushValue("");
 
-     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();
 
-       for (;;) {
 
-         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;
 
-       for (;;) {
 
-         const Value& childValue = value[index];
 
-         writeCommentBeforeValue(childValue);
 
-         if (hasChildValue)
 
-           writeWithIndent(childValues_[index]);
 
-         else {
 
-           if (!indented_) writeIndent();
 
-           indented_ = true;
 
-           writeValue(childValue);
 
-           indented_ = false;
 
-         }
 
-         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 = ((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; ++index) {
 
-       if (hasCommentForValue(value[index])) {
 
-         isMultiLine = true;
 
-       }
 
-       writeValue(value[index]);
 
-       lineLength += int(childValues_[index].length());
 
-     }
 
-     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() {
 
-   // blep intended this to look at the so-far-written string
 
-   // to determine whether we are already indented, but
 
-   // with a stream we cannot do that. So we rely on some saved state.
 
-   // The caller checks indented_.
 
-   *document_ << '\n' << indentString_;
 
- }
 
- void StyledStreamWriter::writeWithIndent(const std::string& value) {
 
-   if (!indented_) writeIndent();
 
-   *document_ << value;
 
-   indented_ = false;
 
- }
 
- 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;
 
-   if (!indented_) writeIndent();
 
-   const std::string& comment = root.getComment(commentBefore);
 
-   std::string::const_iterator iter = comment.begin();
 
-   while (iter != comment.end()) {
 
-     *document_ << *iter;
 
-     if (*iter == '\n' &&
 
-        (iter != comment.end() && *(iter + 1) == '/'))
 
-       // writeIndent();  // would include newline
 
-       *document_ << indentString_;
 
-     ++iter;
 
-   }
 
-   indented_ = false;
 
- }
 
- void StyledStreamWriter::writeCommentAfterValueOnSameLine(const Value& root) {
 
-   if (root.hasComment(commentAfterOnSameLine))
 
-     *document_ << ' ' << root.getComment(commentAfterOnSameLine);
 
-   if (root.hasComment(commentAfter)) {
 
-     writeIndent();
 
-     *document_ << root.getComment(commentAfter);
 
-   }
 
-   indented_ = false;
 
- }
 
- bool StyledStreamWriter::hasCommentForValue(const Value& value) {
 
-   return value.hasComment(commentBefore) ||
 
-          value.hasComment(commentAfterOnSameLine) ||
 
-          value.hasComment(commentAfter);
 
- }
 
- //////////////////////////
 
- // BuiltStyledStreamWriter
 
- /// Scoped enums are not available until C++11.
 
- struct CommentStyle {
 
-   /// Decide whether to write comments.
 
-   enum Enum {
 
-     None,  ///< Drop all comments.
 
-     Most,  ///< Recover odd behavior of previous versions (not implemented yet).
 
-     All  ///< Keep all comments.
 
-   };
 
- };
 
- struct BuiltStyledStreamWriter : public StreamWriter
 
- {
 
-   BuiltStyledStreamWriter(
 
-       std::string const& indentation,
 
-       CommentStyle::Enum cs,
 
-       std::string const& colonSymbol,
 
-       std::string const& nullSymbol,
 
-       std::string const& endingLineFeedSymbol,
 
-       bool useSpecialFloats,
 
-       unsigned int precision);
 
-   int write(Value const& root, std::ostream* sout) override;
 
- private:
 
-   void writeValue(Value const& value);
 
-   void writeArrayValue(Value const& value);
 
-   bool isMultineArray(Value const& value);
 
-   void pushValue(std::string const& value);
 
-   void writeIndent();
 
-   void writeWithIndent(std::string const& value);
 
-   void indent();
 
-   void unindent();
 
-   void writeCommentBeforeValue(Value const& root);
 
-   void writeCommentAfterValueOnSameLine(Value const& root);
 
-   static bool hasCommentForValue(const Value& value);
 
-   typedef std::vector<std::string> ChildValues;
 
-   ChildValues childValues_;
 
-   std::string indentString_;
 
-   int rightMargin_;
 
-   std::string indentation_;
 
-   CommentStyle::Enum cs_;
 
-   std::string colonSymbol_;
 
-   std::string nullSymbol_;
 
-   std::string endingLineFeedSymbol_;
 
-   bool addChildValues_ : 1;
 
-   bool indented_ : 1;
 
-   bool useSpecialFloats_ : 1;
 
-   unsigned int precision_;
 
- };
 
- BuiltStyledStreamWriter::BuiltStyledStreamWriter(
 
-       std::string const& indentation,
 
-       CommentStyle::Enum cs,
 
-       std::string const& colonSymbol,
 
-       std::string const& nullSymbol,
 
-       std::string const& endingLineFeedSymbol,
 
-       bool useSpecialFloats,
 
-       unsigned int precision)
 
-   : rightMargin_(74)
 
-   , indentation_(indentation)
 
-   , cs_(cs)
 
-   , colonSymbol_(colonSymbol)
 
-   , nullSymbol_(nullSymbol)
 
-   , endingLineFeedSymbol_(endingLineFeedSymbol)
 
-   , addChildValues_(false)
 
-   , indented_(false)
 
-   , useSpecialFloats_(useSpecialFloats)
 
-   , precision_(precision)
 
- {
 
- }
 
- int BuiltStyledStreamWriter::write(Value const& root, std::ostream* sout)
 
- {
 
-   sout_ = sout;
 
-   addChildValues_ = false;
 
-   indented_ = true;
 
-   indentString_ = "";
 
-   writeCommentBeforeValue(root);
 
-   if (!indented_) writeIndent();
 
-   indented_ = true;
 
-   writeValue(root);
 
-   writeCommentAfterValueOnSameLine(root);
 
-   *sout_ << endingLineFeedSymbol_;
 
-   sout_ = NULL;
 
-   return 0;
 
- }
 
- void BuiltStyledStreamWriter::writeValue(Value const& value) {
 
-   switch (value.type()) {
 
-   case nullValue:
 
-     pushValue(nullSymbol_);
 
-     break;
 
-   case intValue:
 
-     pushValue(valueToString(value.asLargestInt()));
 
-     break;
 
-   case uintValue:
 
-     pushValue(valueToString(value.asLargestUInt()));
 
-     break;
 
-   case realValue:
 
-     pushValue(valueToString(value.asDouble(), useSpecialFloats_, precision_));
 
-     break;
 
-   case stringValue:
 
-   {
 
-     // Is NULL is possible for value.string_?
 
-     char const* str;
 
-     char const* end;
 
-     bool ok = value.getString(&str, &end);
 
-     if (ok) pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end-str)));
 
-     else pushValue("");
 
-     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();
 
-       for (;;) {
 
-         std::string const& name = *it;
 
-         Value const& childValue = value[name];
 
-         writeCommentBeforeValue(childValue);
 
-         writeWithIndent(valueToQuotedStringN(name.data(), static_cast<unsigned>(name.length())));
 
-         *sout_ << colonSymbol_;
 
-         writeValue(childValue);
 
-         if (++it == members.end()) {
 
-           writeCommentAfterValueOnSameLine(childValue);
 
-           break;
 
-         }
 
-         *sout_ << ",";
 
-         writeCommentAfterValueOnSameLine(childValue);
 
-       }
 
-       unindent();
 
-       writeWithIndent("}");
 
-     }
 
-   } break;
 
-   }
 
- }
 
- void BuiltStyledStreamWriter::writeArrayValue(Value const& value) {
 
-   unsigned size = value.size();
 
-   if (size == 0)
 
-     pushValue("[]");
 
-   else {
 
-     bool isMultiLine = (cs_ == CommentStyle::All) || isMultineArray(value);
 
-     if (isMultiLine) {
 
-       writeWithIndent("[");
 
-       indent();
 
-       bool hasChildValue = !childValues_.empty();
 
-       unsigned index = 0;
 
-       for (;;) {
 
-         Value const& childValue = value[index];
 
-         writeCommentBeforeValue(childValue);
 
-         if (hasChildValue)
 
-           writeWithIndent(childValues_[index]);
 
-         else {
 
-           if (!indented_) writeIndent();
 
-           indented_ = true;
 
-           writeValue(childValue);
 
-           indented_ = false;
 
-         }
 
-         if (++index == size) {
 
-           writeCommentAfterValueOnSameLine(childValue);
 
-           break;
 
-         }
 
-         *sout_ << ",";
 
-         writeCommentAfterValueOnSameLine(childValue);
 
-       }
 
-       unindent();
 
-       writeWithIndent("]");
 
-     } else // output on a single line
 
-     {
 
-       assert(childValues_.size() == size);
 
-       *sout_ << "[";
 
-       if (!indentation_.empty()) *sout_ << " ";
 
-       for (unsigned index = 0; index < size; ++index) {
 
-         if (index > 0)
 
-           *sout_ << ", ";
 
-         *sout_ << childValues_[index];
 
-       }
 
-       if (!indentation_.empty()) *sout_ << " ";
 
-       *sout_ << "]";
 
-     }
 
-   }
 
- }
 
- bool BuiltStyledStreamWriter::isMultineArray(Value const& value) {
 
-   int size = value.size();
 
-   bool isMultiLine = size * 3 >= rightMargin_;
 
-   childValues_.clear();
 
-   for (int index = 0; index < size && !isMultiLine; ++index) {
 
-     Value const& childValue = value[index];
 
-     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; ++index) {
 
-       if (hasCommentForValue(value[index])) {
 
-         isMultiLine = true;
 
-       }
 
-       writeValue(value[index]);
 
-       lineLength += int(childValues_[index].length());
 
-     }
 
-     addChildValues_ = false;
 
-     isMultiLine = isMultiLine || lineLength >= rightMargin_;
 
-   }
 
-   return isMultiLine;
 
- }
 
- void BuiltStyledStreamWriter::pushValue(std::string const& value) {
 
-   if (addChildValues_)
 
-     childValues_.push_back(value);
 
-   else
 
-     *sout_ << value;
 
- }
 
- void BuiltStyledStreamWriter::writeIndent() {
 
-   // blep intended this to look at the so-far-written string
 
-   // to determine whether we are already indented, but
 
-   // with a stream we cannot do that. So we rely on some saved state.
 
-   // The caller checks indented_.
 
-   if (!indentation_.empty()) {
 
-     // In this case, drop newlines too.
 
-     *sout_ << '\n' << indentString_;
 
-   }
 
- }
 
- void BuiltStyledStreamWriter::writeWithIndent(std::string const& value) {
 
-   if (!indented_) writeIndent();
 
-   *sout_ << value;
 
-   indented_ = false;
 
- }
 
- void BuiltStyledStreamWriter::indent() { indentString_ += indentation_; }
 
- void BuiltStyledStreamWriter::unindent() {
 
-   assert(indentString_.size() >= indentation_.size());
 
-   indentString_.resize(indentString_.size() - indentation_.size());
 
- }
 
- void BuiltStyledStreamWriter::writeCommentBeforeValue(Value const& root) {
 
-   if (cs_ == CommentStyle::None) return;
 
-   if (!root.hasComment(commentBefore))
 
-     return;
 
-   if (!indented_) writeIndent();
 
-   const std::string& comment = root.getComment(commentBefore);
 
-   std::string::const_iterator iter = comment.begin();
 
-   while (iter != comment.end()) {
 
-     *sout_ << *iter;
 
-     if (*iter == '\n' &&
 
-        (iter != comment.end() && *(iter + 1) == '/'))
 
-       // writeIndent();  // would write extra newline
 
-       *sout_ << indentString_;
 
-     ++iter;
 
-   }
 
-   indented_ = false;
 
- }
 
- void BuiltStyledStreamWriter::writeCommentAfterValueOnSameLine(Value const& root) {
 
-   if (cs_ == CommentStyle::None) return;
 
-   if (root.hasComment(commentAfterOnSameLine))
 
-     *sout_ << " " + root.getComment(commentAfterOnSameLine);
 
-   if (root.hasComment(commentAfter)) {
 
-     writeIndent();
 
-     *sout_ << root.getComment(commentAfter);
 
-   }
 
- }
 
- // static
 
- bool BuiltStyledStreamWriter::hasCommentForValue(const Value& value) {
 
-   return value.hasComment(commentBefore) ||
 
-          value.hasComment(commentAfterOnSameLine) ||
 
-          value.hasComment(commentAfter);
 
- }
 
- ///////////////
 
- // StreamWriter
 
- StreamWriter::StreamWriter()
 
-     : sout_(NULL)
 
- {
 
- }
 
- StreamWriter::~StreamWriter()
 
- {
 
- }
 
- StreamWriter::Factory::~Factory()
 
- {}
 
- StreamWriterBuilder::StreamWriterBuilder()
 
- {
 
-   setDefaults(&settings_);
 
- }
 
- StreamWriterBuilder::~StreamWriterBuilder()
 
- {}
 
- StreamWriter* StreamWriterBuilder::newStreamWriter() const
 
- {
 
-   std::string indentation = settings_["indentation"].asString();
 
-   std::string cs_str = settings_["commentStyle"].asString();
 
-   bool eyc = settings_["enableYAMLCompatibility"].asBool();
 
-   bool dnp = settings_["dropNullPlaceholders"].asBool();
 
-   bool usf = settings_["useSpecialFloats"].asBool(); 
 
-   unsigned int pre = settings_["precision"].asUInt();
 
-   CommentStyle::Enum cs = CommentStyle::All;
 
-   if (cs_str == "All") {
 
-     cs = CommentStyle::All;
 
-   } else if (cs_str == "None") {
 
-     cs = CommentStyle::None;
 
-   } else {
 
-     throwRuntimeError("commentStyle must be 'All' or 'None'");
 
-   }
 
-   std::string colonSymbol = " : ";
 
-   if (eyc) {
 
-     colonSymbol = ": ";
 
-   } else if (indentation.empty()) {
 
-     colonSymbol = ":";
 
-   }
 
-   std::string nullSymbol = "null";
 
-   if (dnp) {
 
-     nullSymbol = "";
 
-   }
 
-   if (pre > 17) pre = 17;
 
-   std::string endingLineFeedSymbol = "";
 
-   return new BuiltStyledStreamWriter(
 
-       indentation, cs,
 
-       colonSymbol, nullSymbol, endingLineFeedSymbol, usf, pre);
 
- }
 
- static void getValidWriterKeys(std::set<std::string>* valid_keys)
 
- {
 
-   valid_keys->clear();
 
-   valid_keys->insert("indentation");
 
-   valid_keys->insert("commentStyle");
 
-   valid_keys->insert("enableYAMLCompatibility");
 
-   valid_keys->insert("dropNullPlaceholders");
 
-   valid_keys->insert("useSpecialFloats");
 
-   valid_keys->insert("precision");
 
- }
 
- bool StreamWriterBuilder::validate(Json::Value* invalid) const
 
- {
 
-   Json::Value my_invalid;
 
-   if (!invalid) invalid = &my_invalid;  // so we do not need to test for NULL
 
-   Json::Value& inv = *invalid;
 
-   std::set<std::string> valid_keys;
 
-   getValidWriterKeys(&valid_keys);
 
-   Value::Members keys = settings_.getMemberNames();
 
-   size_t n = keys.size();
 
-   for (size_t i = 0; i < n; ++i) {
 
-     std::string const& key = keys[i];
 
-     if (valid_keys.find(key) == valid_keys.end()) {
 
-       inv[key] = settings_[key];
 
-     }
 
-   }
 
-   return 0u == inv.size();
 
- }
 
- Value& StreamWriterBuilder::operator[](std::string key)
 
- {
 
-   return settings_[key];
 
- }
 
- // static
 
- void StreamWriterBuilder::setDefaults(Json::Value* settings)
 
- {
 
-   //! [StreamWriterBuilderDefaults]
 
-   (*settings)["commentStyle"] = "All";
 
-   (*settings)["indentation"] = "\t";
 
-   (*settings)["enableYAMLCompatibility"] = false;
 
-   (*settings)["dropNullPlaceholders"] = false;
 
-   (*settings)["useSpecialFloats"] = false;
 
-   (*settings)["precision"] = 17;
 
-   //! [StreamWriterBuilderDefaults]
 
- }
 
- std::string writeString(StreamWriter::Factory const& builder, Value const& root) {
 
-   std::ostringstream sout;
 
-   StreamWriterPtr const writer(builder.newStreamWriter());
 
-   writer->write(root, &sout);
 
-   return sout.str();
 
- }
 
- std::ostream& operator<<(std::ostream& sout, Value const& root) {
 
-   StreamWriterBuilder builder;
 
-   StreamWriterPtr const writer(builder.newStreamWriter());
 
-   writer->write(root, &sout);
 
-   return sout;
 
- }
 
- } // namespace Json
 
- // //////////////////////////////////////////////////////////////////////
 
- // End of content of file: src/lib_json/json_writer.cpp
 
- // //////////////////////////////////////////////////////////////////////
 
 
  |