src/usr.sbin/rpki-client/json.c
2023-06-22 18:19:48 +00:00

325 lines
6.0 KiB
C

/* $OpenBSD: json.c,v 1.4 2023/06/22 09:08:02 claudio Exp $ */
/*
* Copyright (c) 2020 Claudio Jeker <claudio@openbsd.org>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include <ctype.h>
#include <err.h>
#include <stdarg.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "json.h"
#define JSON_MAX_STACK 16
enum json_type {
NONE,
START,
ARRAY,
OBJECT
};
static struct json_stack {
const char *name;
unsigned int count;
int compact;
enum json_type type;
} stack[JSON_MAX_STACK];
static char indent[JSON_MAX_STACK + 1];
static int level;
static int eb;
static FILE *jsonfh;
static void
do_comma_indent(void)
{
char sp = '\n';
if (stack[level].compact)
sp = ' ';
if (stack[level].count++ > 0) {
if (!eb)
eb = fprintf(jsonfh, ",%c", sp) < 0;
}
if (stack[level].compact)
return;
if (!eb)
eb = fprintf(jsonfh, "\t%.*s", level, indent) < 0;
}
static void
do_name(const char *name)
{
if (stack[level].type == ARRAY)
return;
if (!eb)
eb = fprintf(jsonfh, "\"%s\": ", name) < 0;
}
static int
do_find(enum json_type type, const char *name)
{
int i;
for (i = level; i > 0; i--)
if (type == stack[i].type &&
strcmp(name, stack[i].name) == 0)
return i;
/* not found */
return -1;
}
void
json_do_start(FILE *fh)
{
memset(indent, '\t', JSON_MAX_STACK);
memset(stack, 0, sizeof(stack));
level = 0;
stack[level].type = START;
jsonfh = fh;
eb = 0;
eb = fprintf(jsonfh, "{\n") < 0;
}
int
json_do_finish(void)
{
while (level > 0)
json_do_end();
if (!eb)
eb = fprintf(jsonfh, "\n}\n") < 0;
return -eb;
}
void
json_do_array(const char *name)
{
int i, l;
char sp = '\n';
if ((l = do_find(ARRAY, name)) > 0) {
/* array already in use, close element and move on */
for (i = level - l; i > 0; i--)
json_do_end();
return;
}
/* Do not stack arrays, while allowed this is not needed */
if (stack[level].type == ARRAY)
json_do_end();
if (stack[level].compact)
sp = ' ';
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "[%c", sp) < 0;
if (++level >= JSON_MAX_STACK)
errx(1, "json stack too deep");
stack[level].name = name;
stack[level].type = ARRAY;
stack[level].count = 0;
/* inherit compact setting from above level */
stack[level].compact = stack[level - 1].compact;
}
void
json_do_object(const char *name, int compact)
{
int i, l;
char sp = '\n';
if ((l = do_find(OBJECT, name)) > 0) {
/* roll back to that object and close it */
for (i = level - l; i >= 0; i--)
json_do_end();
}
if (compact)
sp = ' ';
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "{%c", sp) < 0;
if (++level >= JSON_MAX_STACK)
errx(1, "json stack too deep");
stack[level].name = name;
stack[level].type = OBJECT;
stack[level].count = 0;
stack[level].compact = compact;
}
void
json_do_end(void)
{
char c;
if (stack[level].type == ARRAY)
c = ']';
else if (stack[level].type == OBJECT)
c = '}';
else
errx(1, "json bad stack state");
if (!stack[level].compact) {
if (!eb)
eb = fprintf(jsonfh, "\n%.*s%c", level, indent, c) < 0;
} else {
if (!eb)
eb = fprintf(jsonfh, " %c", c) < 0;
}
stack[level].name = NULL;
stack[level].type = NONE;
stack[level].count = 0;
stack[level].compact = 0;
if (level-- <= 0)
errx(1, "json stack underflow");
stack[level].count++;
}
void
json_do_printf(const char *name, const char *fmt, ...)
{
va_list ap;
char *str;
va_start(ap, fmt);
if (!eb) {
if (vasprintf(&str, fmt, ap) == -1)
errx(1, "json printf failed");
json_do_string(name, str);
free(str);
}
va_end(ap);
}
void
json_do_string(const char *name, const char *v)
{
unsigned char c;
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "\"") < 0;
while ((c = *v++) != '\0' && !eb) {
/* skip escaping '/' since our use case does not require it */
switch (c) {
case '"':
eb = fprintf(jsonfh, "\\\"") < 0;
break;
case '\\':
eb = fprintf(jsonfh, "\\\\") < 0;
break;
case '\b':
eb = fprintf(jsonfh, "\\b") < 0;
break;
case '\f':
eb = fprintf(jsonfh, "\\f") < 0;
break;
case '\n':
eb = fprintf(jsonfh, "\\n") < 0;
break;
case '\r':
eb = fprintf(jsonfh, "\\r") < 0;
break;
case '\t':
eb = fprintf(jsonfh, "\\t") < 0;
break;
default:
if (iscntrl(c))
errx(1, "bad control character in string");
eb = putc(c, jsonfh) == EOF;
break;
}
}
if (!eb)
eb = fprintf(jsonfh, "\"") < 0;
}
void
json_do_hexdump(const char *name, void *buf, size_t len)
{
uint8_t *data = buf;
size_t i;
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "\"") < 0;
for (i = 0; i < len; i++)
if (!eb)
eb = fprintf(jsonfh, "%02x", *(data + i)) < 0;
if (!eb)
eb = fprintf(jsonfh, "\"") < 0;
}
void
json_do_bool(const char *name, int v)
{
do_comma_indent();
do_name(name);
if (v) {
if (!eb)
eb = fprintf(jsonfh, "true") < 0;
} else {
if (!eb)
eb = fprintf(jsonfh, "false") < 0;
}
}
void
json_do_uint(const char *name, unsigned long long v)
{
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "%llu", v) < 0;
}
void
json_do_int(const char *name, long long v)
{
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "%lld", v) < 0;
}
void
json_do_double(const char *name, double v)
{
do_comma_indent();
do_name(name);
if (!eb)
eb = fprintf(jsonfh, "%f", v) < 0;
}