summaryrefslogtreecommitdiffstats
path: root/src/jv_print.c
diff options
context:
space:
mode:
authorDavid Tolnay <dtolnay@gmail.com>2015-08-23 20:36:11 -0700
committerDavid Tolnay <dtolnay@gmail.com>2015-08-23 20:36:11 -0700
commit0c93eb3379241dc4775718a9d39f54a6c4de20d6 (patch)
tree67bb5510adb707d54c6f72b51b0718578a2caf5c /src/jv_print.c
parent891f28ef5e406a8d2156ad88d0244ab03fe490eb (diff)
Move source files to src/
Diffstat (limited to 'src/jv_print.c')
-rw-r--r--src/jv_print.c348
1 files changed, 348 insertions, 0 deletions
diff --git a/src/jv_print.c b/src/jv_print.c
new file mode 100644
index 00000000..5f4f234b
--- /dev/null
+++ b/src/jv_print.c
@@ -0,0 +1,348 @@
+#include <assert.h>
+#include <stdio.h>
+#include <float.h>
+#include <string.h>
+
+#ifdef WIN32
+#include <windows.h>
+#include <io.h>
+#include <fileapi.h>
+#endif
+
+#include "jv.h"
+#include "jv_dtoa.h"
+#include "jv_unicode.h"
+
+#define ESC "\033"
+#define COL(c) (ESC "[" c "m")
+#define COLRESET (ESC "[0m")
+
+// Color table. See https://en.wikipedia.org/wiki/ANSI_escape_code#Colors
+// for how to choose these.
+static const jv_kind color_kinds[] =
+ {JV_KIND_NULL, JV_KIND_FALSE, JV_KIND_TRUE, JV_KIND_NUMBER,
+ JV_KIND_STRING, JV_KIND_ARRAY, JV_KIND_OBJECT};
+static const char* const colors[] =
+ {COL("1;30"), COL("0;39"), COL("0;39"), COL("0;39"),
+ COL("0;32"), COL("1;39"), COL("1;39")};
+#define FIELD_COLOR COL("34;1")
+
+static void put_buf(const char *s, int len, FILE *fout, jv *strout, int is_tty) {
+ if (strout) {
+ *strout = jv_string_append_buf(*strout, s, len);
+ } else {
+#ifdef WIN32
+ /* See util.h */
+ if (is_tty)
+ WriteFile((HANDLE)_get_osfhandle(fileno(fout)), s, len, NULL, NULL);
+ else
+ fwrite(s, 1, len, fout);
+#else
+ fwrite(s, 1, len, fout);
+#endif
+ }
+}
+
+static void put_char(char c, FILE* fout, jv* strout, int T) {
+ put_buf(&c, 1, fout, strout, T);
+}
+
+static void put_str(const char* s, FILE* fout, jv* strout, int T) {
+ put_buf(s, strlen(s), fout, strout, T);
+}
+
+static void put_indent(int n, int flags, FILE* fout, jv* strout, int T) {
+ if (flags & JV_PRINT_TAB) {
+ while (n--)
+ put_char('\t', fout, strout, T);
+ } else {
+ n *= ((flags & (JV_PRINT_SPACE0 | JV_PRINT_SPACE1 | JV_PRINT_SPACE2)) >> 8);
+ while (n--)
+ put_char(' ', fout, strout, T);
+ }
+}
+
+static void jvp_dump_string(jv str, int ascii_only, FILE* F, jv* S, int T) {
+ assert(jv_get_kind(str) == JV_KIND_STRING);
+ const char* i = jv_string_value(str);
+ const char* end = i + jv_string_length_bytes(jv_copy(str));
+ const char* cstart;
+ int c = 0;
+ char buf[32];
+ put_char('"', F, S, T);
+ while ((i = jvp_utf8_next((cstart = i), end, &c))) {
+ assert(c != -1);
+ int unicode_escape = 0;
+ if (0x20 <= c && c <= 0x7E) {
+ // printable ASCII
+ if (c == '"' || c == '\\') {
+ put_char('\\', F, S, T);
+ }
+ put_char(c, F, S, T);
+ } else if (c < 0x20 || c == 0x7F) {
+ // ASCII control character
+ switch (c) {
+ case '\b':
+ put_char('\\', F, S, T);
+ put_char('b', F, S, T);
+ break;
+ case '\t':
+ put_char('\\', F, S, T);
+ put_char('t', F, S, T);
+ break;
+ case '\r':
+ put_char('\\', F, S, T);
+ put_char('r', F, S, T);
+ break;
+ case '\n':
+ put_char('\\', F, S, T);
+ put_char('n', F, S, T);
+ break;
+ case '\f':
+ put_char('\\', F, S, T);
+ put_char('f', F, S, T);
+ break;
+ default:
+ unicode_escape = 1;
+ break;
+ }
+ } else {
+ if (ascii_only) {
+ unicode_escape = 1;
+ } else {
+ put_buf(cstart, i - cstart, F, S, T);
+ }
+ }
+ if (unicode_escape) {
+ if (c <= 0xffff) {
+ snprintf(buf, sizeof(buf), "\\u%04x", c);
+ } else {
+ c -= 0x10000;
+ snprintf(buf, sizeof(buf), "\\u%04x\\u%04x",
+ 0xD800 | ((c & 0xffc00) >> 10),
+ 0xDC00 | (c & 0x003ff));
+ }
+ put_str(buf, F, S, T);
+ }
+ }
+ assert(c != -1);
+ put_char('"', F, S, T);
+}
+
+static void put_refcnt(struct dtoa_context* C, int refcnt, FILE *F, jv* S, int T){
+ char buf[JVP_DTOA_FMT_MAX_LEN];
+ put_char(' ', F, S, T);
+ put_char('(', F, S, T);
+ put_str(jvp_dtoa_fmt(C, buf, refcnt), F, S, T);
+ put_char(')', F, S, T);
+}
+
+static void jv_dump_term(struct dtoa_context* C, jv x, int flags, int indent, FILE* F, jv* S) {
+ char buf[JVP_DTOA_FMT_MAX_LEN];
+ const char* color = 0;
+ double refcnt = (flags & JV_PRINT_REFCOUNT) ? jv_get_refcnt(x) - 1 : -1;
+ if (flags & JV_PRINT_COLOR) {
+ for (unsigned i=0; i<sizeof(color_kinds)/sizeof(color_kinds[0]); i++) {
+ if (jv_get_kind(x) == color_kinds[i]) {
+ color = colors[i];
+ put_str(color, F, S, flags & JV_PRINT_ISATTY);
+ break;
+ }
+ }
+ }
+ switch (jv_get_kind(x)) {
+ default:
+ case JV_KIND_INVALID:
+ if (flags & JV_PRINT_INVALID) {
+ jv msg = jv_invalid_get_msg(jv_copy(x));
+ if (jv_get_kind(msg) == JV_KIND_STRING) {
+ put_str("<invalid:", F, S, flags & JV_PRINT_ISATTY);
+ jvp_dump_string(msg, flags | JV_PRINT_ASCII, F, S, flags & JV_PRINT_ISATTY);
+ put_str(">", F, S, flags & JV_PRINT_ISATTY);
+ } else {
+ put_str("<invalid>", F, S, flags & JV_PRINT_ISATTY);
+ }
+ } else {
+ assert(0 && "Invalid value");
+ }
+ break;
+ case JV_KIND_NULL:
+ put_str("null", F, S, flags & JV_PRINT_ISATTY);
+ break;
+ case JV_KIND_FALSE:
+ put_str("false", F, S, flags & JV_PRINT_ISATTY);
+ break;
+ case JV_KIND_TRUE:
+ put_str("true", F, S, flags & JV_PRINT_ISATTY);
+ break;
+ case JV_KIND_NUMBER: {
+ double d = jv_number_value(x);
+ if (d != d) {
+ // JSON doesn't have NaN, so we'll render it as "null"
+ put_str("null", F, S, flags & JV_PRINT_ISATTY);
+ } else {
+ // Normalise infinities to something we can print in valid JSON
+ if (d > DBL_MAX) d = DBL_MAX;
+ if (d < -DBL_MAX) d = -DBL_MAX;
+ put_str(jvp_dtoa_fmt(C, buf, d), F, S, flags & JV_PRINT_ISATTY);
+ }
+ break;
+ }
+ case JV_KIND_STRING:
+ jvp_dump_string(x, flags & JV_PRINT_ASCII, F, S, flags & JV_PRINT_ISATTY);
+ if (flags & JV_PRINT_REFCOUNT)
+ put_refcnt(C, refcnt, F, S, flags & JV_PRINT_ISATTY);
+ break;
+ case JV_KIND_ARRAY: {
+ if (jv_array_length(jv_copy(x)) == 0) {
+ put_str("[]", F, S, flags & JV_PRINT_ISATTY);
+ break;
+ }
+ put_str("[", F, S, flags & JV_PRINT_ISATTY);
+ if (flags & JV_PRINT_PRETTY) {
+ put_char('\n', F, S, flags & JV_PRINT_ISATTY);
+ put_indent(indent + 1, flags, F, S, flags & JV_PRINT_ISATTY);
+ }
+ jv_array_foreach(x, i, elem) {
+ if (i!=0) {
+ if (flags & JV_PRINT_PRETTY) {
+ put_str(",\n", F, S, flags & JV_PRINT_ISATTY);
+ put_indent(indent + 1, flags, F, S, flags & JV_PRINT_ISATTY);
+ } else {
+ put_str(",", F, S, flags & JV_PRINT_ISATTY);
+ }
+ }
+ jv_dump_term(C, elem, flags, indent + 1, F, S);
+ if (color) put_str(color, F, S, flags & JV_PRINT_ISATTY);
+ }
+ if (flags & JV_PRINT_PRETTY) {
+ put_char('\n', F, S, flags & JV_PRINT_ISATTY);
+ put_indent(indent, flags, F, S, flags & JV_PRINT_ISATTY);
+ }
+ if (color) put_str(color, F, S, flags & JV_PRINT_ISATTY);
+ put_char(']', F, S, flags & JV_PRINT_ISATTY);
+ if (flags & JV_PRINT_REFCOUNT)
+ put_refcnt(C, refcnt, F, S, flags & JV_PRINT_ISATTY);
+ break;
+ }
+ case JV_KIND_OBJECT: {
+ if (jv_object_length(jv_copy(x)) == 0) {
+ put_str("{}", F, S, flags & JV_PRINT_ISATTY);
+ break;
+ }
+ put_char('{', F, S, flags & JV_PRINT_ISATTY);
+ if (flags & JV_PRINT_PRETTY) {
+ put_char('\n', F, S, flags & JV_PRINT_ISATTY);
+ put_indent(indent + 1, flags, F, S, flags & JV_PRINT_ISATTY);
+ }
+ int first = 1;
+ int i = 0;
+ jv keyset = jv_null();
+ while (1) {
+ jv key, value;
+ if (flags & JV_PRINT_SORTED) {
+ if (first) {
+ keyset = jv_keys(jv_copy(x));
+ i = 0;
+ } else {
+ i++;
+ }
+ if (i >= jv_array_length(jv_copy(keyset))) {
+ jv_free(keyset);
+ break;
+ }
+ key = jv_array_get(jv_copy(keyset), i);
+ value = jv_object_get(jv_copy(x), jv_copy(key));
+ } else {
+ if (first) {
+ i = jv_object_iter(x);
+ } else {
+ i = jv_object_iter_next(x, i);
+ }
+ if (!jv_object_iter_valid(x, i)) break;
+ key = jv_object_iter_key(x, i);
+ value = jv_object_iter_value(x, i);
+ }
+
+ if (!first) {
+ if (flags & JV_PRINT_PRETTY){
+ put_str(",\n", F, S, flags & JV_PRINT_ISATTY);
+ put_indent(indent + 1, flags, F, S, flags & JV_PRINT_ISATTY);
+ } else {
+ put_str(",", F, S, flags & JV_PRINT_ISATTY);
+ }
+ }
+ if (color) put_str(COLRESET, F, S, flags & JV_PRINT_ISATTY);
+
+ first = 0;
+ if (color) put_str(FIELD_COLOR, F, S, flags & JV_PRINT_ISATTY);
+ jvp_dump_string(key, flags & JV_PRINT_ASCII, F, S, flags & JV_PRINT_ISATTY);
+ jv_free(key);
+ if (color) put_str(COLRESET, F, S, flags & JV_PRINT_ISATTY);
+
+ if (color) put_str(color, F, S, flags & JV_PRINT_ISATTY);
+ put_str((flags & JV_PRINT_PRETTY) ? ": " : ":", F, S, flags & JV_PRINT_ISATTY);
+ if (color) put_str(COLRESET, F, S, flags & JV_PRINT_ISATTY);
+
+ jv_dump_term(C, value, flags, indent + 1, F, S);
+ if (color) put_str(color, F, S, flags & JV_PRINT_ISATTY);
+ }
+ if (flags & JV_PRINT_PRETTY) {
+ put_char('\n', F, S, flags & JV_PRINT_ISATTY);
+ put_indent(indent, flags, F, S, flags & JV_PRINT_ISATTY);
+ }
+ if (color) put_str(color, F, S, flags & JV_PRINT_ISATTY);
+ put_char('}', F, S, flags & JV_PRINT_ISATTY);
+ if (flags & JV_PRINT_REFCOUNT)
+ put_refcnt(C, refcnt, F, S, flags & JV_PRINT_ISATTY);
+ }
+ }
+ jv_free(x);
+ if (color) {
+ put_str(COLRESET, F, S, flags & JV_PRINT_ISATTY);
+ }
+}
+
+void jv_dumpf(jv x, FILE *f, int flags) {
+ struct dtoa_context C;
+ jvp_dtoa_context_init(&C);
+ jv_dump_term(&C, x, flags, 0, f, 0);
+ jvp_dtoa_context_free(&C);
+}
+
+void jv_dump(jv x, int flags) {
+ jv_dumpf(x, stdout, flags);
+}
+
+/* This one is nice for use in debuggers */
+void jv_show(jv x, int flags) {
+ if (flags == -1)
+ flags = JV_PRINT_PRETTY | JV_PRINT_COLOR | JV_PRINT_INDENT_FLAGS(2);
+ jv_dumpf(jv_copy(x), stderr, flags | JV_PRINT_INVALID);
+ fflush(stderr);
+}
+
+jv jv_dump_string(jv x, int flags) {
+ struct dtoa_context C;
+ jvp_dtoa_context_init(&C);
+ jv s = jv_string("");
+ jv_dump_term(&C, x, flags, 0, 0, &s);
+ jvp_dtoa_context_free(&C);
+ return s;
+}
+
+char *jv_dump_string_trunc(jv x, char *outbuf, size_t bufsize) {
+ x = jv_dump_string(x,0);
+ const char* p = jv_string_value(x);
+ const size_t len = strlen(p);
+ strncpy(outbuf, p, bufsize);
+ outbuf[bufsize - 1] = 0;
+ if (len > bufsize - 1 && bufsize >= 4) {
+ // Indicate truncation with '...'
+ outbuf[bufsize - 2]='.';
+ outbuf[bufsize - 3]='.';
+ outbuf[bufsize - 4]='.';
+ }
+ jv_free(x);
+ return outbuf;
+}