1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
|
(* virt-v2v
* Copyright (C) 2009-2019 Red Hat Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*)
(* Simple JSON generator. *)
type field = string * json_t
and json_t =
| Null
| String of string
| Int of int64
| Float of float
| Bool of bool
| List of json_t list
| Dict of field list
and doc = field list
type output_format =
| Compact
| Indented
let spaces_for_indent level =
let len = level * 2 in
let b = Bytes.create len in
Bytes.fill b 0 len ' ';
Bytes.to_string b
let print_dict_after_start ~fmt ~indent ~size =
match size, fmt with
| 0, Compact -> ""
| _, Compact -> " "
| _, Indented -> "\n"
let print_dict_before_end ~fmt ~indent ~size =
match size, fmt with
| 0, _ -> ""
| _, Compact -> " "
| _, Indented -> "\n"
let print_indent ~fmt ~indent =
match fmt with
| Compact -> ""
| Indented -> spaces_for_indent indent
(* JSON quoting. *)
let json_escape_string str =
let res = ref "" in
for i = 0 to String.length str - 1 do
res := !res ^ (match str.[i] with
| '"' -> "\\\""
| '\\' -> "\\\\"
| '\b' -> "\\b"
| '\n' -> "\\n"
| '\r' -> "\\r"
| '\t' -> "\\t"
| c -> String.make 1 c)
done;
!res
let json_quote_string str =
"\"" ^ (json_escape_string str) ^ "\""
let rec output_dict fields ~fmt ~indent =
let size = List.length fields in
let newlinesep =
match fmt with
| Compact -> ", "
| Indented -> ",\n" in
"{" ^ (print_dict_after_start ~fmt ~indent ~size) ^
String.concat newlinesep (
List.map (
fun (n, f) ->
(print_indent ~fmt ~indent:(indent + 1)) ^ (json_quote_string n)
^ ": " ^ (output_field ~fmt ~indent f)
) fields
)
^ (print_dict_before_end ~fmt ~indent ~size) ^ (print_indent ~fmt ~indent) ^ "}"
and output_list fields ~fmt ~indent =
let size = List.length fields in
let newlinesep =
match fmt with
| Compact -> ", "
| Indented -> ",\n" in
"[" ^ (print_dict_after_start ~fmt ~indent ~size) ^
String.concat newlinesep (
List.map (
fun f ->
(print_indent ~fmt ~indent:(indent + 1)) ^ (output_field ~fmt ~indent f)
) fields
)
^ (print_dict_before_end ~fmt ~indent ~size) ^ (print_indent ~fmt ~indent) ^ "]"
and output_field ~indent ~fmt = function
| Null -> "null"
| String s -> json_quote_string s
| Int i -> Int64.to_string i
(* The JSON standard permits either "1" or "1.0" but not "1.".
* OCaml string_of_float will generate "1.", but the %g formatter
* will only generate the valid JSON values.
*)
| Float f -> Printf.sprintf "%g" f
| Bool b -> if b then "true" else "false"
| List l -> output_list ~indent:(indent + 1) ~fmt l
| Dict d -> output_dict ~indent:(indent + 1) ~fmt d
let string_of_doc ?(fmt = Compact) fields =
output_dict fields ~fmt ~indent:0
|