2023-02-08 18:12:00 +01:00
|
|
|
/* Simple Plugin API */
|
|
|
|
|
/* SPDX-FileCopyrightText: Copyright © 2021 Wim Taymans */
|
|
|
|
|
/* SPDX-License-Identifier: MIT */
|
2021-02-13 20:17:27 +01:00
|
|
|
|
|
|
|
|
#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>
|
2025-06-17 17:27:10 +02:00
|
|
|
#include <getopt.h>
|
2021-02-13 20:17:27 +01:00
|
|
|
|
|
|
|
|
#include <spa/utils/result.h>
|
|
|
|
|
#include <spa/utils/json.h>
|
2026-02-26 10:51:17 +01:00
|
|
|
#include <spa/utils/json-builder.h>
|
2024-03-27 15:57:39 +01:00
|
|
|
#include <spa/debug/file.h>
|
2021-02-13 20:17:27 +01:00
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
#define DEFAULT_INDENT 2
|
2025-06-12 17:16:27 +02:00
|
|
|
|
|
|
|
|
struct data {
|
2025-06-17 17:27:10 +02:00
|
|
|
const char *filename;
|
2026-02-26 10:51:17 +01:00
|
|
|
|
|
|
|
|
FILE *out;
|
|
|
|
|
struct spa_json_builder builder;
|
2025-06-17 17:27:10 +02:00
|
|
|
|
|
|
|
|
void *data;
|
|
|
|
|
size_t size;
|
2025-06-12 17:16:27 +02:00
|
|
|
};
|
|
|
|
|
|
2026-02-26 10:51:17 +01:00
|
|
|
#define OPTIONS "hNC:Ri:s"
|
2025-06-17 17:27:10 +02:00
|
|
|
static const struct option long_options[] = {
|
|
|
|
|
{ "help", no_argument, NULL, 'h'},
|
|
|
|
|
|
2026-02-26 10:51:17 +01:00
|
|
|
{ "no-colors", no_argument, NULL, 'N' },
|
|
|
|
|
{ "color", optional_argument, NULL, 'C' },
|
|
|
|
|
{ "raw", no_argument, NULL, 'R' },
|
2025-06-17 17:27:10 +02:00
|
|
|
{ "indent", required_argument, NULL, 'i' },
|
|
|
|
|
{ "spa", no_argument, NULL, 's' },
|
|
|
|
|
|
|
|
|
|
{ NULL, 0, NULL, 0 }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void show_usage(struct data *d, const char *name, bool is_error)
|
|
|
|
|
{
|
|
|
|
|
FILE *fp;
|
|
|
|
|
|
|
|
|
|
fp = is_error ? stderr : stdout;
|
|
|
|
|
|
|
|
|
|
fprintf(fp, "%s [options] [spa-json-file]\n", name);
|
|
|
|
|
fprintf(fp,
|
|
|
|
|
" -h, --help Show this help\n"
|
|
|
|
|
"\n");
|
|
|
|
|
fprintf(fp,
|
2026-02-26 10:51:17 +01:00
|
|
|
" -N, --no-colors disable color output\n"
|
|
|
|
|
" -C, --color[=WHEN] whether to enable color support. WHEN is `never`, `always`, or `auto`\n"
|
|
|
|
|
" -R, --raw force raw output\n"
|
2025-06-17 17:27:10 +02:00
|
|
|
" -i --indent set indent (default %d)\n"
|
|
|
|
|
" -s --spa use simplified SPA JSON\n"
|
|
|
|
|
"\n",
|
|
|
|
|
DEFAULT_INDENT);
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-26 10:51:17 +01:00
|
|
|
static int dump(struct data *d, struct spa_json *it, const char *key, const char *value, int len)
|
2025-06-12 17:16:27 +02:00
|
|
|
{
|
2026-02-26 10:51:17 +01:00
|
|
|
struct spa_json_builder *b = &d->builder;
|
2021-02-13 20:17:27 +01:00
|
|
|
struct spa_json sub;
|
2024-03-23 20:58:37 +02:00
|
|
|
bool toplevel = false;
|
2026-02-26 10:51:17 +01:00
|
|
|
int res;
|
2021-02-13 20:17:27 +01:00
|
|
|
|
2024-03-23 20:58:37 +02:00
|
|
|
if (!value) {
|
|
|
|
|
toplevel = true;
|
|
|
|
|
value = "{";
|
|
|
|
|
len = 1;
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 20:17:27 +01:00
|
|
|
if (spa_json_is_array(value, len)) {
|
2026-02-26 10:51:17 +01:00
|
|
|
spa_json_builder_object_push(b, key, "[");
|
2021-02-13 20:17:27 +01:00
|
|
|
spa_json_enter(it, &sub);
|
|
|
|
|
while ((len = spa_json_next(&sub, &value)) > 0) {
|
2026-02-26 10:51:17 +01:00
|
|
|
if ((res = dump(d, &sub, NULL, value, len)) < 0)
|
2024-03-23 20:58:37 +02:00
|
|
|
return res;
|
2021-02-13 20:17:27 +01:00
|
|
|
}
|
2026-02-26 10:51:17 +01:00
|
|
|
spa_json_builder_pop(b, "]");
|
2021-02-13 20:17:27 +01:00
|
|
|
} else if (spa_json_is_object(value, len)) {
|
2026-02-26 10:51:17 +01:00
|
|
|
char k[1024];
|
|
|
|
|
spa_json_builder_object_push(b, key, "{");
|
2024-03-23 20:58:37 +02:00
|
|
|
if (!toplevel)
|
|
|
|
|
spa_json_enter(it, &sub);
|
|
|
|
|
else
|
|
|
|
|
sub = *it;
|
2026-02-26 10:51:17 +01:00
|
|
|
while ((len = spa_json_object_next(&sub, k, sizeof(k), &value)) > 0) {
|
|
|
|
|
res = dump(d, &sub, k, value, len);
|
2024-03-23 20:58:37 +02:00
|
|
|
if (res < 0) {
|
|
|
|
|
if (toplevel)
|
|
|
|
|
*it = sub;
|
|
|
|
|
return res;
|
|
|
|
|
}
|
2021-02-13 20:17:27 +01:00
|
|
|
}
|
2024-03-23 20:58:37 +02:00
|
|
|
if (toplevel)
|
|
|
|
|
*it = sub;
|
2026-02-26 10:51:17 +01:00
|
|
|
spa_json_builder_pop(b, "}");
|
2021-02-13 20:17:27 +01:00
|
|
|
} else {
|
2026-02-26 10:51:17 +01:00
|
|
|
spa_json_builder_add_simple(b, key, INT_MAX, 0, value, len);
|
2021-02-13 20:17:27 +01:00
|
|
|
}
|
2024-03-27 15:31:48 +01:00
|
|
|
if (spa_json_get_error(it, NULL, NULL))
|
2024-03-23 20:58:37 +02:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
static int process_json(struct data *d)
|
2024-03-23 20:58:37 +02:00
|
|
|
{
|
|
|
|
|
int len, res;
|
|
|
|
|
struct spa_json it;
|
|
|
|
|
const char *value;
|
|
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
if ((len = spa_json_begin(&it, d->data, d->size, &value)) <= 0) {
|
|
|
|
|
fprintf(stderr, "not a valid file '%s': %s\n", d->filename, spa_strerror(len));
|
2024-03-23 20:58:37 +02:00
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
if (!spa_json_is_container(value, len)) {
|
2025-06-17 17:27:10 +02:00
|
|
|
spa_json_init(&it, d->data, d->size);
|
2024-03-23 20:58:37 +02:00
|
|
|
value = NULL;
|
|
|
|
|
len = 0;
|
|
|
|
|
}
|
2025-06-17 17:27:10 +02:00
|
|
|
|
2026-02-26 10:51:17 +01:00
|
|
|
res = dump(d, &it, NULL, value, len);
|
2024-03-23 20:58:37 +02:00
|
|
|
if (spa_json_next(&it, &value) < 0)
|
|
|
|
|
res = -EINVAL;
|
|
|
|
|
|
2026-02-26 10:51:17 +01:00
|
|
|
fprintf(d->builder.f, "\n");
|
|
|
|
|
fflush(d->builder.f);
|
2024-03-23 20:58:37 +02:00
|
|
|
|
|
|
|
|
if (res < 0) {
|
2024-03-27 15:31:48 +01:00
|
|
|
struct spa_error_location loc;
|
2024-03-23 20:58:37 +02:00
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
if (spa_json_get_error(&it, d->data, &loc))
|
2024-03-27 15:57:39 +01:00
|
|
|
spa_debug_file_error_location(stderr, &loc,
|
|
|
|
|
"syntax error in file '%s': %s",
|
2025-06-17 17:27:10 +02:00
|
|
|
d->filename, loc.reason);
|
2024-03-23 20:58:37 +02:00
|
|
|
else
|
2025-06-17 17:27:10 +02:00
|
|
|
fprintf(stderr, "error parsing file '%s': %s\n",
|
|
|
|
|
d->filename, spa_strerror(res));
|
2024-03-23 20:58:37 +02:00
|
|
|
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
2021-02-13 20:17:27 +01:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
static int process_stdin(struct data *d)
|
2024-03-23 20:58:37 +02:00
|
|
|
{
|
|
|
|
|
uint8_t *buf = NULL, *p;
|
|
|
|
|
size_t alloc = 0, size = 0, read_size, res;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
alloc += 1024 + alloc;
|
|
|
|
|
p = realloc(buf, alloc);
|
|
|
|
|
if (!p) {
|
|
|
|
|
fprintf(stderr, "error: %m\n");
|
|
|
|
|
goto error;
|
|
|
|
|
}
|
|
|
|
|
buf = p;
|
|
|
|
|
|
|
|
|
|
read_size = alloc - size;
|
|
|
|
|
res = fread(buf + size, 1, read_size, stdin);
|
|
|
|
|
size += res;
|
|
|
|
|
} while (res == read_size);
|
|
|
|
|
|
|
|
|
|
if (ferror(stdin)) {
|
|
|
|
|
fprintf(stderr, "error: %m\n");
|
|
|
|
|
goto error;
|
|
|
|
|
}
|
2025-06-17 17:27:10 +02:00
|
|
|
d->data = buf;
|
|
|
|
|
d->size = size;
|
2024-03-23 20:58:37 +02:00
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
err = process_json(d);
|
2024-03-23 20:58:37 +02:00
|
|
|
free(buf);
|
|
|
|
|
|
|
|
|
|
return (err == 0) ? EXIT_SUCCESS : EXIT_FAILURE;
|
|
|
|
|
|
|
|
|
|
error:
|
|
|
|
|
free(buf);
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 20:17:27 +01:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
|
{
|
2025-06-17 17:27:10 +02:00
|
|
|
int c;
|
|
|
|
|
int longopt_index = 0;
|
2024-03-23 20:58:37 +02:00
|
|
|
int fd, res, exit_code = EXIT_FAILURE;
|
2026-02-26 10:51:17 +01:00
|
|
|
int flags = 0, indent = -1;
|
2025-06-17 17:27:10 +02:00
|
|
|
struct data d;
|
2021-02-13 20:17:27 +01:00
|
|
|
struct stat sbuf;
|
2026-02-26 10:51:17 +01:00
|
|
|
bool raw = false, colors = false;
|
2021-02-13 20:17:27 +01:00
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
spa_zero(d);
|
|
|
|
|
|
|
|
|
|
d.filename = "-";
|
2026-02-26 10:51:17 +01:00
|
|
|
d.out = stdout;
|
|
|
|
|
|
|
|
|
|
if (getenv("NO_COLOR") == NULL && isatty(fileno(d.out)))
|
|
|
|
|
colors = true;
|
|
|
|
|
setlinebuf(d.out);
|
2025-06-17 17:27:10 +02:00
|
|
|
|
|
|
|
|
while ((c = getopt_long(argc, argv, OPTIONS, long_options, &longopt_index)) != -1) {
|
|
|
|
|
switch (c) {
|
|
|
|
|
case 'h' :
|
|
|
|
|
show_usage(&d, argv[0], false);
|
|
|
|
|
return 0;
|
2026-02-26 10:51:17 +01:00
|
|
|
case 'N' :
|
|
|
|
|
colors = false;
|
|
|
|
|
break;
|
|
|
|
|
case 'C' :
|
|
|
|
|
if (optarg == NULL || !strcmp(optarg, "auto"))
|
|
|
|
|
break; /* nothing to do, tty detection was done
|
|
|
|
|
before parsing options */
|
|
|
|
|
else if (!strcmp(optarg, "never"))
|
|
|
|
|
colors = false;
|
|
|
|
|
else if (!strcmp(optarg, "always"))
|
|
|
|
|
colors = true;
|
|
|
|
|
else {
|
|
|
|
|
fprintf(stderr, "Unknown color: %s\n", optarg);
|
|
|
|
|
show_usage(&d, argv[0], true);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'R':
|
|
|
|
|
raw = true;
|
|
|
|
|
break;
|
2025-06-17 17:27:10 +02:00
|
|
|
case 'i':
|
2026-02-26 10:51:17 +01:00
|
|
|
indent = atoi(optarg);
|
2025-06-17 17:27:10 +02:00
|
|
|
break;
|
|
|
|
|
case 's':
|
2026-02-26 10:51:17 +01:00
|
|
|
flags |= SPA_JSON_BUILDER_FLAG_SIMPLE;
|
2025-06-17 17:27:10 +02:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
show_usage(&d, argv[0], true);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
2021-02-13 20:17:27 +01:00
|
|
|
}
|
2025-06-17 17:27:10 +02:00
|
|
|
|
|
|
|
|
if (optind < argc)
|
|
|
|
|
d.filename = argv[optind++];
|
|
|
|
|
|
|
|
|
|
if (spa_streq(d.filename, "-"))
|
|
|
|
|
return process_stdin(&d);
|
|
|
|
|
|
2025-06-17 18:08:36 +02:00
|
|
|
if ((fd = open(d.filename, O_CLOEXEC | O_RDONLY)) < 0) {
|
2025-06-17 17:27:10 +02:00
|
|
|
fprintf(stderr, "error opening file '%s': %m\n", d.filename);
|
2021-02-13 20:17:27 +01:00
|
|
|
goto error;
|
|
|
|
|
}
|
2025-06-17 17:27:10 +02:00
|
|
|
if (fstat(fd, &sbuf) < 0) {
|
|
|
|
|
fprintf(stderr, "error statting file '%s': %m\n", d.filename);
|
|
|
|
|
goto error_close;
|
|
|
|
|
}
|
|
|
|
|
if ((d.data = mmap(NULL, sbuf.st_size, PROT_READ, MAP_PRIVATE, fd, 0)) == MAP_FAILED) {
|
|
|
|
|
fprintf(stderr, "error mmapping file '%s': %m\n", d.filename);
|
|
|
|
|
goto error_close;
|
2021-02-13 20:17:27 +01:00
|
|
|
}
|
2025-06-17 17:27:10 +02:00
|
|
|
d.size = sbuf.st_size;
|
2021-02-13 20:17:27 +01:00
|
|
|
|
2026-02-26 10:51:17 +01:00
|
|
|
if (!raw)
|
|
|
|
|
flags |= SPA_JSON_BUILDER_FLAG_PRETTY;
|
|
|
|
|
if (colors)
|
|
|
|
|
flags |= SPA_JSON_BUILDER_FLAG_COLOR;
|
|
|
|
|
|
|
|
|
|
spa_json_builder_file(&d.builder, d.out, flags);
|
|
|
|
|
if (indent >= 0)
|
|
|
|
|
d.builder.indent = indent;
|
|
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
res = process_json(&d);
|
2024-03-23 20:58:37 +02:00
|
|
|
if (res < 0)
|
|
|
|
|
exit_code = EXIT_FAILURE;
|
|
|
|
|
else
|
|
|
|
|
exit_code = EXIT_SUCCESS;
|
2021-02-13 20:17:27 +01:00
|
|
|
|
2025-06-17 17:27:10 +02:00
|
|
|
munmap(d.data, sbuf.st_size);
|
2021-02-13 20:17:27 +01:00
|
|
|
error_close:
|
2025-06-17 17:27:10 +02:00
|
|
|
close(fd);
|
2021-02-13 20:17:27 +01:00
|
|
|
error:
|
|
|
|
|
return exit_code;
|
|
|
|
|
}
|