Skip to content

Commit

Permalink
Add better error handling for format
Browse files Browse the repository at this point in the history
  • Loading branch information
stephenamar-db committed Nov 5, 2024
1 parent 6939b1a commit b317a3c
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 9 deletions.
36 changes: 27 additions & 9 deletions sjsonnet/src/sjsonnet/Format.scala
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,9 @@ object Format{
val cooked0 = formatted.conversion match{
case '%' => widenRaw(formatted, "%")
case _ =>

if (values.isInstanceOf[Val.Arr] && i >= values.cast[Val.Arr].length) {
Error.fail("Too few values to format: %d, expected %d".format(i + 1, values.cast[Val.Arr].length))
}
val raw = formatted.label match{
case None => values.cast[Val.Arr].force(i)
case Some(key) =>
Expand All @@ -116,9 +118,12 @@ object Format{
}
i += 1
value match{
case ujson.Str(s) => widenRaw(formatted, s)
case ujson.Str(s) =>
if (formatted.conversion != 's' && formatted.conversion != 'c')
Error.fail("Format required a number at %d, got string".format(i))
widenRaw(formatted, s)
case ujson.Num(s) =>
formatted.conversion match{
formatted.conversion match {
case 'd' | 'i' | 'u' => formatInteger(formatted, s)
case 'o' => formatOctal(formatted, s)
case 'x' => formatHexadecimal(formatted, s)
Expand All @@ -132,20 +137,33 @@ object Format{
case 's' =>
if (s.toLong == s) widenRaw(formatted, s.toLong.toString)
else widenRaw(formatted, s.toString)
case _ => Error.fail("Format required a %s at %d, got string".format(raw.prettyName, i))
}
case ujson.Bool(s) =>
formatted.conversion match {
case 'd' | 'i' | 'u' => formatInteger(formatted, s.compareTo(false))
case 'o' => formatOctal(formatted, s.compareTo(false))
case 'x' => formatHexadecimal(formatted, s.compareTo(false))
case 'X' => formatHexadecimal(formatted, s.compareTo(false)).toUpperCase
case 'e' => formatExponent(formatted, s.compareTo(false)).toLowerCase
case 'E' => formatExponent(formatted, s.compareTo(false))
case 'f' | 'F' => formatFloat(formatted, s.compareTo(false))
case 'g' => formatGeneric(formatted, s.compareTo(false)).toLowerCase
case 'G' => formatGeneric(formatted, s.compareTo(false))
case 'c' => widenRaw(formatted, Character.forDigit(s.compareTo(false), 10).toString)
case 's' => widenRaw(formatted, s.toString)
case _ => Error.fail("Format required a %s at %d, got string".format(raw.prettyName, i))
}
case ujson.True => widenRaw(formatted, "true")
case ujson.False => widenRaw(formatted, "false")
case v => widenRaw(formatted, v.toString)
}

}

output.append(cooked0)
output.append(literal)


}

if (values.isInstanceOf[Val.Arr] && i < values.cast[Val.Arr].length) {
Error.fail("Too many values to format: %d, expected %d".format(values.cast[Val.Arr].length, i))
}
output.toString()
}

Expand Down
13 changes: 13 additions & 0 deletions sjsonnet/test/src/sjsonnet/FormatTests.scala
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,15 @@ object FormatTests extends TestSuite{
assert(formatted == expected)
}

def checkErr(fmt: String, jsonStr: String, expectedErr: String) = {
try {
check(fmt, jsonStr, "")
} catch {
case e: Error =>
assert(e.getMessage == expectedErr)
}
}

def tests = Tests{
test("hash"){
// #
Expand Down Expand Up @@ -290,6 +299,10 @@ object FormatTests extends TestSuite{

// apparently you can pass in positional parameters to named interpolations
check("XXX%(ignored_lols)sXXX %s", """[1.1, 2]""", "XXX1.1XXX 2")

checkErr("%s %s", """["foo"]""", "Too few values to format: 2, expected 1")
checkErr("%s %s", """["foo", "bar", "baz"]""", "Too many values to format: 3, expected 2")
checkErr("%d", """["foo"]""", "Format required a number at 1, got string")
}
}
}

0 comments on commit b317a3c

Please sign in to comment.