Add a simple API for writing JSON records.
To be used by the upcoming JSON audit module.
This commit is contained in:
2
MANIFEST
2
MANIFEST
@@ -89,6 +89,7 @@ include/sudo_event.h
|
|||||||
include/sudo_fatal.h
|
include/sudo_fatal.h
|
||||||
include/sudo_gettext.h
|
include/sudo_gettext.h
|
||||||
include/sudo_iolog.h
|
include/sudo_iolog.h
|
||||||
|
include/sudo_json.h
|
||||||
include/sudo_lbuf.h
|
include/sudo_lbuf.h
|
||||||
include/sudo_plugin.h
|
include/sudo_plugin.h
|
||||||
include/sudo_queue.h
|
include/sudo_queue.h
|
||||||
@@ -141,6 +142,7 @@ lib/util/host_port.c
|
|||||||
lib/util/inet_ntop.c
|
lib/util/inet_ntop.c
|
||||||
lib/util/inet_pton.c
|
lib/util/inet_pton.c
|
||||||
lib/util/isblank.c
|
lib/util/isblank.c
|
||||||
|
lib/util/json.c
|
||||||
lib/util/key_val.c
|
lib/util/key_val.c
|
||||||
lib/util/lbuf.c
|
lib/util/lbuf.c
|
||||||
lib/util/locking.c
|
lib/util/locking.c
|
||||||
|
70
include/sudo_json.h
Normal file
70
include/sudo_json.h
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: ISC
|
||||||
|
*
|
||||||
|
* Copyright (c) 2013-2020 Todd C. Miller <Todd.Miller@sudo.ws>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* JSON values may be of the following types.
|
||||||
|
*/
|
||||||
|
enum json_value_type {
|
||||||
|
JSON_STRING,
|
||||||
|
JSON_ID,
|
||||||
|
JSON_NUMBER,
|
||||||
|
JSON_OBJECT,
|
||||||
|
JSON_ARRAY,
|
||||||
|
JSON_BOOL,
|
||||||
|
JSON_NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* JSON value suitable for printing.
|
||||||
|
* Note: this does not support object values.
|
||||||
|
*/
|
||||||
|
struct json_value {
|
||||||
|
enum json_value_type type;
|
||||||
|
union {
|
||||||
|
const char *string;
|
||||||
|
char * const * array;
|
||||||
|
long long number;
|
||||||
|
id_t id;
|
||||||
|
bool boolean;
|
||||||
|
} u;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct json_container {
|
||||||
|
FILE *fp;
|
||||||
|
int indent_level;
|
||||||
|
int indent_increment;
|
||||||
|
bool need_comma;
|
||||||
|
};
|
||||||
|
|
||||||
|
__dso_public bool sudo_json_init_v1(struct json_container *json, FILE *fp, int indent);
|
||||||
|
#define sudo_json_init(_a, _b, _c) sudo_json_init_v1((_a), (_b), (_c))
|
||||||
|
|
||||||
|
__dso_public bool sudo_json_open_object_v1(struct json_container *json, const char *name);
|
||||||
|
#define sudo_json_open_object(_a, _b) sudo_json_open_object_v1((_a), (_b))
|
||||||
|
|
||||||
|
__dso_public bool sudo_json_close_object_v1(struct json_container *json);
|
||||||
|
#define sudo_json_close_object(_a) sudo_json_close_object_v1((_a))
|
||||||
|
|
||||||
|
__dso_public bool sudo_json_open_array_v1(struct json_container *json, const char *name);
|
||||||
|
#define sudo_json_open_array(_a, _b) sudo_json_open_array_v1((_a), (_b))
|
||||||
|
|
||||||
|
__dso_public bool sudo_json_close_array_v1(struct json_container *json);
|
||||||
|
#define sudo_json_close_array(_a) sudo_json_close_array_v1((_a))
|
||||||
|
|
||||||
|
__dso_public bool sudo_json_add_value_v1(struct json_container *json, const char *name, struct json_value *value);
|
||||||
|
#define sudo_json_add_value(_a, _b, _c) sudo_json_add_value_v1((_a), (_b), (_c))
|
@@ -1,7 +1,7 @@
|
|||||||
#
|
#
|
||||||
# SPDX-License-Identifier: ISC
|
# SPDX-License-Identifier: ISC
|
||||||
#
|
#
|
||||||
# Copyright (c) 2011-2019 Todd C. Miller <Todd.Miller@sudo.ws>
|
# Copyright (c) 2011-2020 Todd C. Miller <Todd.Miller@sudo.ws>
|
||||||
#
|
#
|
||||||
# Permission to use, copy, modify, and distribute this software for any
|
# Permission to use, copy, modify, and distribute this software for any
|
||||||
# purpose with or without fee is hereby granted, provided that the above
|
# purpose with or without fee is hereby granted, provided that the above
|
||||||
@@ -119,7 +119,7 @@ DEVEL = @DEVEL@
|
|||||||
SHELL = @SHELL@
|
SHELL = @SHELL@
|
||||||
|
|
||||||
LTOBJS = @DIGEST@ event.lo fatal.lo key_val.lo gethostname.lo gettime.lo \
|
LTOBJS = @DIGEST@ event.lo fatal.lo key_val.lo gethostname.lo gettime.lo \
|
||||||
getgrouplist.lo gidlist.lo host_port.lo lbuf.lo locking.lo \
|
getgrouplist.lo gidlist.lo host_port.lo json.lo lbuf.lo locking.lo \
|
||||||
logfac.lo logpri.lo mkdir_parents.lo parseln.lo progname.lo \
|
logfac.lo logpri.lo mkdir_parents.lo parseln.lo progname.lo \
|
||||||
roundup.lo secure_path.lo setgroups.lo strsplit.lo strtobool.lo \
|
roundup.lo secure_path.lo setgroups.lo strsplit.lo strtobool.lo \
|
||||||
strtoid.lo strtomode.lo strtonum.lo sudo_conf.lo \
|
strtoid.lo strtomode.lo strtonum.lo sudo_conf.lo \
|
||||||
@@ -803,6 +803,16 @@ isblank.i: $(srcdir)/isblank.c $(incdir)/sudo_compat.h $(top_builddir)/config.h
|
|||||||
$(CC) -E -o $@ $(CPPFLAGS) $<
|
$(CC) -E -o $@ $(CPPFLAGS) $<
|
||||||
isblank.plog: isblank.i
|
isblank.plog: isblank.i
|
||||||
rm -f $@; pvs-studio --cfg $(PVS_CFG) --sourcetree-root $(top_srcdir) --skip-cl-exe yes --source-file $(srcdir)/isblank.c --i-file $< --output-file $@
|
rm -f $@; pvs-studio --cfg $(PVS_CFG) --sourcetree-root $(top_srcdir) --skip-cl-exe yes --source-file $(srcdir)/isblank.c --i-file $< --output-file $@
|
||||||
|
json.lo: $(srcdir)/json.c $(incdir)/compat/stdbool.h $(incdir)/sudo_compat.h \
|
||||||
|
$(incdir)/sudo_debug.h $(incdir)/sudo_fatal.h $(incdir)/sudo_json.h \
|
||||||
|
$(incdir)/sudo_queue.h $(incdir)/sudo_util.h $(top_builddir)/config.h
|
||||||
|
$(LIBTOOL) $(LTFLAGS) --mode=compile $(CC) -c -o $@ $(CPPFLAGS) $(CFLAGS) $(ASAN_CFLAGS) $(PIE_CFLAGS) $(SSP_CFLAGS) $(srcdir)/json.c
|
||||||
|
json.i: $(srcdir)/json.c $(incdir)/compat/stdbool.h $(incdir)/sudo_compat.h \
|
||||||
|
$(incdir)/sudo_debug.h $(incdir)/sudo_fatal.h $(incdir)/sudo_json.h \
|
||||||
|
$(incdir)/sudo_queue.h $(incdir)/sudo_util.h $(top_builddir)/config.h
|
||||||
|
$(CC) -E -o $@ $(CPPFLAGS) $<
|
||||||
|
json.plog: json.i
|
||||||
|
rm -f $@; pvs-studio --cfg $(PVS_CFG) --sourcetree-root $(top_srcdir) --skip-cl-exe yes --source-file $(srcdir)/json.c --i-file $< --output-file $@
|
||||||
key_val.lo: $(srcdir)/key_val.c $(incdir)/compat/stdbool.h \
|
key_val.lo: $(srcdir)/key_val.c $(incdir)/compat/stdbool.h \
|
||||||
$(incdir)/sudo_compat.h $(incdir)/sudo_debug.h \
|
$(incdir)/sudo_compat.h $(incdir)/sudo_debug.h \
|
||||||
$(incdir)/sudo_queue.h $(incdir)/sudo_util.h \
|
$(incdir)/sudo_queue.h $(incdir)/sudo_util.h \
|
||||||
|
258
lib/util/json.c
Normal file
258
lib/util/json.c
Normal file
@@ -0,0 +1,258 @@
|
|||||||
|
/*
|
||||||
|
* SPDX-License-Identifier: ISC
|
||||||
|
*
|
||||||
|
* Copyright (c) 2013-2020 Todd C. Miller <Todd.Miller@sudo.ws>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is an open source non-commercial project. Dear PVS-Studio, please check it.
|
||||||
|
* PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#ifdef HAVE_STDBOOL_H
|
||||||
|
# include <stdbool.h>
|
||||||
|
#else
|
||||||
|
# include "compat/stdbool.h"
|
||||||
|
#endif /* HAVE_STDBOOL_H */
|
||||||
|
#ifdef HAVE_STRING_H
|
||||||
|
# include <string.h>
|
||||||
|
#endif /* HAVE_STRING_H */
|
||||||
|
#ifdef HAVE_STRINGS_H
|
||||||
|
# include <strings.h>
|
||||||
|
#endif /* HAVE_STRINGS_H */
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
|
#include "sudo_compat.h"
|
||||||
|
#include "sudo_debug.h"
|
||||||
|
#include "sudo_fatal.h"
|
||||||
|
#include "sudo_util.h"
|
||||||
|
#include "sudo_json.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Print "indent" number of blank characters.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
print_indent(FILE *fp, int indent)
|
||||||
|
{
|
||||||
|
while (indent--)
|
||||||
|
putc(' ', fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Print a quoted JSON string, escaping special characters.
|
||||||
|
* Does not support unicode escapes.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
json_print_string(struct json_container *json, const char *str)
|
||||||
|
{
|
||||||
|
char ch;
|
||||||
|
|
||||||
|
putc('\"', json->fp);
|
||||||
|
while ((ch = *str++) != '\0') {
|
||||||
|
switch (ch) {
|
||||||
|
case '"':
|
||||||
|
case '\\':
|
||||||
|
putc('\\', json->fp);
|
||||||
|
break;
|
||||||
|
case '\b':
|
||||||
|
ch = 'b';
|
||||||
|
putc('\\', json->fp);
|
||||||
|
break;
|
||||||
|
case '\f':
|
||||||
|
ch = 'f';
|
||||||
|
putc('\\', json->fp);
|
||||||
|
break;
|
||||||
|
case '\n':
|
||||||
|
ch = 'n';
|
||||||
|
putc('\\', json->fp);
|
||||||
|
break;
|
||||||
|
case '\r':
|
||||||
|
ch = 'r';
|
||||||
|
putc('\\', json->fp);
|
||||||
|
break;
|
||||||
|
case '\t':
|
||||||
|
ch = 't';
|
||||||
|
putc('\\', json->fp);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
putc(ch, json->fp);
|
||||||
|
}
|
||||||
|
putc('\"', json->fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
sudo_json_init_v1(struct json_container *json, FILE *fp, int indent)
|
||||||
|
{
|
||||||
|
debug_decl(sudo_json_init, SUDO_DEBUG_UTIL);
|
||||||
|
|
||||||
|
memset(json, 0, sizeof(*json));
|
||||||
|
json->fp = fp;
|
||||||
|
json->indent_level = indent;
|
||||||
|
json->indent_increment = indent;
|
||||||
|
|
||||||
|
debug_return_bool(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
sudo_json_open_object_v1(struct json_container *json, const char *name)
|
||||||
|
{
|
||||||
|
debug_decl(sudo_json_open_object, SUDO_DEBUG_UTIL);
|
||||||
|
|
||||||
|
/* Add comma if we are continuing an object/array. */
|
||||||
|
if (json->need_comma)
|
||||||
|
putc(',', json->fp);
|
||||||
|
putc('\n', json->fp);
|
||||||
|
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
|
||||||
|
json_print_string(json, name);
|
||||||
|
putc(':', json->fp);
|
||||||
|
putc(' ', json->fp);
|
||||||
|
putc('{', json->fp);
|
||||||
|
|
||||||
|
json->indent_level += json->indent_increment;
|
||||||
|
json->need_comma = false;
|
||||||
|
|
||||||
|
debug_return_bool(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
sudo_json_close_object_v1(struct json_container *json)
|
||||||
|
{
|
||||||
|
debug_decl(sudo_json_close_object, SUDO_DEBUG_UTIL);
|
||||||
|
|
||||||
|
json->indent_level -= json->indent_increment;
|
||||||
|
putc('\n', json->fp);
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
putc('}', json->fp);
|
||||||
|
|
||||||
|
debug_return_bool(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
sudo_json_open_array_v1(struct json_container *json, const char *name)
|
||||||
|
{
|
||||||
|
debug_decl(sudo_json_open_array, SUDO_DEBUG_UTIL);
|
||||||
|
|
||||||
|
/* Add comma if we are continuing an object/array. */
|
||||||
|
if (json->need_comma)
|
||||||
|
putc(',', json->fp);
|
||||||
|
putc('\n', json->fp);
|
||||||
|
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
|
||||||
|
json_print_string(json, name);
|
||||||
|
putc(':', json->fp);
|
||||||
|
putc(' ', json->fp);
|
||||||
|
putc('[', json->fp);
|
||||||
|
|
||||||
|
json->indent_level += json->indent_increment;
|
||||||
|
json->need_comma = false;
|
||||||
|
|
||||||
|
debug_return_bool(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
sudo_json_close_array_v1(struct json_container *json)
|
||||||
|
{
|
||||||
|
debug_decl(sudo_json_close_array, SUDO_DEBUG_UTIL);
|
||||||
|
|
||||||
|
json->indent_level -= json->indent_increment;
|
||||||
|
putc('\n', json->fp);
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
putc(']', json->fp);
|
||||||
|
|
||||||
|
debug_return_bool(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
sudo_json_add_value_v1(struct json_container *json, const char *name,
|
||||||
|
struct json_value *value)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
debug_decl(sudo_json_add_value, SUDO_DEBUG_UTIL);
|
||||||
|
|
||||||
|
/* Add comma if we are continuing an object/array. */
|
||||||
|
if (json->need_comma)
|
||||||
|
putc(',', json->fp);
|
||||||
|
putc('\n', json->fp);
|
||||||
|
json->need_comma = true;
|
||||||
|
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
|
||||||
|
/* name */
|
||||||
|
json_print_string(json, name);
|
||||||
|
putc(':', json->fp);
|
||||||
|
putc(' ', json->fp);
|
||||||
|
|
||||||
|
/* value */
|
||||||
|
switch (value->type) {
|
||||||
|
case JSON_STRING:
|
||||||
|
json_print_string(json, value->u.string);
|
||||||
|
break;
|
||||||
|
case JSON_ID:
|
||||||
|
fprintf(json->fp, "%u", (unsigned int)value->u.id);
|
||||||
|
break;
|
||||||
|
case JSON_NUMBER:
|
||||||
|
fprintf(json->fp, "%lld", value->u.number);
|
||||||
|
break;
|
||||||
|
case JSON_NULL:
|
||||||
|
fputs("null", json->fp);
|
||||||
|
break;
|
||||||
|
case JSON_BOOL:
|
||||||
|
fputs(value->u.boolean ? "true" : "false", json->fp);
|
||||||
|
break;
|
||||||
|
case JSON_ARRAY:
|
||||||
|
if (value->u.array[0] == NULL || value->u.array[1] == NULL) {
|
||||||
|
putc('[', json->fp);
|
||||||
|
putc(' ', json->fp);
|
||||||
|
if (value->u.array[0] != NULL) {
|
||||||
|
json_print_string(json, value->u.array[0]);
|
||||||
|
putc(' ', json->fp);
|
||||||
|
}
|
||||||
|
putc(']', json->fp);
|
||||||
|
} else {
|
||||||
|
putc('[', json->fp);
|
||||||
|
putc('\n', json->fp);
|
||||||
|
json->indent_level += json->indent_increment;
|
||||||
|
for (i = 0; value->u.array[i] != NULL; i++) {
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
json_print_string(json, value->u.array[i]);
|
||||||
|
if (value->u.array[i + 1] != NULL) {
|
||||||
|
putc(',', json->fp);
|
||||||
|
putc(' ', json->fp);
|
||||||
|
}
|
||||||
|
putc('\n', json->fp);
|
||||||
|
}
|
||||||
|
json->indent_level -= json->indent_increment;
|
||||||
|
print_indent(json->fp, json->indent_level);
|
||||||
|
putc(']', json->fp);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case JSON_OBJECT:
|
||||||
|
sudo_fatalx("internal error: can't print JSON_OBJECT");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
debug_return_bool(true);
|
||||||
|
}
|
@@ -37,8 +37,8 @@ sudo_debug_register_v1
|
|||||||
sudo_debug_set_active_instance_v1
|
sudo_debug_set_active_instance_v1
|
||||||
sudo_debug_update_fd_v1
|
sudo_debug_update_fd_v1
|
||||||
sudo_debug_vprintf2_v1
|
sudo_debug_vprintf2_v1
|
||||||
sudo_debug_write2_v1
|
|
||||||
sudo_debug_needed_v1
|
sudo_debug_needed_v1
|
||||||
|
sudo_debug_write2_v1
|
||||||
sudo_digest_alloc_v1
|
sudo_digest_alloc_v1
|
||||||
sudo_digest_final_v1
|
sudo_digest_final_v1
|
||||||
sudo_digest_free_v1
|
sudo_digest_free_v1
|
||||||
@@ -83,6 +83,12 @@ sudo_gethostname_v1
|
|||||||
sudo_gettime_awake_v1
|
sudo_gettime_awake_v1
|
||||||
sudo_gettime_mono_v1
|
sudo_gettime_mono_v1
|
||||||
sudo_gettime_real_v1
|
sudo_gettime_real_v1
|
||||||
|
sudo_json_add_value_v1
|
||||||
|
sudo_json_close_array_v1
|
||||||
|
sudo_json_close_object_v1
|
||||||
|
sudo_json_init_v1
|
||||||
|
sudo_json_open_array_v1
|
||||||
|
sudo_json_open_object_v1
|
||||||
sudo_lbuf_append_quoted_v1
|
sudo_lbuf_append_quoted_v1
|
||||||
sudo_lbuf_append_v1
|
sudo_lbuf_append_v1
|
||||||
sudo_lbuf_clearerr_v1
|
sudo_lbuf_clearerr_v1
|
||||||
|
Reference in New Issue
Block a user