From 77c13cdf566ffedc70d8860571afae8a6d43b552 Mon Sep 17 00:00:00 2001 From: Eelco Dolstra Date: Tue, 19 Nov 2013 00:03:11 +0100 Subject: Add a toJSON primop --- src/libexpr/value-to-json.cc | 93 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 93 insertions(+) create mode 100644 src/libexpr/value-to-json.cc (limited to 'src/libexpr/value-to-json.cc') diff --git a/src/libexpr/value-to-json.cc b/src/libexpr/value-to-json.cc new file mode 100644 index 000000000000..671a3c119741 --- /dev/null +++ b/src/libexpr/value-to-json.cc @@ -0,0 +1,93 @@ +#include "value-to-xml.hh" +#include "xml-writer.hh" +#include "eval-inline.hh" +#include "util.hh" + +#include + + +namespace nix { + + +static void escapeJSON(std::ostream & str, const string & s) +{ + str << "\""; + foreach (string::const_iterator, i, s) + if (*i == '\"' || *i == '\\') str << "\\" << *i; + else if (*i == '\n') str << "\\n"; + else if (*i == '\r') str << "\\r"; + else if (*i == '\t') str << "\\t"; + else str << *i; + str << "\""; +} + + +void printValueAsJSON(EvalState & state, bool strict, + Value & v, std::ostream & str, PathSet & context) +{ + checkInterrupt(); + + if (strict) state.forceValue(v); + + switch (v.type) { + + case tInt: + str << v.integer; + break; + + case tBool: + str << (v.boolean ? "true" : "false"); + break; + + case tString: + copyContext(v, context); + escapeJSON(str, v.string.s); + break; + + case tPath: + escapeJSON(str, state.copyPathToStore(context, v.path)); + break; + + case tNull: + str << "null"; + break; + + case tAttrs: { + Bindings::iterator i = v.attrs->find(state.sOutPath); + if (i == v.attrs->end()) { + str << "{"; + StringSet names; + foreach (Bindings::iterator, i, *v.attrs) + names.insert(i->name); + bool first = true; + foreach (StringSet::iterator, i, names) { + if (!first) str << ","; else first = false; + Attr & a(*v.attrs->find(state.symbols.create(*i))); + escapeJSON(str, *i); + str << ":"; + printValueAsJSON(state, strict, *a.value, str, context); + } + str << "}"; + } else + printValueAsJSON(state, strict, *i->value, str, context); + break; + } + + case tList: { + str << "["; + bool first = true; + for (unsigned int n = 0; n < v.list.length; ++n) { + if (!first) str << ","; else first = false; + printValueAsJSON(state, strict, *v.list.elems[n], str, context); + } + str << "]"; + break; + } + + default: + throw TypeError(format("cannot convert %1% to JSON") % showType(v)); + } +} + + +} -- cgit 1.4.1