Skip to content

Commit f294721

Browse files
committed
finish json parser example
1 parent acb53c7 commit f294721

File tree

1 file changed

+39
-11
lines changed

1 file changed

+39
-11
lines changed

test/json_test.gleam

+39-11
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,10 @@ import gleam/int
33
import gleam/string
44
import gleeunit/should
55
import pears.{type ParseResult, type Parser, Parsed}
6-
import pears/chars.{type Char, char, digit, string}
6+
import pears/chars.{type Char, digit, string}
77
import pears/combinators.{
88
alt, between, choice, eof, just, lazy, left, many0, map, maybe, none_of,
9-
one_of, pair, recognize, right, sep_by0, to,
9+
one_of, pair, recognize, right, sep_by0, seq, to,
1010
}
1111
import gleam/dict.{type Dict}
1212

@@ -74,13 +74,41 @@ fn value_parser() -> Parser(Char, Json) {
7474

7575
let null = to(string("null"), Null)
7676

77-
let str = fn() -> Parser(_, String) {
78-
let quote = char("\"")
79-
let value =
80-
many0(none_of(["\""]))
81-
|> map(string.concat)
82-
between(value, quote, quote)
83-
}
77+
let hex_digit =
78+
one_of([
79+
"0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e",
80+
"f", "A", "B", "C", "D", "E", "F",
81+
])
82+
83+
let unicode_escape_digits =
84+
recognize(seq([hex_digit, hex_digit, hex_digit, hex_digit]))
85+
86+
let escape =
87+
just("\\")
88+
|> right(
89+
choice([
90+
just("\\"),
91+
just("/"),
92+
just("\""),
93+
to(just("b"), "\u{0008}"),
94+
to(just("f"), "\u{000C}"),
95+
to(just("n"), "\n"),
96+
to(just("r"), "\r"),
97+
to(just("t"), "\t"),
98+
map(right(just("u"), unicode_escape_digits), fn(value) {
99+
let assert Ok(number) = int.base_parse(string.concat(value), 16)
100+
let assert Ok(codepoint) = string.utf_codepoint(number)
101+
string.from_utf_codepoints([codepoint])
102+
}),
103+
]),
104+
)
105+
106+
let str =
107+
none_of(["\""])
108+
|> alt(escape)
109+
|> many0()
110+
|> map(string.concat)
111+
|> between(just("\""), just("\""))
84112

85113
let value = lazy(value_parser)
86114

@@ -90,7 +118,7 @@ fn value_parser() -> Parser(Char, Json) {
90118
|> map(Array)
91119

92120
let key_value =
93-
str()
121+
str
94122
|> left(symbol(":"))
95123
|> pair(value)
96124

@@ -104,7 +132,7 @@ fn value_parser() -> Parser(Char, Json) {
104132
|> between(symbol("{"), symbol("}"))
105133
|> map(Obj)
106134

107-
choice([num, bool, null, map(str(), Str), array, obj])
135+
choice([num, bool, null, map(str, Str), array, obj])
108136
|> padded()
109137
}
110138

0 commit comments

Comments
 (0)