@@ -3,10 +3,10 @@ import gleam/int
3
3
import gleam/string
4
4
import gleeunit/should
5
5
import pears . { type ParseResult , type Parser , Parsed }
6
- import pears/chars . { type Char , char , digit , string }
6
+ import pears/chars . { type Char , digit , string }
7
7
import pears/combinators . {
8
8
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 ,
10
10
}
11
11
import gleam/dict . { type Dict }
12
12
@@ -74,13 +74,41 @@ fn value_parser() -> Parser(Char, Json) {
74
74
75
75
let null = to ( string ( "null" ) , Null )
76
76
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 ( "\" " ) )
84
112
85
113
let value = lazy ( value_parser )
86
114
@@ -90,7 +118,7 @@ fn value_parser() -> Parser(Char, Json) {
90
118
|> map ( Array )
91
119
92
120
let key_value =
93
- str ( )
121
+ str
94
122
|> left ( symbol ( ":" ) )
95
123
|> pair ( value )
96
124
@@ -104,7 +132,7 @@ fn value_parser() -> Parser(Char, Json) {
104
132
|> between ( symbol ( "{" ) , symbol ( "}" ) )
105
133
|> map ( Obj )
106
134
107
- choice ( [ num , bool , null , map ( str ( ) , Str ) , array , obj ] )
135
+ choice ( [ num , bool , null , map ( str , Str ) , array , obj ] )
108
136
|> padded ( )
109
137
}
110
138
0 commit comments