@@ -3,7 +3,7 @@ use crate::input::Input;
3
3
use crate :: value:: { Number , Position , SpannedValue , Value } ;
4
4
use nom:: bytes:: complete:: take_till;
5
5
use nom:: character:: complete:: digit0;
6
- use nom:: combinator:: eof;
6
+ use nom:: combinator:: { eof, opt } ;
7
7
use nom:: error:: ParseError ;
8
8
use nom:: multi:: many_till;
9
9
use nom:: {
39
39
I : Clone ,
40
40
{
41
41
move |i : I | {
42
- let i = i. clone ( ) ;
43
42
let result = parser. parse ( i. clone ( ) ) ;
44
43
45
44
match result {
49
48
}
50
49
}
51
50
51
+ pub fn map_parser < I , O1 , O2 , E : ParseError < I > , F , G > (
52
+ mut parser : F ,
53
+ mut applied_parser : G ,
54
+ ) -> impl FnMut ( I ) -> IResult < I , O2 , E >
55
+ where
56
+ F : Parser < I , O1 , E > ,
57
+ G : FnMut ( ( I , O1 ) ) -> IResult < I , O2 , E > ,
58
+ {
59
+ move |input : I | applied_parser ( parser. parse ( input) ?)
60
+ }
61
+
52
62
fn parse_true ( i : Span ) -> Result < bool > {
53
63
value ( true , tag ( "rue" ) ) ( i) . or_else ( |_: Err < Error > | {
54
64
let start = Position :: from_ahead ( i) ;
@@ -278,7 +288,16 @@ fn array(i: Span) -> Result<Vec<SpannedValue>> {
278
288
}
279
289
} ) ,
280
290
) ,
281
- json_value,
291
+ or_else ( json_value, |e : Err < Error > , i| {
292
+ // If it succeeds, it means that it's a trailing comma
293
+ let _ = preceded ( multispace0, char ( ']' ) ) ( i) . map_err ( |_: Err < Error > | e) ?;
294
+
295
+ Err ( Err :: Failure ( Error :: new (
296
+ Position :: from_ahead ( i) ,
297
+ Position :: from_ahead ( i) ,
298
+ Kind :: TrailingComma ,
299
+ ) ) )
300
+ } ) ,
282
301
) ,
283
302
preceded (
284
303
multispace0,
@@ -298,23 +317,27 @@ fn array(i: Span) -> Result<Vec<SpannedValue>> {
298
317
}
299
318
300
319
fn key_value ( i : Span < ' _ > ) -> Result < ( String , SpannedValue ) > {
320
+ let ( i, comma) = opt ( char ( ',' ) ) ( i) ?;
321
+
301
322
let pos_before_space = Position :: from ( i) ;
302
323
303
324
let ( i, _) = multispace0 ( i) ?;
304
325
305
- if i. starts_with ( '}' ) || i. is_empty ( ) {
326
+ if ( i. starts_with ( '}' ) || i. is_empty ( ) ) && comma . is_none ( ) {
306
327
// Key value is called in a loop, and only an error can stop it
307
328
return Err ( Err :: Error ( Error :: default ( ) ) ) ;
308
329
}
309
330
310
331
let ( i, key) = preceded ( char ( '"' ) , string) ( i) . or_else ( |e| match e {
311
332
Err :: Error ( mut e) => {
312
333
let ( i, key) = take_until_delimiter ( i, true ) ?;
334
+
335
+ let end = Position :: from_ahead ( i) ;
336
+
313
337
if key. is_empty ( ) {
314
- e. start = pos_before_space;
338
+ e. start = pos_before_space. clone ( ) ;
315
339
}
316
340
e. kind = Kind :: InvalidKey ( key) ;
317
- let end = Position :: from_ahead ( i) ;
318
341
e. end = end;
319
342
320
343
Err ( Err :: Failure ( e) )
@@ -341,25 +364,41 @@ fn key_value(i: Span<'_>) -> Result<(String, SpannedValue)> {
341
364
fn hash ( i : Span < ' _ > ) -> Result < HashMap < String , SpannedValue > > {
342
365
let start = Position :: from_ahead ( i) ;
343
366
344
- let result = terminated (
367
+ let result: Result < HashMap < String , SpannedValue > > = terminated (
345
368
map (
346
369
separated_list0 (
347
370
preceded (
348
371
multispace0,
349
- or_else ( char ( ',' ) , |e : Err < Error > , i| {
350
- let ( i, _) = multispace0 ( i) ?;
351
-
352
- match e {
353
- Err :: Error ( mut e) if !i. is_empty ( ) && !i. starts_with ( '}' ) => {
354
- e. kind = Kind :: MissingComma ;
355
- e. start = start. clone ( ) ;
356
- e. end . col -= 1 ;
357
-
358
- Err ( Err :: Failure ( e) )
372
+ or_else (
373
+ map_parser ( char ( ',' ) , |( i, _) : ( Span , char ) | {
374
+ let ( j, _) = multispace0 ( i) ?;
375
+
376
+ if j. starts_with ( '}' ) {
377
+ let position = Position :: from_ahead ( i) ;
378
+ Err ( Err :: Failure ( Error :: new (
379
+ position. clone ( ) ,
380
+ position,
381
+ Kind :: TrailingComma ,
382
+ ) ) )
383
+ } else {
384
+ Ok ( ( i, ',' ) )
359
385
}
360
- e => Err ( e) ,
361
- }
362
- } ) ,
386
+ } ) ,
387
+ |e : Err < Error > , i| {
388
+ let ( i, _) = multispace0 ( i) ?;
389
+
390
+ match e {
391
+ Err :: Error ( mut e) if !i. is_empty ( ) && !i. starts_with ( '}' ) => {
392
+ e. kind = Kind :: MissingComma ;
393
+ e. start = start. clone ( ) ;
394
+ e. end . col -= 1 ;
395
+
396
+ Err ( Err :: Failure ( e) )
397
+ }
398
+ e => Err ( e) ,
399
+ }
400
+ } ,
401
+ ) ,
363
402
) ,
364
403
key_value,
365
404
) ,
0 commit comments