1
1
import gleam/float
2
2
import gleam/int
3
+ import gleam/option . { None , Some }
4
+ import gleam/result
3
5
import gleam/string
4
6
import pears . { type Parser }
5
7
import pears/chars . { type Char , digit , string }
@@ -40,34 +42,41 @@ fn value_parser() -> Parser(Char, Json) {
40
42
digits0 ,
41
43
) ) ,
42
44
)
45
+ |> map ( string . concat )
43
46
44
47
let frac =
45
48
just ( "." )
46
49
|> right ( digits1 )
50
+ |> map ( string . concat )
47
51
48
52
let exp =
49
53
alt ( just ( "e" ) , just ( "E" ) )
50
54
|> pair ( maybe ( one_of ( [ "+" , "-" ] ) ) )
51
55
|> pair ( digits1 )
56
+ |> recognize ( )
57
+ |> map ( string . concat )
52
58
53
59
let num =
54
60
maybe ( just ( "-" ) )
55
61
|> pair ( whole )
56
62
|> pair ( maybe ( frac ) )
57
63
|> 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
+ } )
70
78
} )
79
+ |> map ( Num )
71
80
72
81
let bool =
73
82
alt ( to ( string ( "true" ) , Boolean ( True ) ) , to ( string ( "false" ) , Boolean ( False ) ) )
@@ -104,7 +113,7 @@ fn value_parser() -> Parser(Char, Json) {
104
113
)
105
114
106
115
let str =
107
- none_of ( [ "\" " ] )
116
+ none_of ( [ "\" " , " \\ " ] )
108
117
|> alt ( escape )
109
118
|> many0 ( )
110
119
|> map ( string . concat )
@@ -145,6 +154,16 @@ pub fn parse_numbers_test() {
145
154
json_parser ( )
146
155
|> should_parse ( "4.2" , Num ( 4.2 ) )
147
156
|> 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 ) )
148
167
}
149
168
150
169
pub fn parse_booleans_test ( ) {
@@ -161,8 +180,9 @@ pub fn parse_null_test() {
161
180
pub fn parse_strings_test ( ) {
162
181
json_parser ( )
163
182
|> 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\n world" ) )
184
+ |> should_parse ( "\" \\ u0048\\ u0065\\ u006c\\ u006c\\ u006f\" " , Str ( "Hello" ) )
185
+ |> should_parse ( "\" \\ \" \" " , Str ( "\" " ) )
166
186
}
167
187
168
188
pub fn parse_arrays_test ( ) {
0 commit comments