982709199f
2740c4f Merge branch '2015_11_escape_plan' into bitcoin 7482163 Add new testcase to Makefile.am 46098ee Version 1.0.1. ccf3575 parser: Ensure multiple values cannot follow each other eb6cd64 Omit Obj/Arr open token from jsonTokenIsValue() test bfef9e2 Makefile.am: list recently added test data, fail{35,36}.json 3e319f3 parser: Tighten array, object syntax checks. c74185c parser: transform C++ variables into bitmask f2568bc Prefer C++ STL vector .at() for accessing object values. 8eafa26 travis: run parallel 'make distcheck' fd448da test: Improve tester diagnostics. Add failing test case from #15 2158205 Use internal, locale-independent isspace(), isdigit() implementations. 2ab9ad4 travis: Make 'make distcheck' for more comprehensive checks. 3339191 Escape all control characters git-subtree-dir: src/univalue git-subtree-split: 2740c4f71242086a7eb3dc32f812546ba9fad913
126 lines
2.9 KiB
C++
126 lines
2.9 KiB
C++
// Copyright 2014 BitPay Inc.
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include <iomanip>
|
|
#include <sstream>
|
|
#include <stdio.h>
|
|
#include "univalue.h"
|
|
#include "univalue_escapes.h"
|
|
|
|
// TODO: Using UTF8
|
|
|
|
using namespace std;
|
|
|
|
static string json_escape(const string& inS)
|
|
{
|
|
string outS;
|
|
outS.reserve(inS.size() * 2);
|
|
|
|
for (unsigned int i = 0; i < inS.size(); i++) {
|
|
unsigned char ch = inS[i];
|
|
const char *escStr = escapes[ch];
|
|
|
|
if (escStr)
|
|
outS += escStr;
|
|
|
|
else if (ch < 0x80)
|
|
outS += ch;
|
|
|
|
else { // TODO handle UTF-8 properly
|
|
char tmpesc[16];
|
|
sprintf(tmpesc, "\\u%04x", ch);
|
|
outS += tmpesc;
|
|
}
|
|
}
|
|
|
|
return outS;
|
|
}
|
|
|
|
string UniValue::write(unsigned int prettyIndent,
|
|
unsigned int indentLevel) const
|
|
{
|
|
string s;
|
|
s.reserve(1024);
|
|
|
|
unsigned int modIndent = indentLevel;
|
|
if (modIndent == 0)
|
|
modIndent = 1;
|
|
|
|
switch (typ) {
|
|
case VNULL:
|
|
s += "null";
|
|
break;
|
|
case VOBJ:
|
|
writeObject(prettyIndent, modIndent, s);
|
|
break;
|
|
case VARR:
|
|
writeArray(prettyIndent, modIndent, s);
|
|
break;
|
|
case VSTR:
|
|
s += "\"" + json_escape(val) + "\"";
|
|
break;
|
|
case VNUM:
|
|
s += val;
|
|
break;
|
|
case VBOOL:
|
|
s += (val == "1" ? "true" : "false");
|
|
break;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
static void indentStr(unsigned int prettyIndent, unsigned int indentLevel, string& s)
|
|
{
|
|
s.append(prettyIndent * indentLevel, ' ');
|
|
}
|
|
|
|
void UniValue::writeArray(unsigned int prettyIndent, unsigned int indentLevel, string& s) const
|
|
{
|
|
s += "[";
|
|
if (prettyIndent)
|
|
s += "\n";
|
|
|
|
for (unsigned int i = 0; i < values.size(); i++) {
|
|
if (prettyIndent)
|
|
indentStr(prettyIndent, indentLevel, s);
|
|
s += values[i].write(prettyIndent, indentLevel + 1);
|
|
if (i != (values.size() - 1)) {
|
|
s += ",";
|
|
if (prettyIndent)
|
|
s += " ";
|
|
}
|
|
if (prettyIndent)
|
|
s += "\n";
|
|
}
|
|
|
|
if (prettyIndent)
|
|
indentStr(prettyIndent, indentLevel - 1, s);
|
|
s += "]";
|
|
}
|
|
|
|
void UniValue::writeObject(unsigned int prettyIndent, unsigned int indentLevel, string& s) const
|
|
{
|
|
s += "{";
|
|
if (prettyIndent)
|
|
s += "\n";
|
|
|
|
for (unsigned int i = 0; i < keys.size(); i++) {
|
|
if (prettyIndent)
|
|
indentStr(prettyIndent, indentLevel, s);
|
|
s += "\"" + json_escape(keys[i]) + "\":";
|
|
if (prettyIndent)
|
|
s += " ";
|
|
s += values.at(i).write(prettyIndent, indentLevel + 1);
|
|
if (i != (values.size() - 1))
|
|
s += ",";
|
|
if (prettyIndent)
|
|
s += "\n";
|
|
}
|
|
|
|
if (prettyIndent)
|
|
indentStr(prettyIndent, indentLevel - 1, s);
|
|
s += "}";
|
|
}
|
|
|