/* * Copyright (c) The FFmpeg developers * * This file is part of FFmpeg. * * FFmpeg is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * FFmpeg is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with FFmpeg; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include #include "avtextformat.h" #include "libavutil/bprint.h" #include "libavutil/opt.h" #include "tf_internal.h" /* JSON output */ typedef struct JSONContext { const AVClass *class; int indent_level; int compact; const char *item_sep, *item_start_end; } JSONContext; #undef OFFSET #define OFFSET(x) offsetof(JSONContext, x) static const AVOption json_options[] = { { "compact", "enable compact output", OFFSET(compact), AV_OPT_TYPE_BOOL, { .i64 = 0 }, 0, 1 }, { "c", "enable compact output", OFFSET(compact), AV_OPT_TYPE_BOOL, { .i64 = 0 }, 0, 1 }, { NULL } }; DEFINE_FORMATTER_CLASS(json); static av_cold int json_init(AVTextFormatContext *wctx) { JSONContext *json = wctx->priv; json->item_sep = json->compact ? ", " : ",\n"; json->item_start_end = json->compact ? " " : "\n"; return 0; } static const char *json_escape_str(AVBPrint *dst, const char *src, void *log_ctx) { static const char json_escape[] = { '"', '\\', '\b', '\f', '\n', '\r', '\t', 0 }; static const char json_subst[] = { '"', '\\', 'b', 'f', 'n', 'r', 't', 0 }; const char *p; if (!src) { av_log(log_ctx, AV_LOG_WARNING, "Cannot escape NULL string, returning NULL\n"); return NULL; } for (p = src; *p; p++) { char *s = strchr(json_escape, *p); if (s) { av_bprint_chars(dst, '\\', 1); av_bprint_chars(dst, json_subst[s - json_escape], 1); } else if ((unsigned char)*p < 32) { av_bprintf(dst, "\\u00%02x", (unsigned char)*p); } else { av_bprint_chars(dst, *p, 1); } } return dst->str; } #define JSON_INDENT() writer_printf(wctx, "%*c", json->indent_level * 4, ' ') static void json_print_section_header(AVTextFormatContext *wctx, const void *data) { const AVTextFormatSection *section = tf_get_section(wctx, wctx->level); const AVTextFormatSection *parent_section = tf_get_parent_section(wctx, wctx->level); JSONContext *json = wctx->priv; AVBPrint buf; if (!section) return; if (wctx->level && wctx->nb_item[wctx->level - 1]) writer_put_str(wctx, ",\n"); if (section->flags & AV_TEXTFORMAT_SECTION_FLAG_IS_WRAPPER) { writer_put_str(wctx, "{\n"); json->indent_level++; } else { av_bprint_init(&buf, 1, AV_BPRINT_SIZE_UNLIMITED); json_escape_str(&buf, section->name, wctx); JSON_INDENT(); json->indent_level++; if (section->flags & AV_TEXTFORMAT_SECTION_FLAG_IS_ARRAY) { writer_printf(wctx, "\"%s\": [\n", buf.str); } else if (parent_section && !(parent_section->flags & AV_TEXTFORMAT_SECTION_FLAG_IS_ARRAY)) { writer_printf(wctx, "\"%s\": {%s", buf.str, json->item_start_end); } else { writer_printf(wctx, "{%s", json->item_start_end); /* this is required so the parser can distinguish between packets and frames */ if (parent_section && parent_section->flags & AV_TEXTFORMAT_SECTION_FLAG_NUMBERING_BY_TYPE) { if (!json->compact) JSON_INDENT(); writer_printf(wctx, "\"type\": \"%s\"", section->name); wctx->nb_item[wctx->level]++; } } av_bprint_finalize(&buf, NULL); } } static void json_print_section_footer(AVTextFormatContext *wctx) { const AVTextFormatSection *section = tf_get_section(wctx, wctx->level); JSONContext *json = wctx->priv; if (!section) return; if (wctx->level == 0) { json->indent_level--; writer_put_str(wctx, "\n}\n"); } else if (section->flags & AV_TEXTFORMAT_SECTION_FLAG_IS_ARRAY) { writer_w8(wctx, '\n'); json->indent_level--; JSON_INDENT(); writer_w8(wctx, ']'); } else { writer_put_str(wctx, json->item_start_end); json->indent_level--; if (!json->compact) JSON_INDENT(); writer_w8(wctx, '}'); } } static inline void json_print_item_str(AVTextFormatContext *wctx, const char *key, const char *value) { AVBPrint buf; av_bprint_init(&buf, 1, AV_BPRINT_SIZE_UNLIMITED); writer_printf(wctx, "\"%s\":", json_escape_str(&buf, key, wctx)); av_bprint_clear(&buf); writer_printf(wctx, " \"%s\"", json_escape_str(&buf, value, wctx)); av_bprint_finalize(&buf, NULL); } static void json_print_str(AVTextFormatContext *wctx, const char *key, const char *value) { const AVTextFormatSection *section = tf_get_section(wctx, wctx->level); const AVTextFormatSection *parent_section = tf_get_parent_section(wctx, wctx->level); JSONContext *json = wctx->priv; if (!section) return; if (wctx->nb_item[wctx->level] || (parent_section && parent_section->flags & AV_TEXTFORMAT_SECTION_FLAG_NUMBERING_BY_TYPE)) writer_put_str(wctx, json->item_sep); if (!json->compact) JSON_INDENT(); json_print_item_str(wctx, key, value); } static void json_print_int(AVTextFormatContext *wctx, const char *key, int64_t value) { const AVTextFormatSection *section = tf_get_section(wctx, wctx->level); const AVTextFormatSection *parent_section = tf_get_parent_section(wctx, wctx->level); JSONContext *json = wctx->priv; AVBPrint buf; if (!section) return; if (wctx->nb_item[wctx->level] || (parent_section && parent_section->flags & AV_TEXTFORMAT_SECTION_FLAG_NUMBERING_BY_TYPE)) writer_put_str(wctx, json->item_sep); if (!json->compact) JSON_INDENT(); av_bprint_init(&buf, 1, AV_BPRINT_SIZE_UNLIMITED); writer_printf(wctx, "\"%s\": %"PRId64, json_escape_str(&buf, key, wctx), value); av_bprint_finalize(&buf, NULL); } const AVTextFormatter avtextformatter_json = { .name = "json", .priv_size = sizeof(JSONContext), .init = json_init, .print_section_header = json_print_section_header, .print_section_footer = json_print_section_footer, .print_integer = json_print_int, .print_string = json_print_str, .flags = AV_TEXTFORMAT_FLAG_SUPPORTS_MIXED_ARRAY_CONTENT, .priv_class = &json_class, };