1
  
2
  
3
  
4
  
5
  
6
  
7
  
8
  
9
  
10
  
11
  
12
  
13
  
14
  
15
  
16
  
17
  
18
  
19
  
20
  
21
  
22
  
23
  
24
  
25
  
26
  
27
  
28
  
29
  
30
  
31
  
32
  
33
  
34
  
35
  
36
  
37
  
38
  
39
  
40
  
41
  
42
  
43
  
44
  
45
  
46
  
47
  
48
  
49
  
50
  
51
  
52
  
53
  
54
  
55
  
56
  
57
  
58
  
59
  
60
  
61
  
62
  
63
  
64
  
65
  
66
  
67
  
68
  
69
  
70
  
71
  
72
  
73
  
74
  
75
  
76
  
77
  
78
  
79
  
80
  
81
  
82
  
83
  
84
  
85
  
86
  
87
  
88
  
89
  
90
  
91
  
92
  
// $Id: JSON.pmod.in,v 1.4 2010/06/10 15:19:51 arne Exp $ 
 
#pike __REAL_VERSION__ 
 
//! @ignore 
#if !constant (@module@) 
constant this_program_does_not_exist = 1; 
#else 
inherit @module@; 
//! @endignore 
 
//! @appears Standards.JSON module 
//! 
//! Tools for handling the JSON structured data format. See 
//! @url{http://www.json.org/@} and RFC 4627. 
 
True true = True(); 
False false = False(); 
Null null = Null(); 
//! Objects representing the three JSON literals @expr{true@}, 
//! @expr{false@} and @expr{null@}. @expr{true@} and @expr{false@} 
//! should behave as expected in boolean contexts. @expr{null@} is 
//! false in a boolean context. 
 
class True 
//! Type for the @[true] object. 
{ 
  string encode_json() {return "true";} 
  protected string _sprintf (int flag) {return flag == 'O' && "JSON.true";} 
} 
 
class False 
//! Type for the @[false] object. 
{ 
  int `!() {return 1;} 
  string encode_json() {return "false";} 
  protected string _sprintf (int flag) {return flag == 'O' && "JSON.false";} 
} 
 
class Null 
//! Type for the @[null] object. 
{ 
  int `!() {return 1;} 
  string encode_json() {return "null";} 
  protected string _sprintf (int flag) {return flag == 'O' && "JSON.null";} 
} 
 
//! Error thrown when JSON decode fails. 
class DecodeError { 
    inherit Error.Generic; 
 
    constant error_type = "json_decode"; 
    constant is_json_decode_error = 1; 
 
    //! The string that failed to be decoded. 
    string err_str; 
 
    //! The failing position in @[err_str]. 
    int err_pos; 
 
    protected void create(string err_str, int err_pos, void|string reason, void|array bt) { 
        this_program::err_str = err_str; 
        this_program::err_pos = err_pos; 
 
        string pre_context = err_pos > 13 ? 
            sprintf ("...%O", err_str[err_pos - 10..err_pos - 1]) : 
            err_pos > 0 ? 
            sprintf ("%O", err_str[..err_pos - 1]) : 
            ""; 
        string post_context = err_pos < sizeof (err_str) - 14 ? 
            sprintf ("%O...", err_str[err_pos + 1..err_pos + 10]) : 
            err_pos + 1 < sizeof (err_str) ? 
            sprintf ("%O", err_str[err_pos + 1..]) : 
            ""; 
        err_str = sprintf ("%s->[%c]<-%s", pre_context, err_str[err_pos], post_context); 
 
        if (reason) 
            ::create(sprintf("Error decoding JSON at position %d %s: %s.\n", 
                             err_pos, err_str, reason), bt); 
        else  
            ::create(sprintf("Error decoding JSON at position %d %s\n", 
                             err_pos, err_str), bt); 
    } 
} 
 
void decode_error(string err_str, int err_pos, void|string reason, void|mixed ... args) { 
    if (sizeof(args)) reason = sprintf(reason, @args); 
    throw(DecodeError(err_str, err_pos, reason, backtrace()[..<1])); 
} 
 
#endif  // constant (@module@)