lix/src/libexpr/value-to-json.cc
Shea Levy 320659b0cd Allow external code using libnixexpr to add types
Code that links to libnixexpr (e.g. plugins loaded with importNative, or
nix-exec) may want to provide custom value types and operations on
values of those types. For example, nix-exec is currently using sets
where a custom IO value type would be more appropriate. This commit
provides a generic hook for such types in the form of tExternal and the
ExternalBase virtual class, which contains all functions necessary for
libnixexpr's type-polymorphic functions (e.g. `showType`) to be
implemented.
2014-12-02 10:27:04 -05:00

100 lines
2.6 KiB
C++

#include "value-to-json.hh"
#include "eval-inline.hh"
#include "util.hh"
#include <cstdlib>
#include <iomanip>
namespace nix {
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 if (*i >= 0 && *i < 32)
str << "\\u" << std::setfill('0') << std::setw(4) << std::hex << (uint16_t) *i << std::dec;
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()) {
JSONObject json(str);
StringSet names;
foreach (Bindings::iterator, i, *v.attrs)
names.insert(i->name);
foreach (StringSet::iterator, i, names) {
Attr & a(*v.attrs->find(state.symbols.create(*i)));
json.attr(*i);
printValueAsJSON(state, strict, *a.value, str, context);
}
} else
printValueAsJSON(state, strict, *i->value, str, context);
break;
}
case tList: {
JSONList json(str);
for (unsigned int n = 0; n < v.list.length; ++n) {
json.elem();
printValueAsJSON(state, strict, *v.list.elems[n], str, context);
}
break;
}
case tExternal:
v.external->printValueAsJSON(state, strict, str, context);
break;
default:
throw TypeError(format("cannot convert %1% to JSON") % showType(v));
}
}
void ExternalValueBase::printValueAsJSON(EvalState & state, bool strict,
std::ostream & str, PathSet & context)
{
throw TypeError(format("cannot convert %1% to JSON") % showType());
}
}