google/jsonnet

add manifestToml(v) function

Closed this issue ยท 9 comments

It should be fairly easy to write a TOML output function; the syntax is simple and it maps unambiguously to json objects.

Is there a formal grammar of it anywhere?

Oddly enough I don't think they have a finalized grammar, but there is an ABNF description in toml-lang/toml#236. I suppose the canonical specification is in this document: https://github.com/toml-lang/toml/blob/master/versions/en/toml-v0.4.0.md, which links to a ton of implementations and a validation suite.

I've taken a stab at a basic version of this: https://gist.github.com/benley/4fe6d2d354c3ac004de82ab51499eccb

It probably still has some weird edge cases, but it works for what I've tried so far.

Are there any plans to include the gist into the stdlib?

@flyingmutant We're happy to accept PRs adding this functionality.

oh hey, I had forgotten all about this. I have some free time and will try to get a PR put together for it in the next couple of days.

Hey @benley ... so it's been a few days :)

Do you need some help with this? I'd love to see this merged.

I've added support for table arrays. And also fixed escapeKeyToml. Will try submitting a PR in the next few days.

But here is the code that worked for me:

// Render any jsonnet value (except functions) to TOML format.

local
  inlineTable(body) =
    "{" +
    std.join(", ", ["%s = %s" % [escapeKeyToml(k), renderBody(body[k])] for k in std.objectFields(body)]) +
    "}",

  renderArray(body) =
    local
      types = std.set([std.type(x) for x in body]),
      // If any of the values are non-integer, we need to force all values to be rendered as floats.
      forceFloat = std.foldl(function(result, item) result || (std.floor(item) != item), body, false);
    if std.length(types) > 1
    then error "TOML Arrays must be uniform-type. Multiple types found: %s" % std.join(", ", types)
    else "[" + std.join(", ", [renderBody(x, forceFloat=forceFloat) for x in body]) + "]",

  renderBody(body, forceFloat=false) =
    if std.type(body) == "object"               then inlineTable(body) else
    if std.type(body) == "array"                then renderArray(body) else
    if std.type(body) == "number" && forceFloat then "%f" % body else
    if std.type(body) == "number"               then body else
    if std.type(body) == "string"               then escapeStringToml(body) else
    if std.type(body) == "boolean"              then body else
      error "unsupported value for toml: got %s" % std.type(body),

  renderItem(k, v) =
    ["%s = %s" % [escapeKeyToml(k), renderBody(v)]],

  renderSection(path, v) =
    ["[%s]" % std.join(".", std.map(escapeKeyToml, path))] + topTable(v, path=path),

  renderSectionArray(path, v) =
    local lines = std.flattenArrays([
      (["[[%s]]" % std.join(".", std.map(escapeKeyToml, path))] + topTable(s, path=path) + [""])
      for s in v
    ]);
    std.slice(lines, 0, std.length(lines) - 1, 1),

  topTable(body, path=[]) =
    local
      nonObjects = std.flattenArrays([renderItem(k, body[k]) for k in std.objectFields(body) if std.type(body[k]) != "object" && (std.type(body[k]) != "array" || std.type(body[k][0]) != "object")]),
      objects = std.flattenArrays([renderSection(path + [k], body[k]) for k in std.objectFields(body) if std.type(body[k]) == "object"]),
      hasBoth = std.length(nonObjects) > 0 && std.length(objects) > 0,
      objectArrays = std.flattenArrays([renderSectionArray(path + [k], body[k]) for k in std.objectFields(body) if std.type(body[k]) == "array" && std.type(body[k][0]) == "object"]);
    nonObjects
    + (if hasBoth then [""] else [])
    + objects
    + (if std.length(objectArrays) > 0 && (std.length(nonObjects) > 0 || std.length(objects) > 0) then [""] else [])
    + objectArrays,

  escapeKeyToml(str) =
    local bare_allowed = std.set(std.stringChars("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_-"));
    if std.setUnion(std.set(std.stringChars(str)), bare_allowed) == bare_allowed then str else "%s" % escapeStringToml(str),

  // JSON string encoding rules are a superset of TOML's - ie. all valid JSON strings are valid TOML strings
  // but not vice versa (specifically, TOML only requires escaping control chars U+000, U+001f and U+007f,
  // whereas JSON does not allow any control chars). Allowed escapes are the same.
  escapeStringToml = std.escapeStringJson;

function(body)
  if std.type(body) != "object"
  then error "TOML body must be an object. Got %s" % std.type(body)
  else std.lines(topTable(body))

agh, I'm sorry for dropping the ball on this. I was unemployed for a year and was finding it hard to stay motivated on projects like these. @anyname2, please feel free to take over and finish this if you're so inclined.