| author | wenzelm |
| Wed, 01 Mar 2023 13:30:35 +0100 | |
| changeset 77438 | 0030eabbe6c3 |
| parent 77347 | 739a9c34c538 |
| child 78597 | ecf0b65ada9e |
| permissions | -rw-r--r-- |
| 63992 | 1 |
/* Title: Pure/General/json.scala |
| 63644 | 2 |
Author: Makarius |
3 |
||
| 67819 | 4 |
Support for JSON: https://www.json.org/. |
| 67874 | 5 |
|
6 |
See also http://seriot.ch/parsing_json.php "Parsing JSON is a Minefield". |
|
| 63644 | 7 |
*/ |
8 |
||
9 |
package isabelle |
|
10 |
||
11 |
||
| 71601 | 12 |
import scala.util.matching.Regex |
| 75405 | 13 |
import scala.util.parsing.combinator |
| 66922 | 14 |
import scala.util.parsing.combinator.lexical.Scanners |
15 |
import scala.util.parsing.input.CharArrayReader.EofCh |
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
16 |
|
| 66922 | 17 |
|
| 75393 | 18 |
object JSON {
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
19 |
type T = Any |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
20 |
type S = String |
| 63644 | 21 |
|
| 75393 | 22 |
object Object {
|
| 67857 | 23 |
type Entry = (String, JSON.T) |
24 |
||
| 67841 | 25 |
type T = Map[String, JSON.T] |
| 67850 | 26 |
val empty: Object.T = Map.empty |
27 |
||
| 67857 | 28 |
def apply(entries: Entry*): Object.T = Map(entries:_*) |
| 67841 | 29 |
|
| 71776 | 30 |
def unapply(obj: Any): Option[Object.T] = |
| 67841 | 31 |
obj match {
|
32 |
case m: Map[_, _] if m.keySet.forall(_.isInstanceOf[String]) => |
|
33 |
Some(m.asInstanceOf[Object.T]) |
|
34 |
case _ => None |
|
35 |
} |
|
| 77347 | 36 |
|
37 |
def parse(s: S): Object.T = |
|
38 |
unapply(JSON.parse(s)).getOrElse(error("Bad JSON object " + quote(s)))
|
|
| 67841 | 39 |
} |
| 66928 | 40 |
|
| 63644 | 41 |
|
| 66922 | 42 |
/* lexer */ |
43 |
||
| 75393 | 44 |
object Kind extends Enumeration {
|
| 75406 | 45 |
val KEYWORD, STRING, NUMBER, ERROR = this.Value |
| 66922 | 46 |
} |
47 |
||
| 75393 | 48 |
sealed case class Token(kind: Kind.Value, text: String) {
|
| 66922 | 49 |
def is_keyword: Boolean = kind == Kind.KEYWORD |
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
50 |
def is_keyword(name: String): Boolean = kind == Kind.KEYWORD && text == name |
| 66922 | 51 |
def is_string: Boolean = kind == Kind.STRING |
52 |
def is_number: Boolean = kind == Kind.NUMBER |
|
53 |
def is_error: Boolean = kind == Kind.ERROR |
|
54 |
} |
|
55 |
||
| 75393 | 56 |
object Lexer extends Scanners with Scan.Parsers {
|
| 66922 | 57 |
override type Elem = Char |
58 |
type Token = JSON.Token |
|
59 |
||
60 |
def errorToken(msg: String): Token = Token(Kind.ERROR, msg) |
|
61 |
||
|
67111
42f290d8ccbd
more accurate JSON parsing according to http://seriot.ch/parsing_json.php
wenzelm
parents:
66928
diff
changeset
|
62 |
val white_space: String = " \t\n\r" |
| 71601 | 63 |
override val whiteSpace: Regex = ("[" + white_space + "]+").r
|
|
67111
42f290d8ccbd
more accurate JSON parsing according to http://seriot.ch/parsing_json.php
wenzelm
parents:
66928
diff
changeset
|
64 |
def whitespace: Parser[Any] = many(character(white_space.contains(_))) |
| 66922 | 65 |
|
| 71601 | 66 |
val letter: Parser[String] = one(character(Symbol.is_ascii_letter)) |
67 |
val letters1: Parser[String] = many1(character(Symbol.is_ascii_letter)) |
|
| 66922 | 68 |
|
| 71601 | 69 |
def digits: Parser[String] = many(character(Symbol.is_ascii_digit)) |
70 |
def digits1: Parser[String] = many1(character(Symbol.is_ascii_digit)) |
|
| 66922 | 71 |
|
72 |
||
73 |
/* keyword */ |
|
74 |
||
75 |
def keyword: Parser[Token] = |
|
| 66924 | 76 |
(letters1 | one(character("{}[],:".contains(_)))) ^^ (s => Token(Kind.KEYWORD, s))
|
| 66922 | 77 |
|
78 |
||
79 |
/* string */ |
|
80 |
||
81 |
def string: Parser[Token] = |
|
82 |
'\"' ~> rep(string_body) <~ '\"' ^^ (cs => Token(Kind.STRING, cs.mkString)) |
|
83 |
||
84 |
def string_body: Parser[Char] = |
|
|
67111
42f290d8ccbd
more accurate JSON parsing according to http://seriot.ch/parsing_json.php
wenzelm
parents:
66928
diff
changeset
|
85 |
elem("", c => c > '\u001f' && c != '\"' && c != '\\' && c != EofCh) | '\\' ~> string_escape
|
| 66922 | 86 |
|
87 |
def string_escape: Parser[Char] = |
|
88 |
elem("", "\"\\/".contains(_)) |
|
|
89 |
elem("", "bfnrt".contains(_)) ^^
|
|
90 |
{ case 'b' => '\b' case 'f' => '\f' case 'n' => '\n' case 'r' => '\r' case 't' => '\t' } |
|
|
91 |
'u' ~> repeated(character("0123456789abcdefABCDEF".contains(_)), 4, 4) ^^
|
|
92 |
(s => Integer.parseInt(s, 16).toChar) |
|
93 |
||
94 |
def string_failure: Parser[Token] = '\"' ~> failure("Unterminated string")
|
|
95 |
||
96 |
||
97 |
/* number */ |
|
98 |
||
99 |
def number: Parser[Token] = |
|
|
67111
42f290d8ccbd
more accurate JSON parsing according to http://seriot.ch/parsing_json.php
wenzelm
parents:
66928
diff
changeset
|
100 |
opt("-") ~ number_body ~ opt(letter) ^^ {
|
| 66922 | 101 |
case a ~ b ~ None => Token(Kind.NUMBER, a.getOrElse("") + b)
|
102 |
case a ~ b ~ Some(c) => |
|
103 |
errorToken("Invalid number format: " + quote(a.getOrElse("") + b + c))
|
|
104 |
} |
|
105 |
||
106 |
def number_body: Parser[String] = |
|
107 |
(zero | positive) ~ opt(number_fract) ~ opt(number_exp) ^^ |
|
108 |
{ case a ~ b ~ c => a + b.getOrElse("") + c.getOrElse("") }
|
|
109 |
||
110 |
def number_fract: Parser[String] = "." ~ digits1 ^^ { case a ~ b => a + b }
|
|
111 |
||
112 |
def number_exp: Parser[String] = |
|
113 |
one(character("eE".contains(_))) ~ maybe(character("-+".contains(_))) ~ digits1 ^^
|
|
114 |
{ case a ~ b ~ c => a + b + c }
|
|
115 |
||
| 71601 | 116 |
def zero: Parser[String] = one(character(c => c == '0')) |
117 |
def nonzero: Parser[String] = one(character(c => c != '0' && Symbol.is_ascii_digit(c))) |
|
| 66922 | 118 |
def positive: Parser[String] = nonzero ~ digits ^^ { case a ~ b => a + b }
|
119 |
||
120 |
||
121 |
/* token */ |
|
122 |
||
123 |
def token: Parser[Token] = |
|
124 |
keyword | (string | (string_failure | (number | failure("Illegal character"))))
|
|
125 |
} |
|
126 |
||
127 |
||
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
128 |
/* parser */ |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
129 |
|
| 75405 | 130 |
trait Parsers extends combinator.Parsers {
|
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
131 |
type Elem = Token |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
132 |
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
133 |
def $$$(name: String): Parser[Token] = elem(name, _.is_keyword(name)) |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
134 |
def string: Parser[String] = elem("string", _.is_string) ^^ (_.text)
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
135 |
def number: Parser[Double] = elem("number", _.is_number) ^^ (tok => tok.text.toDouble)
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
136 |
|
| 67841 | 137 |
def json_object: Parser[Object.T] = |
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
138 |
$$$("{") ~>
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
139 |
repsep(string ~ ($$$(":") ~> json_value) ^^ { case a ~ b => (a, b) }, $$$(",")) <~
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
140 |
$$$("}") ^^ (_.toMap)
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
141 |
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
142 |
def json_array: Parser[List[T]] = |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
143 |
$$$("[") ~> repsep(json_value, $$$(",")) <~ $$$("]")
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
144 |
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
145 |
def json_value: Parser[T] = |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
146 |
json_object | (json_array | (number | (string | |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
147 |
($$$("true") ^^^ true | ($$$("false") ^^^ false | ($$$("null") ^^^ null))))))
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
148 |
|
| 75393 | 149 |
def parse(input: CharSequence, strict: Boolean): T = {
|
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
150 |
val scanner = new Lexer.Scanner(Scan.char_reader(input)) |
| 75420 | 151 |
val result = phrase(if (strict) json_object | json_array else json_value)(scanner) |
152 |
(result : @unchecked) match {
|
|
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
153 |
case Success(json, _) => json |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
154 |
case NoSuccess(_, next) => error("Malformed JSON input at " + next.pos)
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
155 |
} |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
156 |
} |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
157 |
} |
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
158 |
|
| 75405 | 159 |
object Parsers extends Parsers |
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
160 |
|
|
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
161 |
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
162 |
/* standard format */ |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
163 |
|
| 75405 | 164 |
def parse(s: S, strict: Boolean = true): T = Parsers.parse(s, strict) |
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
165 |
|
| 75393 | 166 |
object Format {
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
167 |
def unapply(s: S): Option[T] = |
|
66925
8b117dc73278
separate JSON parser, which is slightly more general than (deprecated) scala.util.parsing.json.JSON;
wenzelm
parents:
66924
diff
changeset
|
168 |
try { Some(parse(s, strict = false)) }
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
169 |
catch { case ERROR(_) => None }
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
170 |
|
| 75957 | 171 |
private def output_string(s: String, result: StringBuilder): Unit = {
|
172 |
result += '"' |
|
173 |
result ++= |
|
174 |
s.iterator.map {
|
|
175 |
case '"' => "\\\"" |
|
176 |
case '\\' => "\\\\" |
|
177 |
case '\b' => "\\b" |
|
178 |
case '\f' => "\\f" |
|
179 |
case '\n' => "\\n" |
|
180 |
case '\r' => "\\r" |
|
181 |
case '\t' => "\\t" |
|
182 |
case c => |
|
183 |
if (c <= '\u001f' || c >= '\u007f' && c <= '\u009f') "\\u%04x".format(c.toInt) |
|
184 |
else c |
|
185 |
}.mkString |
|
186 |
result += '"' |
|
187 |
} |
|
188 |
||
189 |
private def output_atom(x: T, result: StringBuilder): Boolean = |
|
190 |
x match {
|
|
191 |
case null => result ++= "null"; true |
|
192 |
case _: Int | _ : Long | _: Boolean => result ++= x.toString; true |
|
193 |
case n: Double => |
|
194 |
val i = n.toLong |
|
195 |
result ++= (if (i.toDouble == n) i.toString else n.toString) |
|
196 |
true |
|
197 |
case s: String => output_string(s, result); true |
|
198 |
case _ => false |
|
199 |
} |
|
200 |
||
| 75393 | 201 |
def apply(json: T): S = {
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
202 |
val result = new StringBuilder |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
203 |
|
| 75957 | 204 |
def output(x: T): Unit = {
|
205 |
if (!output_atom(x, result)) {
|
|
206 |
x match {
|
|
207 |
case Object(obj) => |
|
208 |
result += '{'
|
|
209 |
Library.separate(None, obj.toList.map(Some(_))).foreach({
|
|
210 |
case None => result += ',' |
|
211 |
case Some((x, y)) => |
|
212 |
output_string(x, result) |
|
213 |
result += ':' |
|
214 |
output(y) |
|
215 |
}) |
|
216 |
result += '}' |
|
217 |
case list: List[T] => |
|
218 |
result += '[' |
|
219 |
Library.separate(None, list.map(Some(_))).foreach({
|
|
220 |
case None => result += ',' |
|
221 |
case Some(x) => output(x) |
|
222 |
}) |
|
223 |
result += ']' |
|
224 |
case _ => error("Bad JSON value: " + x.toString)
|
|
225 |
} |
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
226 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
227 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
228 |
|
| 75957 | 229 |
output(json) |
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
230 |
result.toString |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
231 |
} |
| 75959 | 232 |
|
233 |
private def pretty_atom(x: T): Option[XML.Tree] = {
|
|
234 |
val result = new StringBuilder |
|
235 |
val ok = output_atom(x, result) |
|
236 |
if (ok) Some(XML.Text(result.toString)) else None |
|
237 |
} |
|
238 |
||
239 |
private def pretty_string(s: String): XML.Tree = {
|
|
240 |
val result = new StringBuilder |
|
241 |
output_string(s, result) |
|
242 |
XML.Text(result.toString) |
|
243 |
} |
|
244 |
||
245 |
private def pretty_tree(x: T): XML.Tree = |
|
246 |
x match {
|
|
247 |
case Object(obj) => |
|
248 |
Pretty.`enum`( |
|
249 |
for ((x, y) <- obj.toList) |
|
250 |
yield Pretty.block(List(pretty_string(x), XML.Text(":"), Pretty.brk(1), pretty(y))),
|
|
251 |
bg = "{", en = "}", indent = 1)
|
|
252 |
case list: List[T] => |
|
253 |
Pretty.`enum`(list.map(pretty), bg = "[", en = "]", indent = 1) |
|
254 |
case _ => error("Bad JSON value: " + x.toString)
|
|
255 |
} |
|
256 |
||
257 |
def pretty(x: T): XML.Tree = pretty_atom(x) getOrElse pretty_tree(x) |
|
258 |
||
| 75960 | 259 |
def pretty_print(x: T, margin: Int = Pretty.default_margin.toInt): JSON.S = |
| 75959 | 260 |
Pretty.string_of(List(pretty(x)), margin = margin.toDouble) |
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
261 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
262 |
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
263 |
|
| 67843 | 264 |
/* typed values */ |
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
265 |
|
| 75393 | 266 |
object Value {
|
267 |
object UUID {
|
|
| 69458 | 268 |
def unapply(json: T): Option[isabelle.UUID.T] = |
| 67885 | 269 |
json match {
|
| 69458 | 270 |
case x: java.lang.String => isabelle.UUID.unapply(x) |
| 67885 | 271 |
case _ => None |
272 |
} |
|
273 |
} |
|
274 |
||
| 75393 | 275 |
object String {
|
| 67842 | 276 |
def unapply(json: T): Option[java.lang.String] = |
277 |
json match {
|
|
278 |
case x: java.lang.String => Some(x) |
|
279 |
case _ => None |
|
280 |
} |
|
281 |
} |
|
282 |
||
| 75393 | 283 |
object String0 {
|
| 71305 | 284 |
def unapply(json: T): Option[java.lang.String] = |
285 |
json match {
|
|
286 |
case null => Some("")
|
|
287 |
case x: java.lang.String => Some(x) |
|
288 |
case _ => None |
|
289 |
} |
|
290 |
} |
|
291 |
||
| 75393 | 292 |
object Double {
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
293 |
def unapply(json: T): Option[scala.Double] = |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
294 |
json match {
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
295 |
case x: scala.Double => Some(x) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
296 |
case x: scala.Long => Some(x.toDouble) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
297 |
case x: scala.Int => Some(x.toDouble) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
298 |
case _ => None |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
299 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
300 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
301 |
|
| 75393 | 302 |
object Long {
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
303 |
def unapply(json: T): Option[scala.Long] = |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
304 |
json match {
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
305 |
case x: scala.Double if x.toLong.toDouble == x => Some(x.toLong) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
306 |
case x: scala.Long => Some(x) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
307 |
case x: scala.Int => Some(x.toLong) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
308 |
case _ => None |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
309 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
310 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
311 |
|
| 75393 | 312 |
object Int {
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
313 |
def unapply(json: T): Option[scala.Int] = |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
314 |
json match {
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
315 |
case x: scala.Double if x.toInt.toDouble == x => Some(x.toInt) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
316 |
case x: scala.Long if x.toInt.toLong == x => Some(x.toInt) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
317 |
case x: scala.Int => Some(x) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
318 |
case _ => None |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
319 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
320 |
} |
| 67842 | 321 |
|
| 75393 | 322 |
object Boolean {
|
| 67842 | 323 |
def unapply(json: T): Option[scala.Boolean] = |
324 |
json match {
|
|
325 |
case x: scala.Boolean => Some(x) |
|
326 |
case _ => None |
|
327 |
} |
|
328 |
} |
|
| 67843 | 329 |
|
| 75393 | 330 |
object List {
|
| 67843 | 331 |
def unapply[A](json: T, unapply: T => Option[A]): Option[List[A]] = |
332 |
json match {
|
|
333 |
case xs: List[T] => |
|
334 |
val ys = xs.map(unapply) |
|
335 |
if (ys.forall(_.isDefined)) Some(ys.map(_.get)) else None |
|
336 |
case _ => None |
|
337 |
} |
|
338 |
} |
|
| 68943 | 339 |
|
| 75393 | 340 |
object Seconds {
|
| 68943 | 341 |
def unapply(json: T): Option[Time] = |
| 71601 | 342 |
Double.unapply(json).map(Time.seconds) |
| 68943 | 343 |
} |
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
344 |
} |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
345 |
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
346 |
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
347 |
/* object values */ |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
348 |
|
| 67841 | 349 |
def optional(entry: (String, Option[T])): Object.T = |
| 64878 | 350 |
entry match {
|
| 67850 | 351 |
case (name, Some(x)) => Object(name -> x) |
| 67841 | 352 |
case (_, None) => Object.empty |
| 64878 | 353 |
} |
354 |
||
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
355 |
def value(obj: T, name: String): Option[T] = |
| 63644 | 356 |
obj match {
|
| 67841 | 357 |
case Object(m) => m.get(name) |
| 63644 | 358 |
case _ => None |
359 |
} |
|
360 |
||
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
361 |
def value[A](obj: T, name: String, unapply: T => Option[A]): Option[A] = |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
362 |
for {
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
363 |
json <- value(obj, name) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
364 |
x <- unapply(json) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
365 |
} yield x |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
366 |
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
367 |
def array(obj: T, name: String): Option[List[T]] = |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
368 |
value(obj, name) match {
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
369 |
case Some(a: List[T]) => Some(a) |
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
370 |
case _ => None |
| 63644 | 371 |
} |
372 |
||
| 67864 | 373 |
def value_default[A](obj: T, name: String, unapply: T => Option[A], default: => A): Option[A] = |
| 67843 | 374 |
value(obj, name) match {
|
375 |
case None => Some(default) |
|
376 |
case Some(json) => unapply(json) |
|
377 |
} |
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
378 |
|
| 69458 | 379 |
def uuid(obj: T, name: String): Option[UUID.T] = |
| 67885 | 380 |
value(obj, name, Value.UUID.unapply) |
381 |
||
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
382 |
def string(obj: T, name: String): Option[String] = |
| 67842 | 383 |
value(obj, name, Value.String.unapply) |
| 67864 | 384 |
def string_default(obj: T, name: String, default: => String = ""): Option[String] = |
| 67843 | 385 |
value_default(obj, name, Value.String.unapply, default) |
| 63644 | 386 |
|
| 71305 | 387 |
def string0(obj: T, name: String): Option[String] = |
388 |
value(obj, name, Value.String0.unapply) |
|
389 |
def string0_default(obj: T, name: String, default: => String = ""): Option[String] = |
|
390 |
value_default(obj, name, Value.String0.unapply, default) |
|
391 |
||
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
392 |
def double(obj: T, name: String): Option[Double] = |
| 67842 | 393 |
value(obj, name, Value.Double.unapply) |
| 67864 | 394 |
def double_default(obj: T, name: String, default: => Double = 0.0): Option[Double] = |
| 67843 | 395 |
value_default(obj, name, Value.Double.unapply, default) |
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
396 |
|
|
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
397 |
def long(obj: T, name: String): Option[Long] = |
| 67842 | 398 |
value(obj, name, Value.Long.unapply) |
| 67864 | 399 |
def long_default(obj: T, name: String, default: => Long = 0): Option[Long] = |
| 67843 | 400 |
value_default(obj, name, Value.Long.unapply, default) |
| 63644 | 401 |
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
402 |
def int(obj: T, name: String): Option[Int] = |
| 67842 | 403 |
value(obj, name, Value.Int.unapply) |
| 67864 | 404 |
def int_default(obj: T, name: String, default: => Int = 0): Option[Int] = |
| 67843 | 405 |
value_default(obj, name, Value.Int.unapply, default) |
| 63644 | 406 |
|
|
64545
25045094d7bb
clarified JSON operations (see isabelle_vscode/a7931dc2a1ab);
wenzelm
parents:
63992
diff
changeset
|
407 |
def bool(obj: T, name: String): Option[Boolean] = |
| 67842 | 408 |
value(obj, name, Value.Boolean.unapply) |
| 67864 | 409 |
def bool_default(obj: T, name: String, default: => Boolean = false): Option[Boolean] = |
| 67843 | 410 |
value_default(obj, name, Value.Boolean.unapply, default) |
411 |
||
412 |
def list[A](obj: T, name: String, unapply: T => Option[A]): Option[List[A]] = |
|
413 |
value(obj, name, Value.List.unapply(_, unapply)) |
|
| 67864 | 414 |
def list_default[A](obj: T, name: String, unapply: T => Option[A], default: => List[A] = Nil) |
| 67843 | 415 |
: Option[List[A]] = value_default(obj, name, Value.List.unapply(_, unapply), default) |
| 68742 | 416 |
|
417 |
def strings(obj: T, name: String): Option[List[String]] = |
|
| 71601 | 418 |
list(obj, name, JSON.Value.String.unapply) |
| 68742 | 419 |
def strings_default(obj: T, name: String, default: => List[String] = Nil): Option[List[String]] = |
| 71601 | 420 |
list_default(obj, name, JSON.Value.String.unapply, default) |
| 68943 | 421 |
|
422 |
def seconds(obj: T, name: String): Option[Time] = |
|
423 |
value(obj, name, Value.Seconds.unapply) |
|
424 |
def seconds_default(obj: T, name: String, default: => Time = Time.zero): Option[Time] = |
|
425 |
value_default(obj, name, Value.Seconds.unapply, default) |
|
| 63644 | 426 |
} |