pipewire/spa/tools/spa-json-dump.c
Barnabás Pőcze 934ab3036e treewide: use SPDX tags to specify copyright information
SPDX tags make the licensing information easy to understand and clear,
and they are machine parseable.

See https://spdx.dev for more information.
2023-02-16 10:54:48 +00:00

145 lines
3.5 KiB
C

/* Simple Plugin API */
/* SPDX-FileCopyrightText: Copyright © 2021 Wim Taymans */
/* SPDX-License-Identifier: MIT */
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <errno.h>
#include <spa/utils/result.h>
#include <spa/utils/json.h>
static void encode_string(FILE *f, const char *val, int len)
{
int i;
fprintf(f, "\"");
for (i = 0; i < len; i++) {
char v = val[i];
switch (v) {
case '\n':
fprintf(f, "\\n");
break;
case '\r':
fprintf(f, "\\r");
break;
case '\b':
fprintf(f, "\\b");
break;
case '\t':
fprintf(f, "\\t");
break;
case '\f':
fprintf(f, "\\f");
break;
case '\\': case '"':
fprintf(f, "\\%c", v);
break;
default:
if (v > 0 && v < 0x20)
fprintf(f, "\\u%04x", v);
else
fprintf(f, "%c", v);
break;
}
}
fprintf(f, "\"");
}
static int dump(FILE *file, int indent, struct spa_json *it, const char *value, int len)
{
struct spa_json sub;
int count = 0;
char key[1024];
if (spa_json_is_array(value, len)) {
fprintf(file, "[");
spa_json_enter(it, &sub);
while ((len = spa_json_next(&sub, &value)) > 0) {
fprintf(file, "%s\n%*s", count++ > 0 ? "," : "",
indent+2, "");
dump(file, indent+2, &sub, value, len);
}
fprintf(file, "%s%*s]", count > 0 ? "\n" : "",
count > 0 ? indent : 0, "");
} else if (spa_json_is_object(value, len)) {
fprintf(file, "{");
spa_json_enter(it, &sub);
while (spa_json_get_string(&sub, key, sizeof(key)) > 0) {
fprintf(file, "%s\n%*s",
count++ > 0 ? "," : "",
indent+2, "");
encode_string(file, key, strlen(key));
fprintf(file, ": ");
if ((len = spa_json_next(&sub, &value)) <= 0)
break;
dump(file, indent+2, &sub, value, len);
}
fprintf(file, "%s%*s}", count > 0 ? "\n" : "",
count > 0 ? indent : 0, "");
} else if (spa_json_is_string(value, len) ||
spa_json_is_null(value, len) ||
spa_json_is_bool(value, len) ||
spa_json_is_int(value, len) ||
spa_json_is_float(value, len)) {
fprintf(file, "%.*s", len, value);
} else {
encode_string(file, value, len);
}
return 0;
}
int main(int argc, char *argv[])
{
int fd, len, res, exit_code = EXIT_FAILURE;
void *data;
struct stat sbuf;
struct spa_json it;
const char *value;
if (argc < 2) {
fprintf(stderr, "usage: %s <spa-json-file>\n", argv[0]);
goto error;
}
if ((fd = open(argv[1], O_CLOEXEC | O_RDONLY)) < 0) {
fprintf(stderr, "error opening file '%s': %m\n", argv[1]);
goto error;
}
if (fstat(fd, &sbuf) < 0) {
fprintf(stderr, "error statting file '%s': %m\n", argv[1]);
goto error_close;
}
if ((data = mmap(NULL, sbuf.st_size, PROT_READ, MAP_PRIVATE, fd, 0)) == MAP_FAILED) {
fprintf(stderr, "error mmapping file '%s': %m\n", argv[1]);
goto error_close;
}
spa_json_init(&it, data, sbuf.st_size);
if ((len = spa_json_next(&it, &value)) <= 0) {
fprintf(stderr, "not a valid file '%s': %s\n", argv[1], spa_strerror(len));
goto error_unmap;
}
if (!spa_json_is_container(value, len)) {
spa_json_init(&it, data, sbuf.st_size);
value = "{";
len = 1;
}
if ((res = dump(stdout, 0, &it, value, len)) < 0) {
fprintf(stderr, "error parsing file '%s': %s\n", argv[1], spa_strerror(res));
goto error_unmap;
}
fprintf(stdout, "\n");
exit_code = EXIT_SUCCESS;
error_unmap:
munmap(data, sbuf.st_size);
error_close:
close(fd);
error:
return exit_code;
}