Skip to content

Commit a206801

Browse files
authored
fix json parser example (#3)
* fix json string parser escapes * fix number parsing in json example
1 parent e933ca4 commit a206801

File tree

1 file changed

+35
-15
lines changed

1 file changed

+35
-15
lines changed

test/json_test.gleam

+35-15
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
import gleam/float
22
import gleam/int
3+
import gleam/option.{None, Some}
4+
import gleam/result
35
import gleam/string
46
import pears.{type Parser}
57
import pears/chars.{type Char, digit, string}
@@ -40,34 +42,41 @@ fn value_parser() -> Parser(Char, Json) {
4042
digits0,
4143
)),
4244
)
45+
|> map(string.concat)
4346

4447
let frac =
4548
just(".")
4649
|> right(digits1)
50+
|> map(string.concat)
4751

4852
let exp =
4953
alt(just("e"), just("E"))
5054
|> pair(maybe(one_of(["+", "-"])))
5155
|> pair(digits1)
56+
|> recognize()
57+
|> map(string.concat)
5258

5359
let num =
5460
maybe(just("-"))
5561
|> pair(whole)
5662
|> pair(maybe(frac))
5763
|> pair(maybe(exp))
58-
|> recognize()
59-
|> map(fn(chars) {
60-
let str = string.concat(chars)
61-
let number = float.parse(str)
62-
63-
case number {
64-
Ok(num) -> Num(num)
65-
Error(_) -> {
66-
let assert Ok(number) = int.parse(str)
67-
Num(int.to_float(number))
68-
}
69-
}
64+
|> map(fn(p) {
65+
let #(#(#(neg, whole), fraction), ex) = p
66+
let str =
67+
option.unwrap(neg, "")
68+
<> whole
69+
<> "."
70+
<> option.unwrap(fraction, "0")
71+
<> option.unwrap(ex, "")
72+
str
73+
|> float.parse()
74+
|> result.unwrap(case neg {
75+
Some(_) -> -1.7976931348623158e308
76+
None -> 1.7976931348623158e308
77+
})
7078
})
79+
|> map(Num)
7180

7281
let bool =
7382
alt(to(string("true"), Boolean(True)), to(string("false"), Boolean(False)))
@@ -104,7 +113,7 @@ fn value_parser() -> Parser(Char, Json) {
104113
)
105114

106115
let str =
107-
none_of(["\""])
116+
none_of(["\"", "\\"])
108117
|> alt(escape)
109118
|> many0()
110119
|> map(string.concat)
@@ -145,6 +154,16 @@ pub fn parse_numbers_test() {
145154
json_parser()
146155
|> should_parse("4.2", Num(4.2))
147156
|> should_parse("42", Num(42.0))
157+
|> should_parse("0", Num(0.0))
158+
|> should_parse("-0", Num(-0.0))
159+
|> should_parse("1e10", Num(1.0e10))
160+
}
161+
162+
pub fn parse_large_floats() {
163+
// floats which are too large to be represented should be parsed as the max or min representable float
164+
json_parser()
165+
|> should_parse("1e1000", Num(1.7976931348623158e308))
166+
|> should_parse("-1e1000", Num(-1.7976931348623158e308))
148167
}
149168

150169
pub fn parse_booleans_test() {
@@ -161,8 +180,9 @@ pub fn parse_null_test() {
161180
pub fn parse_strings_test() {
162181
json_parser()
163182
|> should_parse("\"hello\"", Str("hello"))
164-
// |> should_parse("\"hello\\nworld\"", Str("hello\nworld"))
165-
// |> should_parse("\"\\u0048\\u0065\\u006c\\u006c\\u006f\"", Str("Hello"))
183+
|> should_parse("\"hello\\nworld\"", Str("hello\nworld"))
184+
|> should_parse("\"\\u0048\\u0065\\u006c\\u006c\\u006f\"", Str("Hello"))
185+
|> should_parse("\"\\\"\"", Str("\""))
166186
}
167187

168188
pub fn parse_arrays_test() {

0 commit comments

Comments
 (0)