forked from lix-project/lix
78 lines
2.1 KiB
C++
78 lines
2.1 KiB
C++
#include "primops.hh"
|
|
#include "eval-inline.hh"
|
|
|
|
#include "cpptoml/cpptoml.h"
|
|
|
|
namespace nix {
|
|
|
|
static void prim_fromTOML(EvalState & state, const Pos & pos, Value * * args, Value & v)
|
|
{
|
|
using namespace cpptoml;
|
|
|
|
auto toml = state.forceStringNoCtx(*args[0], pos);
|
|
|
|
std::istringstream tomlStream(toml);
|
|
|
|
std::function<void(Value &, std::shared_ptr<base>)> visit;
|
|
|
|
visit = [&](Value & v, std::shared_ptr<base> t) {
|
|
|
|
if (auto t2 = t->as_table()) {
|
|
|
|
size_t size = 0;
|
|
for (auto & i : *t2) size++;
|
|
|
|
state.mkAttrs(v, size);
|
|
|
|
for (auto & i : *t2) {
|
|
auto & v2 = *state.allocAttr(v, state.symbols.create(i.first));
|
|
|
|
if (auto i2 = i.second->as_table_array()) {
|
|
size_t size2 = i2->get().size();
|
|
state.mkList(v2, size2);
|
|
for (size_t j = 0; j < size2; ++j)
|
|
visit(*(v2.listElems()[j] = state.allocValue()), i2->get()[j]);
|
|
}
|
|
else
|
|
visit(v2, i.second);
|
|
}
|
|
|
|
v.attrs->sort();
|
|
}
|
|
|
|
else if (auto t2 = t->as_array()) {
|
|
size_t size = t2->get().size();
|
|
|
|
state.mkList(v, size);
|
|
|
|
for (size_t i = 0; i < size; ++i)
|
|
visit(*(v.listElems()[i] = state.allocValue()), t2->get()[i]);
|
|
}
|
|
|
|
else if (t->is_value()) {
|
|
if (auto val = t->as<NixInt>())
|
|
mkInt(v, val->get());
|
|
else if (auto val = t->as<NixFloat>())
|
|
mkFloat(v, val->get());
|
|
else if (auto val = t->as<bool>())
|
|
mkBool(v, val->get());
|
|
else if (auto val = t->as<std::string>())
|
|
mkString(v, val->get());
|
|
else
|
|
throw EvalError("unsupported value type in TOML");
|
|
}
|
|
|
|
else abort();
|
|
};
|
|
|
|
try {
|
|
visit(v, parser(tomlStream).parse());
|
|
} catch (std::runtime_error & e) {
|
|
throw EvalError("while parsing a TOML string at %s: %s", pos, e.what());
|
|
}
|
|
}
|
|
|
|
static RegisterPrimOp r("fromTOML", 1, prim_fromTOML);
|
|
|
|
}
|