]> granicus.if.org Git - jq/commitdiff
String -> number conversions with "tonumber".
authorStephen Dolan <mu@netsoc.tcd.ie>
Mon, 10 Sep 2012 16:08:13 +0000 (17:08 +0100)
committerStephen Dolan <mu@netsoc.tcd.ie>
Mon, 10 Sep 2012 16:08:13 +0000 (17:08 +0100)
c/builtin.c
c/testdata

index b65bb09b3c589cb7bdcfc40eceecb0f37f37f94b..7943675d43f3f6b5799d4b109ee6672fa884c498 100644 (file)
@@ -100,6 +100,30 @@ static void f_divide(jv input[], jv output[]) {
   }  
 }
 
+static void f_tonumber(jv input[], jv output[]) {
+  if (jv_get_kind(input[0]) == JV_KIND_NUMBER) {
+    output[0] = input[0];
+  } else if (jv_get_kind(input[0]) == JV_KIND_STRING) {
+    jv parsed = jv_parse(jv_string_value(input[0]));
+    if (!jv_is_valid(parsed)) {
+      jv_free(input[0]);
+      output[0] = parsed;
+    } else if (jv_get_kind(parsed) != JV_KIND_NUMBER) {
+      output[0] = jv_invalid_with_msg(jv_string_fmt("'%s' is not a number",
+                                                    jv_string_value(input[0])));
+      jv_free(input[0]);
+    } else {
+      jv_free(input[0]);
+      output[0] = parsed;
+    }
+  } else {
+    output[0] = jv_invalid_with_msg(jv_string_fmt("Cannot parse %s as a number",
+                                                  jv_kind_name(jv_get_kind(input[0]))));
+    jv_free(input[0]);
+    return;
+  }
+}
+
 struct cfunction function_list[] = {
   {f_true, "true", CALL_BUILTIN_1_1},
   {f_false, "false", CALL_BUILTIN_1_1},
@@ -108,5 +132,6 @@ struct cfunction function_list[] = {
   {f_minus, "_minus", CALL_BUILTIN_3_1},
   {f_multiply, "_multiply", CALL_BUILTIN_3_1},
   {f_divide, "_divide", CALL_BUILTIN_3_1},
+  {f_tonumber, "tonumber", CALL_BUILTIN_1_1},
 };
 struct symbol_table builtins = {function_list, sizeof(function_list)/sizeof(function_list[0])};
index eb9f0c5faef72a467da30cc276fd94950a814c10..45140242749921d53e8fa04015e13ddc00ff2395 100644 (file)
@@ -183,6 +183,10 @@ null
 null
 10
 
+1 + tonumber + ("10" | tonumber)
+4
+15
+
 #
 # User-defined functions
 # Oh god.