line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
#include "perl-jsonsl.h" |
2
|
|
|
|
|
|
|
#include "jsonxs_inline.h" |
3
|
|
|
|
|
|
|
|
4
|
|
|
|
|
|
|
/* |
5
|
|
|
|
|
|
|
* JSON::SL, JSON::SL::Boolean |
6
|
|
|
|
|
|
|
*/ |
7
|
|
|
|
|
|
|
#define MY_CXT_KEY "JSON::SL::_guts" XS_VERSION |
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
typedef struct { |
10
|
|
|
|
|
|
|
PLJSONSL* quick; |
11
|
|
|
|
|
|
|
HV *stash_obj; |
12
|
|
|
|
|
|
|
HV *stash_boolean; |
13
|
|
|
|
|
|
|
HV *stash_tuba; |
14
|
|
|
|
|
|
|
} my_cxt_t; |
15
|
|
|
|
|
|
|
START_MY_CXT |
16
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
static int PLJSONSL_Escape_Table_dfl[0x80]; |
18
|
|
|
|
|
|
|
#define ESCTBL PLJSONSL_Escape_Table_dfl |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
#define PLJSONSL_CROAK_USAGE(m) \ |
21
|
|
|
|
|
|
|
die("JSON::SL: %s %s", GvNAME(CvGV(cv)), m) |
22
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
#ifdef PLJSONSL_HAVE_HV_COMMON |
25
|
|
|
|
|
|
|
#define storeget_he(pjsn, hv, buf, len, value, is_utf8) \ |
26
|
|
|
|
|
|
|
hv_common((HV*)(hv), NULL, /*KEYSV*/ \ |
27
|
|
|
|
|
|
|
buf, len, \ |
28
|
|
|
|
|
|
|
((is_utf8)?HVhek_UTF8:0), /*flags*/\ |
29
|
|
|
|
|
|
|
HV_FETCH_ISSTORE, /*action*/ \ |
30
|
|
|
|
|
|
|
value, \ |
31
|
|
|
|
|
|
|
0 /* HASH (tell perl to compute) */) |
32
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
#define delete_he(pjsn,hv,he) delete_he_THX(aTHX_ pjsn, hv, he) |
34
|
|
|
|
|
|
|
static void |
35
|
6
|
|
|
|
|
|
delete_he_THX(pTHX_ const PLJSONSL *dummy, HV* hv, HE* he) |
36
|
|
|
|
|
|
|
{ |
37
|
|
|
|
|
|
|
char *kbuf; |
38
|
|
|
|
|
|
|
STRLEN klen; |
39
|
|
|
|
|
|
|
int is_utf8; |
40
|
|
|
|
|
|
|
|
41
|
6
|
50
|
|
|
|
|
kbuf = HePV(he, klen); |
|
|
0
|
|
|
|
|
|
42
|
6
|
50
|
|
|
|
|
is_utf8 = HeUTF8(he); |
43
|
6
|
|
|
|
|
|
hv_common(hv, NULL, /* KEYSV */ |
44
|
|
|
|
|
|
|
kbuf, klen, |
45
|
|
|
|
|
|
|
(is_utf8 ? HVhek_UTF8 : 0), /*flags*/ |
46
|
|
|
|
|
|
|
HV_DELETE|G_DISCARD, /*action*/ |
47
|
|
|
|
|
|
|
NULL, /* value */ |
48
|
|
|
|
|
|
|
HeHASH(he) /*hash value - already computed */); |
49
|
6
|
|
|
|
|
|
} |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
#define PLJSONSL_INIT_KSV(blah) |
52
|
|
|
|
|
|
|
#define PLJSONSL_DESTROY_KSV(blah) |
53
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
|
55
|
|
|
|
|
|
|
#else |
56
|
|
|
|
|
|
|
/* probably very dangerous, but the beginning of hv_store_common |
57
|
|
|
|
|
|
|
* looks quite simple... |
58
|
|
|
|
|
|
|
*/ |
59
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
#define CLOBBER_PV(sv,buf,len) \ |
61
|
|
|
|
|
|
|
SvCUR_set(sv, len); \ |
62
|
|
|
|
|
|
|
SvPVX(sv) = buf; \ |
63
|
|
|
|
|
|
|
SvUTF8_off(sv); |
64
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
#define UNCLOBBER_PV(sv) \ |
66
|
|
|
|
|
|
|
SvCUR_set(sv, 0); \ |
67
|
|
|
|
|
|
|
SvPVX(sv) = NULL; \ |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
#define PLJSONSL_INIT_KSV(pjsn) \ |
70
|
|
|
|
|
|
|
pjsn->ksv = newSV(16); \ |
71
|
|
|
|
|
|
|
pjsn->ksv_origpv = SvPVX(pjsn->ksv); \ |
72
|
|
|
|
|
|
|
SvLEN_set(pjsn->ksv, 0); \ |
73
|
|
|
|
|
|
|
SvPOK_on(pjsn->ksv); |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
#define PLJSONSL_DESTROY_KSV(pjsn) \ |
76
|
|
|
|
|
|
|
CLOBBER_PV(pjsn->ksv, pjsn->ksv_origpv, 0); \ |
77
|
|
|
|
|
|
|
SvREFCNT_dec(pjsn->ksv); |
78
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
#define storeget_he(...) storeget_he_THX(aTHX_ __VA_ARGS__) |
80
|
|
|
|
|
|
|
static HE* |
81
|
|
|
|
|
|
|
storeget_he_THX(pTHX_ PLJSONSL *pjsn, HV *hv, const char *buf, size_t len, |
82
|
|
|
|
|
|
|
SV *value, int is_utf8) |
83
|
|
|
|
|
|
|
{ |
84
|
|
|
|
|
|
|
HE *ret; |
85
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
CLOBBER_PV(pjsn->ksv, buf, len); |
87
|
|
|
|
|
|
|
if (is_utf8) { |
88
|
|
|
|
|
|
|
SvUTF8_on(pjsn->ksv); |
89
|
|
|
|
|
|
|
} |
90
|
|
|
|
|
|
|
ret = hv_store_ent(hv, pjsn->ksv, value, 0); |
91
|
|
|
|
|
|
|
UNCLOBBER_PV(pjsn->ksv); |
92
|
|
|
|
|
|
|
return ret; |
93
|
|
|
|
|
|
|
} |
94
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
#define delete_he(...) delete_he_THX(aTHX_ __VA_ARGS__) |
96
|
|
|
|
|
|
|
static void |
97
|
|
|
|
|
|
|
delete_he_THX(pTHX_ PLJSONSL *pjsn, HV *hv, HE* he) |
98
|
|
|
|
|
|
|
{ |
99
|
|
|
|
|
|
|
char *kbuf; |
100
|
|
|
|
|
|
|
STRLEN klen; |
101
|
|
|
|
|
|
|
kbuf = HePV(he, klen); |
102
|
|
|
|
|
|
|
|
103
|
|
|
|
|
|
|
CLOBBER_PV(pjsn->ksv, buf, len); |
104
|
|
|
|
|
|
|
if (HeUTF8(he)) { |
105
|
|
|
|
|
|
|
SvUTF8_on(pjsn->ksv); |
106
|
|
|
|
|
|
|
} |
107
|
|
|
|
|
|
|
(void)hv_delete_ent(hv, pjsn->ksv, G_DISCARD, HeHASH(he)); |
108
|
|
|
|
|
|
|
UNCLOBBER_PV(pjsn->ksv); |
109
|
|
|
|
|
|
|
} |
110
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
#endif /* HAVE_HV_COMMON */ |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
#define REFDEC_FIELD(pjsn, fld) \ |
115
|
|
|
|
|
|
|
if (pjsn->fld != NULL) \ |
116
|
|
|
|
|
|
|
{ \ |
117
|
|
|
|
|
|
|
SvREFCNT_dec(pjsn->fld); \ |
118
|
|
|
|
|
|
|
pjsn->fld = NULL; \ |
119
|
|
|
|
|
|
|
} \ |
120
|
|
|
|
|
|
|
|
121
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
#define GET_STATE_BUFFER(pjsn, pos) \ |
124
|
|
|
|
|
|
|
(char*)(SvPVX(pjsn->buf) + (pos - pjsn->pos_min_valid)) |
125
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
#define PLJSONSL_NEWSVUV_fast(sv, val) \ |
127
|
|
|
|
|
|
|
sv = newSV(0); \ |
128
|
|
|
|
|
|
|
sv_upgrade(sv, SVt_IV); \ |
129
|
|
|
|
|
|
|
SvIOK_only(sv); \ |
130
|
|
|
|
|
|
|
SvUVX(sv) = val; |
131
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
/** |
133
|
|
|
|
|
|
|
* These 'common' functions are generic enough to work |
134
|
|
|
|
|
|
|
* on all objects wiht a common pjsn head. |
135
|
|
|
|
|
|
|
*/ |
136
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
#define pljsonsl_common_mknumeric(s,b,n) \ |
138
|
|
|
|
|
|
|
pljsonsl_common_mknumeric_THX(aTHX_ s,b,n) |
139
|
|
|
|
|
|
|
static SV * |
140
|
28
|
|
|
|
|
|
pljsonsl_common_mknumeric_THX(pTHX_ |
141
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
142
|
|
|
|
|
|
|
const char *buf, |
143
|
|
|
|
|
|
|
size_t nbuf) |
144
|
|
|
|
|
|
|
{ |
145
|
|
|
|
|
|
|
#define die_numeric(err) \ |
146
|
|
|
|
|
|
|
die("JSON::SL - Malformed number (%s)", err); |
147
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
SV *newsv; |
149
|
28
|
|
|
|
|
|
switch (state->special_flags) { |
150
|
|
|
|
|
|
|
/* Simple signed/unsigned numbers, no exponents or fractions to worry about */ |
151
|
|
|
|
|
|
|
case JSONSL_SPECIALf_UNSIGNED: |
152
|
15
|
100
|
|
|
|
|
if (nbuf == 1) { |
153
|
12
|
50
|
|
|
|
|
PLJSONSL_NEWSVUV_fast(newsv, state->nelem); |
154
|
12
|
|
|
|
|
|
break; |
155
|
|
|
|
|
|
|
} /* else, ndigits > 1 */ |
156
|
3
|
50
|
|
|
|
|
if (*buf == '0') { die_numeric("leading zero for non-fraction"); } |
157
|
3
|
50
|
|
|
|
|
if (nbuf < (UV_DIG-1)) { |
158
|
3
|
50
|
|
|
|
|
PLJSONSL_NEWSVUV_fast(newsv, state->nelem); |
159
|
3
|
|
|
|
|
|
break; |
160
|
|
|
|
|
|
|
} /* else, potential overflow */ |
161
|
0
|
|
|
|
|
|
newsv = jsonxs_inline_process_number(buf); |
162
|
0
|
|
|
|
|
|
break; |
163
|
|
|
|
|
|
|
|
164
|
|
|
|
|
|
|
case JSONSL_SPECIALf_SIGNED: |
165
|
1
|
|
|
|
|
|
nbuf--; |
166
|
1
|
50
|
|
|
|
|
if (nbuf == 0) { die_numeric("found lone '-'"); } |
167
|
1
|
50
|
|
|
|
|
if (nbuf > 1 && buf[1] == '0') { |
|
|
50
|
|
|
|
|
|
168
|
0
|
|
|
|
|
|
die_numeric("Leading 0 after '-'"); |
169
|
|
|
|
|
|
|
} |
170
|
1
|
50
|
|
|
|
|
if (nbuf < (IV_DIG-1)) { |
171
|
1
|
|
|
|
|
|
newsv = newSViv(-((IV)state->nelem)); |
172
|
1
|
|
|
|
|
|
break; |
173
|
|
|
|
|
|
|
} /*else */ |
174
|
0
|
|
|
|
|
|
newsv = jsonxs_inline_process_number(buf); |
175
|
0
|
|
|
|
|
|
break; |
176
|
|
|
|
|
|
|
|
177
|
|
|
|
|
|
|
default: |
178
|
12
|
50
|
|
|
|
|
if (state->special_flags & JSONSL_SPECIALf_NUMNOINT) { |
179
|
12
|
|
|
|
|
|
newsv = jsonxs_inline_process_number(buf); |
180
|
|
|
|
|
|
|
} |
181
|
12
|
|
|
|
|
|
break; |
182
|
|
|
|
|
|
|
} |
183
|
28
|
|
|
|
|
|
return newsv; |
184
|
|
|
|
|
|
|
#undef die_numeric |
185
|
|
|
|
|
|
|
} |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
#define pljsonsl_common_mkboolean(pjsn_head, value) \ |
188
|
|
|
|
|
|
|
pljsonsl_common_mkboolean_THX(aTHX_ pjsn_head, value) |
189
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
static SV * |
191
|
4
|
|
|
|
|
|
pljsonsl_common_mkboolean_THX(pTHX_ |
192
|
|
|
|
|
|
|
PLJSONSL *pjsn_head, |
193
|
|
|
|
|
|
|
jsonsl_special_t specialf) |
194
|
|
|
|
|
|
|
{ |
195
|
|
|
|
|
|
|
SV *retsv, *ivsv; |
196
|
4
|
|
|
|
|
|
ivsv = newSViv(specialf == JSONSL_SPECIALf_TRUE); |
197
|
4
|
|
|
|
|
|
retsv = newRV_noinc(ivsv); |
198
|
4
|
|
|
|
|
|
sv_bless(retsv, pjsn_head->stash_boolean); |
199
|
4
|
|
|
|
|
|
return retsv; |
200
|
|
|
|
|
|
|
} |
201
|
|
|
|
|
|
|
|
202
|
|
|
|
|
|
|
#define pljsonsl_common_initialize(mycxt, pjsn_head, max_levels) \ |
203
|
|
|
|
|
|
|
pljsonsl_common_initialize_THX(aTHX_ mycxt, pjsn_head, max_levels) |
204
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
static void |
206
|
33
|
|
|
|
|
|
pljsonsl_common_initialize_THX(pTHX_ |
207
|
|
|
|
|
|
|
my_cxt_t *mycxt, |
208
|
|
|
|
|
|
|
PLJSONSL *pjsn_head, |
209
|
|
|
|
|
|
|
size_t max_levels) |
210
|
|
|
|
|
|
|
{ |
211
|
33
|
|
|
|
|
|
pjsn_head->jsn = jsonsl_new(max_levels+2); |
212
|
33
|
|
|
|
|
|
pjsn_head->jsn->data = pjsn_head; |
213
|
33
|
|
|
|
|
|
pjsn_head->stash_boolean = mycxt->stash_boolean; |
214
|
|
|
|
|
|
|
PLJSONSL_mkTHX(pjsn_head); |
215
|
33
|
|
|
|
|
|
memcpy(pjsn_head->escape_table, ESCTBL, sizeof(ESCTBL)); |
216
|
33
|
|
|
|
|
|
} |
217
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
#define process_special(pjsn,st) process_special_THX(aTHX_ pjsn,st) |
220
|
|
|
|
|
|
|
static inline void |
221
|
30
|
|
|
|
|
|
process_special_THX(pTHX_ |
222
|
|
|
|
|
|
|
PLJSONSL *pjsn, |
223
|
|
|
|
|
|
|
struct jsonsl_state_st *state) |
224
|
|
|
|
|
|
|
{ |
225
|
|
|
|
|
|
|
SV *newsv; |
226
|
30
|
|
|
|
|
|
char *buf = GET_STATE_BUFFER(pjsn, state->pos_begin); |
227
|
30
|
|
|
|
|
|
size_t poscur = pjsn->jsn->pos; |
228
|
|
|
|
|
|
|
|
229
|
30
|
|
|
|
|
|
switch (state->special_flags) { |
230
|
|
|
|
|
|
|
/* might look redundant, but is most common, so it's first */ |
231
|
|
|
|
|
|
|
case JSONSL_SPECIALf_UNSIGNED: |
232
|
|
|
|
|
|
|
case JSONSL_SPECIALf_SIGNED: |
233
|
16
|
|
|
|
|
|
newsv = pljsonsl_common_mknumeric(state, |
234
|
|
|
|
|
|
|
buf, |
235
|
|
|
|
|
|
|
poscur - state->pos_begin); |
236
|
16
|
|
|
|
|
|
break; |
237
|
|
|
|
|
|
|
|
238
|
|
|
|
|
|
|
case JSONSL_SPECIALf_TRUE: |
239
|
|
|
|
|
|
|
case JSONSL_SPECIALf_FALSE: |
240
|
2
|
|
|
|
|
|
newsv = pljsonsl_common_mkboolean(pjsn, state->special_flags); |
241
|
2
|
|
|
|
|
|
break; |
242
|
|
|
|
|
|
|
case JSONSL_SPECIALf_NULL: |
243
|
2
|
|
|
|
|
|
newsv = newSV(0); |
244
|
2
|
|
|
|
|
|
break; |
245
|
|
|
|
|
|
|
default: |
246
|
10
|
|
|
|
|
|
newsv = pljsonsl_common_mknumeric(state, |
247
|
|
|
|
|
|
|
buf, |
248
|
|
|
|
|
|
|
poscur - state->pos_begin); |
249
|
10
|
|
|
|
|
|
break; |
250
|
|
|
|
|
|
|
} |
251
|
|
|
|
|
|
|
|
252
|
30
|
50
|
|
|
|
|
if (newsv == NULL) { |
253
|
0
|
|
|
|
|
|
warn("Buffer is %p", buf); |
254
|
0
|
|
|
|
|
|
warn("Length is %lu", poscur - state->pos_begin); |
255
|
0
|
|
|
|
|
|
warn("Special flag is %d", state->special_flags); |
256
|
0
|
|
|
|
|
|
die("WTF!"); |
257
|
|
|
|
|
|
|
} |
258
|
|
|
|
|
|
|
|
259
|
30
|
|
|
|
|
|
state->sv = newsv; |
260
|
30
|
|
|
|
|
|
return; |
261
|
|
|
|
|
|
|
} |
262
|
|
|
|
|
|
|
|
263
|
|
|
|
|
|
|
/** |
264
|
|
|
|
|
|
|
* This is called to clean up any quotes, and possibly |
265
|
|
|
|
|
|
|
* handle \u-escapes in the future |
266
|
|
|
|
|
|
|
*/ |
267
|
|
|
|
|
|
|
#define process_string(pjsn,st) process_string_THX(aTHX_ pjsn,st) |
268
|
|
|
|
|
|
|
static void |
269
|
25
|
|
|
|
|
|
process_string_THX(pTHX_ |
270
|
|
|
|
|
|
|
PLJSONSL* pjsn, |
271
|
|
|
|
|
|
|
struct jsonsl_state_st *state) |
272
|
|
|
|
|
|
|
{ |
273
|
|
|
|
|
|
|
SV *retsv; |
274
|
25
|
|
|
|
|
|
char *buf = GET_STATE_BUFFER(pjsn, state->pos_begin); |
275
|
|
|
|
|
|
|
size_t buflen; |
276
|
25
|
|
|
|
|
|
size_t poscur = pjsn->jsn->pos; |
277
|
25
|
|
|
|
|
|
buf++; |
278
|
25
|
|
|
|
|
|
buflen = (poscur - state->pos_begin) - 1; |
279
|
25
|
|
|
|
|
|
retsv = newSV(buflen); |
280
|
|
|
|
|
|
|
|
281
|
25
|
|
|
|
|
|
sv_upgrade(retsv, SVt_PV); |
282
|
25
|
|
|
|
|
|
SvPOK_on(retsv); |
283
|
|
|
|
|
|
|
|
284
|
25
|
100
|
|
|
|
|
if (state->nescapes == 0) { |
285
|
16
|
|
|
|
|
|
SvCUR_set(retsv, buflen); |
286
|
16
|
|
|
|
|
|
memcpy(SvPVX(retsv), buf, buflen); |
287
|
|
|
|
|
|
|
} else { |
288
|
|
|
|
|
|
|
jsonsl_error_t err; |
289
|
|
|
|
|
|
|
jsonsl_special_t flags; |
290
|
|
|
|
|
|
|
size_t newlen; |
291
|
9
|
|
|
|
|
|
newlen = jsonsl_util_unescape_ex(buf, |
292
|
|
|
|
|
|
|
SvPVX(retsv), |
293
|
|
|
|
|
|
|
buflen, |
294
|
9
|
|
|
|
|
|
pjsn->escape_table, |
295
|
|
|
|
|
|
|
&flags, |
296
|
|
|
|
|
|
|
&err, NULL); |
297
|
9
|
50
|
|
|
|
|
if (!newlen) { |
298
|
0
|
|
|
|
|
|
SvREFCNT_dec(retsv); |
299
|
0
|
|
|
|
|
|
die("Could not unescape string: %s", jsonsl_strerror(err)); |
300
|
|
|
|
|
|
|
} |
301
|
|
|
|
|
|
|
/* Shrink the buffer to the effective new size */ |
302
|
9
|
|
|
|
|
|
SvCUR_set(retsv, newlen); |
303
|
9
|
50
|
|
|
|
|
if (flags & JSONSL_SPECIALf_NONASCII) { |
304
|
9
|
|
|
|
|
|
SvUTF8_on(retsv); |
305
|
|
|
|
|
|
|
} |
306
|
|
|
|
|
|
|
} |
307
|
|
|
|
|
|
|
|
308
|
25
|
|
|
|
|
|
state->sv = retsv; |
309
|
25
|
100
|
|
|
|
|
if (pjsn->options.utf8) { |
310
|
1
|
|
|
|
|
|
SvUTF8_on(state->sv); |
311
|
|
|
|
|
|
|
} |
312
|
25
|
|
|
|
|
|
} |
313
|
|
|
|
|
|
|
|
314
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
/** |
316
|
|
|
|
|
|
|
* This function will try and determine if the current |
317
|
|
|
|
|
|
|
* item is a matched result (which should be returned to |
318
|
|
|
|
|
|
|
* the user). |
319
|
|
|
|
|
|
|
* If this is a complete match, the SV (along with relevant info) |
320
|
|
|
|
|
|
|
* will be pushed to the result stack and return true. Returns |
321
|
|
|
|
|
|
|
* false otherwise. |
322
|
|
|
|
|
|
|
*/ |
323
|
|
|
|
|
|
|
#define object_mkresult(pjsn,st_p,st_c) object_mkresult_THX(aTHX_ pjsn, st_p,st_c) |
324
|
|
|
|
|
|
|
static inline int |
325
|
1160
|
|
|
|
|
|
object_mkresult_THX(pTHX_ |
326
|
|
|
|
|
|
|
PLJSONSL *pjsn, |
327
|
|
|
|
|
|
|
struct jsonsl_state_st *parent, |
328
|
|
|
|
|
|
|
struct jsonsl_state_st *child) |
329
|
|
|
|
|
|
|
{ |
330
|
|
|
|
|
|
|
#define STORE_INFO(b, v) \ |
331
|
|
|
|
|
|
|
(void)hv_stores(info_hv, PLJSONSL_INFO_KEY_##b, v) |
332
|
|
|
|
|
|
|
HV *info_hv; |
333
|
|
|
|
|
|
|
|
334
|
1160
|
100
|
|
|
|
|
if (pjsn->options.object_drip == 0 && |
|
|
100
|
|
|
|
|
|
335
|
8
|
50
|
|
|
|
|
(child->matchres != JSONSL_MATCH_COMPLETE || child->type == JSONSL_T_HKEY)) { |
336
|
1147
|
|
|
|
|
|
return 0; |
337
|
|
|
|
|
|
|
} |
338
|
|
|
|
|
|
|
|
339
|
13
|
|
|
|
|
|
info_hv = newHV(); |
340
|
13
|
100
|
|
|
|
|
if (SvTYPE(child->sv) == SVt_PVHV || SvTYPE(child->sv) == SVt_PVAV) { |
|
|
100
|
|
|
|
|
|
341
|
6
|
|
|
|
|
|
STORE_INFO(VALUE, newRV_noinc(child->sv)); |
342
|
|
|
|
|
|
|
} else { |
343
|
7
|
|
|
|
|
|
STORE_INFO(VALUE, child->sv); |
344
|
|
|
|
|
|
|
} |
345
|
|
|
|
|
|
|
|
346
|
13
|
100
|
|
|
|
|
if (pjsn->options.noqstr == 0 && pjsn->options.object_drip == 0) { |
|
|
100
|
|
|
|
|
|
347
|
7
|
|
|
|
|
|
STORE_INFO(QUERY, newSVpvn_share(child->matchjpr->orig, |
348
|
|
|
|
|
|
|
child->matchjpr->norig, 0)); |
349
|
|
|
|
|
|
|
} |
350
|
|
|
|
|
|
|
|
351
|
13
|
100
|
|
|
|
|
if (pjsn->options.nopath == 0) { |
352
|
|
|
|
|
|
|
SV *pathstr; |
353
|
|
|
|
|
|
|
int ii; |
354
|
12
|
|
|
|
|
|
pathstr = newSVpvs("/"); |
355
|
33
|
100
|
|
|
|
|
for (ii = 2; ii <= (child->level); ii++) { |
356
|
21
|
|
|
|
|
|
struct jsonsl_state_st *cur = pjsn->jsn->stack + ii; |
357
|
21
|
|
|
|
|
|
struct jsonsl_state_st *prev = jsonsl_last_state(pjsn->jsn, cur); |
358
|
21
|
100
|
|
|
|
|
if (prev->type == JSONSL_T_LIST) { |
359
|
13
|
|
|
|
|
|
sv_catpvf(pathstr, "%d/", cur->u_loc.idx); |
360
|
|
|
|
|
|
|
} else { |
361
|
|
|
|
|
|
|
char *kbuf; |
362
|
|
|
|
|
|
|
STRLEN klen; |
363
|
|
|
|
|
|
|
assert(cur->u_loc.key); |
364
|
8
|
50
|
|
|
|
|
kbuf = HePV(cur->u_loc.key, klen); |
|
|
0
|
|
|
|
|
|
365
|
8
|
|
|
|
|
|
sv_catpvn(pathstr, kbuf, klen); |
366
|
8
|
|
|
|
|
|
sv_catpvs(pathstr, "/"); |
367
|
8
|
50
|
|
|
|
|
if (HeKUTF8(cur->u_loc.key)) { |
368
|
8
|
|
|
|
|
|
SvUTF8_on(pathstr); |
369
|
|
|
|
|
|
|
} |
370
|
|
|
|
|
|
|
} |
371
|
|
|
|
|
|
|
} |
372
|
|
|
|
|
|
|
/* Trim the trailing '/' from the path string */ |
373
|
12
|
50
|
|
|
|
|
if (SvCUR(pathstr) != 1) { |
374
|
12
|
|
|
|
|
|
SvCUR_set(pathstr, SvCUR(pathstr)-1); |
375
|
|
|
|
|
|
|
} |
376
|
12
|
|
|
|
|
|
STORE_INFO(PATH, pathstr); |
377
|
|
|
|
|
|
|
} |
378
|
|
|
|
|
|
|
|
379
|
|
|
|
|
|
|
/** |
380
|
|
|
|
|
|
|
* For the sake of allowing inspection of the object tree, array |
381
|
|
|
|
|
|
|
* and hash types are always added to their parents, even if they |
382
|
|
|
|
|
|
|
* are a complete match to be removed from the stack. |
383
|
|
|
|
|
|
|
*/ |
384
|
13
|
50
|
|
|
|
|
if (parent && parent->sv) { |
|
|
50
|
|
|
|
|
|
385
|
13
|
|
|
|
|
|
SvREADONLY_off(parent->sv); |
386
|
13
|
|
|
|
|
|
SvREFCNT_inc_simple_void_NN(child->sv); |
387
|
13
|
100
|
|
|
|
|
if (parent->type == JSONSL_T_LIST) { |
388
|
7
|
|
|
|
|
|
SV *popped_sv = av_pop((AV*)parent->sv); |
389
|
7
|
50
|
|
|
|
|
if (popped_sv) { |
390
|
7
|
|
|
|
|
|
SvREFCNT_dec(popped_sv); |
391
|
|
|
|
|
|
|
} |
392
|
|
|
|
|
|
|
} else { |
393
|
6
|
|
|
|
|
|
HE* child_he = child->u_loc.key; |
394
|
6
|
|
|
|
|
|
SvREADONLY_off(HeVAL(child_he)); |
395
|
6
|
|
|
|
|
|
delete_he(pjsn, (HV*)parent->sv, child_he); |
396
|
|
|
|
|
|
|
#if 0 |
397
|
|
|
|
|
|
|
/* for perls with hv_common, the above should be a macro for this: */ |
398
|
|
|
|
|
|
|
hv_common((HV*)parent->sv, |
399
|
|
|
|
|
|
|
NULL, kbuf, klen, |
400
|
|
|
|
|
|
|
0, /*maybe HVhek_UTF8?*/ |
401
|
|
|
|
|
|
|
HV_DELETE|G_DISCARD , |
402
|
|
|
|
|
|
|
NULL, |
403
|
|
|
|
|
|
|
HeHASH(child->u_loc.key)); |
404
|
|
|
|
|
|
|
#endif |
405
|
6
|
|
|
|
|
|
child->u_loc.key = NULL; |
406
|
|
|
|
|
|
|
} |
407
|
|
|
|
|
|
|
|
408
|
13
|
|
|
|
|
|
SvREADONLY_on(parent->sv); |
409
|
13
|
|
|
|
|
|
SvREADONLY_off(child->sv); |
410
|
|
|
|
|
|
|
} |
411
|
|
|
|
|
|
|
|
412
|
13
|
|
|
|
|
|
av_push(pjsn->results, newRV_noinc((SV*)info_hv)); |
413
|
13
|
|
|
|
|
|
return 1; |
414
|
|
|
|
|
|
|
#undef STORE_INFO |
415
|
|
|
|
|
|
|
} |
416
|
|
|
|
|
|
|
|
417
|
|
|
|
|
|
|
|
418
|
|
|
|
|
|
|
/** |
419
|
|
|
|
|
|
|
* Because we only want to maintain 'complete' elements, for |
420
|
|
|
|
|
|
|
* strings we ensure that their SVs do not get created until |
421
|
|
|
|
|
|
|
* the entire string is done (as a partial string would |
422
|
|
|
|
|
|
|
* not be of much use to the user anyway). |
423
|
|
|
|
|
|
|
* The opposite is true of hashes and arrays, which we create |
424
|
|
|
|
|
|
|
* immediately. |
425
|
|
|
|
|
|
|
*/ |
426
|
3219
|
|
|
|
|
|
static void body_push_callback(jsonsl_t jsn, |
427
|
|
|
|
|
|
|
jsonsl_action_t action, |
428
|
|
|
|
|
|
|
register struct jsonsl_state_st *state, |
429
|
|
|
|
|
|
|
const char *at) |
430
|
|
|
|
|
|
|
{ |
431
|
|
|
|
|
|
|
struct jsonsl_state_st *parent; |
432
|
|
|
|
|
|
|
SV *newsv; |
433
|
|
|
|
|
|
|
char *mkey; |
434
|
|
|
|
|
|
|
size_t mnkey; |
435
|
3219
|
|
|
|
|
|
register PLJSONSL *pjsn = (PLJSONSL*)jsn->data; |
436
|
|
|
|
|
|
|
PLJSONSL_dTHX(pjsn); |
437
|
|
|
|
|
|
|
|
438
|
|
|
|
|
|
|
/* Reset the position first */ |
439
|
|
|
|
|
|
|
|
440
|
3219
|
|
|
|
|
|
pjsn->keep_pos = state->pos_begin; |
441
|
|
|
|
|
|
|
|
442
|
3219
|
|
|
|
|
|
parent = jsonsl_last_state(jsn, state); |
443
|
|
|
|
|
|
|
/* Here we set up parent positioning variables.. */ |
444
|
3219
|
100
|
|
|
|
|
if (parent->type == JSONSL_T_OBJECT) { |
445
|
2137
|
100
|
|
|
|
|
if (state->type == JSONSL_T_HKEY) { |
446
|
1071
|
|
|
|
|
|
return; |
447
|
|
|
|
|
|
|
} |
448
|
|
|
|
|
|
|
assert(pjsn->curhk); |
449
|
1066
|
|
|
|
|
|
mkey = HeKEY(pjsn->curhk); |
450
|
1066
|
|
|
|
|
|
mnkey = HeKLEN(pjsn->curhk); |
451
|
|
|
|
|
|
|
/** |
452
|
|
|
|
|
|
|
* Set the HE of our current value to the current HK, and then |
453
|
|
|
|
|
|
|
* remove curhk's visibility. |
454
|
|
|
|
|
|
|
*/ |
455
|
1066
|
|
|
|
|
|
state->u_loc.key = pjsn->curhk; |
456
|
1066
|
|
|
|
|
|
pjsn->curhk = NULL; |
457
|
|
|
|
|
|
|
} else { |
458
|
1082
|
|
|
|
|
|
state->u_loc.idx = parent->nelem - 1; |
459
|
1082
|
|
|
|
|
|
mkey = NULL; |
460
|
1082
|
|
|
|
|
|
mnkey = state->u_loc.idx; |
461
|
|
|
|
|
|
|
} |
462
|
|
|
|
|
|
|
|
463
|
2148
|
100
|
|
|
|
|
if (parent->matchres == JSONSL_MATCH_POSSIBLE) { |
464
|
12
|
|
|
|
|
|
state->matchjpr = jsonsl_jpr_match_state(jsn, state, mkey, mnkey, |
465
|
12
|
|
|
|
|
|
&state->matchres); |
466
|
|
|
|
|
|
|
} else { |
467
|
2136
|
|
|
|
|
|
state->matchjpr = NULL; |
468
|
2136
|
|
|
|
|
|
state->matchres = JSONSL_MATCH_NOMATCH; |
469
|
|
|
|
|
|
|
} |
470
|
|
|
|
|
|
|
|
471
|
|
|
|
|
|
|
/** |
472
|
|
|
|
|
|
|
* Ignore warnings about uninitialized newsv variable. |
473
|
|
|
|
|
|
|
*/ |
474
|
2148
|
100
|
|
|
|
|
if (!JSONSL_STATE_IS_CONTAINER(state)) { |
|
|
100
|
|
|
|
|
|
475
|
61
|
|
|
|
|
|
return; /* nothing more to do here. String types are added at POP */ |
476
|
|
|
|
|
|
|
} |
477
|
|
|
|
|
|
|
|
478
|
2087
|
100
|
|
|
|
|
if (state->type == JSONSL_T_OBJECT) { |
479
|
1030
|
|
|
|
|
|
newsv = (SV*)newHV(); |
480
|
1057
|
50
|
|
|
|
|
} else if (state->type == JSONSL_T_LIST) { |
481
|
1057
|
|
|
|
|
|
newsv = (SV*)newAV(); |
482
|
|
|
|
|
|
|
} else { |
483
|
0
|
|
|
|
|
|
die("WTF"); |
484
|
|
|
|
|
|
|
} |
485
|
|
|
|
|
|
|
|
486
|
2087
|
|
|
|
|
|
SvREADONLY_on(newsv); |
487
|
2087
|
100
|
|
|
|
|
if (parent->type == JSONSL_T_LIST) { |
488
|
1059
|
|
|
|
|
|
SvREADONLY_off(parent->sv); |
489
|
1059
|
|
|
|
|
|
av_push((AV*)parent->sv, newRV_noinc(newsv)); |
490
|
1059
|
|
|
|
|
|
SvREADONLY_on(parent->sv); |
491
|
|
|
|
|
|
|
} else { |
492
|
|
|
|
|
|
|
/* we have the HE. */ |
493
|
1028
|
|
|
|
|
|
HeVAL(state->u_loc.key) = newRV_noinc(newsv); |
494
|
1028
|
|
|
|
|
|
SvREADONLY_on(HeVAL(state->u_loc.key)); |
495
|
|
|
|
|
|
|
} |
496
|
|
|
|
|
|
|
|
497
|
2087
|
|
|
|
|
|
state->sv = newsv; |
498
|
|
|
|
|
|
|
} |
499
|
|
|
|
|
|
|
|
500
|
|
|
|
|
|
|
/** |
501
|
|
|
|
|
|
|
* Creates a new HE*. We use this HE later on using HeVAL to assign the value. |
502
|
|
|
|
|
|
|
*/ |
503
|
|
|
|
|
|
|
|
504
|
|
|
|
|
|
|
#define create_hk(pjsn,st_c,st_p) create_hk_THX(aTHX_ pjsn,st_c,st_p) |
505
|
|
|
|
|
|
|
static void |
506
|
1071
|
|
|
|
|
|
create_hk_THX(pTHX_ PLJSONSL *pjsn, |
507
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
508
|
|
|
|
|
|
|
struct jsonsl_state_st *parent) |
509
|
|
|
|
|
|
|
{ |
510
|
1071
|
|
|
|
|
|
char *buf = GET_STATE_BUFFER(pjsn, state->pos_begin); |
511
|
1071
|
|
|
|
|
|
size_t poscur = pjsn->jsn->pos; |
512
|
1071
|
|
|
|
|
|
STRLEN len = (poscur - state->pos_begin)-1; |
513
|
|
|
|
|
|
|
|
514
|
|
|
|
|
|
|
assert(pjsn->curhk == NULL); |
515
|
1071
|
|
|
|
|
|
buf++; |
516
|
|
|
|
|
|
|
|
517
|
1071
|
|
|
|
|
|
SvREADONLY_off(parent->sv); |
518
|
|
|
|
|
|
|
|
519
|
1071
|
100
|
|
|
|
|
if (state->nescapes) { |
520
|
|
|
|
|
|
|
/* we have escapes within a key. rare, but allowable. No choice |
521
|
|
|
|
|
|
|
* but to allocate a new buffer for it |
522
|
|
|
|
|
|
|
*/ |
523
|
|
|
|
|
|
|
|
524
|
|
|
|
|
|
|
/* This sets state->sv to the key sv. would be nice if there was a cleaner |
525
|
|
|
|
|
|
|
* path to this |
526
|
|
|
|
|
|
|
*/ |
527
|
2
|
|
|
|
|
|
process_string(pjsn, state); |
528
|
2
|
|
|
|
|
|
pjsn->curhk = hv_store_ent((HV*)parent->sv, state->sv, &PL_sv_undef, 0); |
529
|
2
|
|
|
|
|
|
SvREFCNT_dec(state->sv); |
530
|
2
|
|
|
|
|
|
state->sv = NULL; |
531
|
|
|
|
|
|
|
} else { |
532
|
|
|
|
|
|
|
|
533
|
|
|
|
|
|
|
/** |
534
|
|
|
|
|
|
|
* Fast path, no copying to new SV. |
535
|
|
|
|
|
|
|
* We need to store &PL_sv_undef first to fool hv_common |
536
|
|
|
|
|
|
|
* into thinking we're not doing anything special. Then |
537
|
|
|
|
|
|
|
* we switch it out to &PL_sv_placeholder so it doesn't appear |
538
|
|
|
|
|
|
|
* visible. |
539
|
|
|
|
|
|
|
*/ |
540
|
1069
|
100
|
|
|
|
|
pjsn->curhk = storeget_he(pjsn, parent->sv, buf, len, &PL_sv_undef, |
|
|
50
|
|
|
|
|
|
541
|
|
|
|
|
|
|
/*determine if UTF8: */ |
542
|
|
|
|
|
|
|
pjsn->options.utf8 || state->special_flags == JSONSL_SPECIALf_NONASCII); |
543
|
|
|
|
|
|
|
#if 0 |
544
|
|
|
|
|
|
|
/* which is really this: */ |
545
|
|
|
|
|
|
|
pjsn->curhk = hv_common((HV*)parent->sv, /* HV*/ |
546
|
|
|
|
|
|
|
NULL, /* keysv */ |
547
|
|
|
|
|
|
|
buf, len, |
548
|
|
|
|
|
|
|
0, /* flags. Maybe setting utf8 */ |
549
|
|
|
|
|
|
|
HV_FETCH_ISSTORE, /*action*/ |
550
|
|
|
|
|
|
|
&PL_sv_undef, /*value*/ |
551
|
|
|
|
|
|
|
0); |
552
|
|
|
|
|
|
|
#endif |
553
|
|
|
|
|
|
|
} |
554
|
|
|
|
|
|
|
|
555
|
1071
|
|
|
|
|
|
HeVAL(pjsn->curhk) = &PL_sv_placeholder; |
556
|
1071
|
|
|
|
|
|
SvREADONLY_on(parent->sv); |
557
|
1071
|
|
|
|
|
|
} |
558
|
|
|
|
|
|
|
|
559
|
|
|
|
|
|
|
/* forward-declare initial state handler */ |
560
|
|
|
|
|
|
|
static void initial_callback(jsonsl_t jsn, |
561
|
|
|
|
|
|
|
jsonsl_action_t action, |
562
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
563
|
|
|
|
|
|
|
const char *at); |
564
|
|
|
|
|
|
|
|
565
|
|
|
|
|
|
|
/** |
566
|
|
|
|
|
|
|
* In this callback we ensure to clean up our strings and push it |
567
|
|
|
|
|
|
|
* into the parent SV |
568
|
|
|
|
|
|
|
*/ |
569
|
2231
|
|
|
|
|
|
static void body_pop_callback(jsonsl_t jsn, |
570
|
|
|
|
|
|
|
jsonsl_action_t action, |
571
|
|
|
|
|
|
|
register struct jsonsl_state_st *state, |
572
|
|
|
|
|
|
|
const char *at) |
573
|
|
|
|
|
|
|
{ |
574
|
|
|
|
|
|
|
/* Ending of an element */ |
575
|
2231
|
|
|
|
|
|
struct jsonsl_state_st *parent = jsonsl_last_state(jsn, state); |
576
|
2231
|
|
|
|
|
|
register PLJSONSL *pjsn = (PLJSONSL*)jsn->data; |
577
|
|
|
|
|
|
|
PLJSONSL_dTHX(pjsn); |
578
|
|
|
|
|
|
|
|
579
|
|
|
|
|
|
|
#define INSERT_STRING \ |
580
|
|
|
|
|
|
|
if (parent && object_mkresult(pjsn, parent, state) == 0) { \ |
581
|
|
|
|
|
|
|
SvREADONLY_off(parent->sv); \ |
582
|
|
|
|
|
|
|
if (parent->type == JSONSL_T_OBJECT) { \ |
583
|
|
|
|
|
|
|
assert(state->u_loc.key); \ |
584
|
|
|
|
|
|
|
HeVAL(state->u_loc.key) = state->sv; \ |
585
|
|
|
|
|
|
|
} else { \ |
586
|
|
|
|
|
|
|
av_push((AV*)parent->sv, state->sv); \ |
587
|
|
|
|
|
|
|
} \ |
588
|
|
|
|
|
|
|
SvREADONLY_on(parent->sv); \ |
589
|
|
|
|
|
|
|
} \ |
590
|
|
|
|
|
|
|
|
591
|
2231
|
100
|
|
|
|
|
if (state->type == JSONSL_T_STRING) { |
592
|
23
|
|
|
|
|
|
process_string(pjsn, state); |
593
|
23
|
50
|
|
|
|
|
INSERT_STRING; |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
594
|
2208
|
100
|
|
|
|
|
} else if (state->type == JSONSL_T_HKEY) { |
595
|
|
|
|
|
|
|
assert(parent->type == JSONSL_T_OBJECT); |
596
|
1071
|
|
|
|
|
|
create_hk(pjsn, state, parent); |
597
|
1137
|
100
|
|
|
|
|
} else if (state->type == JSONSL_T_SPECIAL) { |
598
|
|
|
|
|
|
|
assert(state->special_flags); |
599
|
30
|
|
|
|
|
|
process_special(pjsn, state); |
600
|
30
|
50
|
|
|
|
|
INSERT_STRING; |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
601
|
|
|
|
|
|
|
} else { |
602
|
1107
|
|
|
|
|
|
SvREADONLY_off(state->sv); |
603
|
1107
|
|
|
|
|
|
object_mkresult(pjsn, parent, state); |
604
|
|
|
|
|
|
|
} |
605
|
|
|
|
|
|
|
|
606
|
|
|
|
|
|
|
#undef INSERT_STRING |
607
|
|
|
|
|
|
|
|
608
|
2231
|
100
|
|
|
|
|
if (state->sv == pjsn->root) { |
609
|
43
|
100
|
|
|
|
|
if (pjsn->njprs == 0 && pjsn->options.object_drip == 0) { |
|
|
50
|
|
|
|
|
|
610
|
38
|
|
|
|
|
|
av_push(pjsn->results, newRV_noinc(pjsn->root)); |
611
|
|
|
|
|
|
|
} /* otherwise, already pushed */ |
612
|
43
|
|
|
|
|
|
pjsn->root = NULL; |
613
|
43
|
|
|
|
|
|
jsn->action_callback_PUSH = initial_callback; |
614
|
|
|
|
|
|
|
} |
615
|
|
|
|
|
|
|
|
616
|
2231
|
|
|
|
|
|
state->u_loc.idx = -1; |
617
|
2231
|
|
|
|
|
|
state->sv = NULL; |
618
|
2231
|
|
|
|
|
|
pjsn->keep_pos = 0; |
619
|
|
|
|
|
|
|
|
620
|
2231
|
|
|
|
|
|
} |
621
|
|
|
|
|
|
|
|
622
|
14
|
|
|
|
|
|
static int error_callback(jsonsl_t jsn, |
623
|
|
|
|
|
|
|
jsonsl_error_t err, |
624
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
625
|
|
|
|
|
|
|
char *at) |
626
|
|
|
|
|
|
|
{ |
627
|
14
|
|
|
|
|
|
PLJSONSL *pjsn = (PLJSONSL*)jsn->data; |
628
|
|
|
|
|
|
|
PLJSONSL_dTHX(pjsn); |
629
|
|
|
|
|
|
|
/** |
630
|
|
|
|
|
|
|
* TODO: allow option for user-defined recovery function |
631
|
|
|
|
|
|
|
*/ |
632
|
|
|
|
|
|
|
|
633
|
14
|
|
|
|
|
|
die("JSON::SL - Got error %s at position %lu", jsonsl_strerror(err), jsn->pos); |
634
|
|
|
|
|
|
|
return 0; |
635
|
|
|
|
|
|
|
} |
636
|
|
|
|
|
|
|
|
637
|
4
|
|
|
|
|
|
static void invoke_root_cb(PLJSONSL *pjsn) |
638
|
|
|
|
|
|
|
{ |
639
|
|
|
|
|
|
|
PLJSONSL_dTHX(pjsn); |
640
|
4
|
50
|
|
|
|
|
if (!pjsn->options.root_callback) { |
641
|
0
|
|
|
|
|
|
return; |
642
|
|
|
|
|
|
|
} |
643
|
4
|
|
|
|
|
|
dSP; |
644
|
4
|
|
|
|
|
|
ENTER; |
645
|
4
|
|
|
|
|
|
SAVETMPS; |
646
|
4
|
50
|
|
|
|
|
PUSHMARK(SP); |
647
|
4
|
50
|
|
|
|
|
XPUSHs(sv_2mortal(newRV_inc(pjsn->root))); |
648
|
4
|
|
|
|
|
|
PUTBACK; |
649
|
4
|
|
|
|
|
|
call_sv(pjsn->options.root_callback, G_DISCARD); |
650
|
4
|
50
|
|
|
|
|
FREETMPS; |
651
|
4
|
|
|
|
|
|
LEAVE; |
652
|
|
|
|
|
|
|
} |
653
|
|
|
|
|
|
|
|
654
|
60
|
|
|
|
|
|
static void initial_callback(jsonsl_t jsn, |
655
|
|
|
|
|
|
|
jsonsl_action_t action, |
656
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
657
|
|
|
|
|
|
|
const char *at) |
658
|
|
|
|
|
|
|
{ |
659
|
60
|
|
|
|
|
|
PLJSONSL *pjsn = (PLJSONSL*)jsn->data; |
660
|
|
|
|
|
|
|
PLJSONSL_dTHX(pjsn); |
661
|
|
|
|
|
|
|
|
662
|
|
|
|
|
|
|
assert(action == JSONSL_ACTION_PUSH); |
663
|
60
|
100
|
|
|
|
|
if (state->type == JSONSL_T_LIST) { |
664
|
26
|
|
|
|
|
|
pjsn->root = (SV*)newAV(); |
665
|
34
|
50
|
|
|
|
|
} else if (state->type == JSONSL_T_OBJECT) { |
666
|
34
|
|
|
|
|
|
pjsn->root = (SV*)newHV(); |
667
|
|
|
|
|
|
|
} else { |
668
|
0
|
|
|
|
|
|
die("Found type %s as root element", jsonsl_strtype(state->type)); |
669
|
|
|
|
|
|
|
} |
670
|
|
|
|
|
|
|
|
671
|
60
|
100
|
|
|
|
|
if (pjsn->options.root_callback) { |
672
|
4
|
|
|
|
|
|
invoke_root_cb(pjsn); |
673
|
|
|
|
|
|
|
} |
674
|
|
|
|
|
|
|
|
675
|
60
|
|
|
|
|
|
state->sv = pjsn->root; |
676
|
60
|
|
|
|
|
|
jsn->action_callback = NULL; |
677
|
60
|
|
|
|
|
|
jsn->action_callback_PUSH = body_push_callback; |
678
|
60
|
|
|
|
|
|
jsn->action_callback_POP = body_pop_callback; |
679
|
60
|
|
|
|
|
|
jsonsl_jpr_match_state(jsn, state, NULL, 0, &state->matchres); |
680
|
|
|
|
|
|
|
/* Mark root element as read only */ |
681
|
60
|
|
|
|
|
|
SvREADONLY_on(pjsn->root); |
682
|
60
|
|
|
|
|
|
} |
683
|
|
|
|
|
|
|
|
684
|
|
|
|
|
|
|
#define CHECK_MAX_SIZE(pjsn,input) \ |
685
|
|
|
|
|
|
|
if (pjsn->options.max_size && SvCUR(input) > pjsn->options.max_size) { \ |
686
|
|
|
|
|
|
|
die("JSON::SL - max_size is %lu, but input is %lu bytes", \ |
687
|
|
|
|
|
|
|
pjsn->options.max_size, SvCUR(input)); \ |
688
|
|
|
|
|
|
|
} |
689
|
|
|
|
|
|
|
|
690
|
|
|
|
|
|
|
#define pljsonsl_feed_incr(pjsn,str) pljsonsl_feed_incr_THX(aTHX_ pjsn,str) |
691
|
|
|
|
|
|
|
static void |
692
|
24
|
|
|
|
|
|
pljsonsl_feed_incr_THX(pTHX_ PLJSONSL* pjsn, SV *input) |
693
|
|
|
|
|
|
|
{ |
694
|
24
|
|
|
|
|
|
size_t start_pos = pjsn->jsn->pos; |
695
|
24
|
|
|
|
|
|
STRLEN cur_len = SvCUR(pjsn->buf); |
696
|
|
|
|
|
|
|
|
697
|
24
|
50
|
|
|
|
|
if (!SvPOK(input)) { |
698
|
0
|
|
|
|
|
|
die("Input is not a string"); |
699
|
|
|
|
|
|
|
} |
700
|
|
|
|
|
|
|
|
701
|
24
|
100
|
|
|
|
|
CHECK_MAX_SIZE(pjsn, input) |
|
|
100
|
|
|
|
|
|
702
|
|
|
|
|
|
|
|
703
|
23
|
|
|
|
|
|
pjsn->pos_min_valid = pjsn->jsn->pos - cur_len; |
704
|
23
|
100
|
|
|
|
|
if (SvUTF8(input)) { |
705
|
1
|
|
|
|
|
|
pjsn->options.utf8 = 1; |
706
|
|
|
|
|
|
|
} |
707
|
23
|
|
|
|
|
|
sv_catpvn(pjsn->buf, SvPVX_const(input), SvCUR(input)); |
708
|
23
|
|
|
|
|
|
jsonsl_feed(pjsn->jsn, |
709
|
23
|
|
|
|
|
|
SvPVX_const(pjsn->buf) + (SvCUR(pjsn->buf)-SvCUR(input)), |
710
|
23
|
|
|
|
|
|
SvCUR(input)); |
711
|
|
|
|
|
|
|
/** |
712
|
|
|
|
|
|
|
* Callbacks may detect the beginning of a string, in which case |
713
|
|
|
|
|
|
|
* we need to ensure the continuity of the string. In this case |
714
|
|
|
|
|
|
|
* pos_keep is set to the position of the input stream (not the SV *input, |
715
|
|
|
|
|
|
|
* but rather jsn->pos) from which we should begin buffering data. |
716
|
|
|
|
|
|
|
* |
717
|
|
|
|
|
|
|
* Now we might need to chop. The amount of bytes to chop is the |
718
|
|
|
|
|
|
|
* difference between start_pos and the keep_pos |
719
|
|
|
|
|
|
|
* variable (if any) |
720
|
|
|
|
|
|
|
*/ |
721
|
11
|
50
|
|
|
|
|
if (pjsn->keep_pos == 0) { |
722
|
11
|
|
|
|
|
|
SvCUR_set(pjsn->buf, 0); |
723
|
0
|
0
|
|
|
|
|
} else if (pjsn->keep_pos > start_pos) { |
724
|
0
|
|
|
|
|
|
sv_chop(pjsn->buf, SvPVX_const(pjsn->buf) + (pjsn->keep_pos - start_pos)); |
725
|
|
|
|
|
|
|
} |
726
|
11
|
|
|
|
|
|
} |
727
|
|
|
|
|
|
|
|
728
|
|
|
|
|
|
|
static PLJSONSL* |
729
|
32
|
|
|
|
|
|
pljsonsl_get_and_initialize_global(pTHX) |
730
|
|
|
|
|
|
|
{ |
731
|
|
|
|
|
|
|
dMY_CXT; |
732
|
|
|
|
|
|
|
PLJSONSL *pjsn; |
733
|
32
|
100
|
|
|
|
|
if (MY_CXT.quick == NULL) { |
734
|
7
|
|
|
|
|
|
Newxz(pjsn, 1, PLJSONSL); |
735
|
7
|
|
|
|
|
|
pljsonsl_common_initialize(&MY_CXT, pjsn, PLJSONSL_MAX_DEFAULT-1); |
736
|
7
|
|
|
|
|
|
pjsn->priv_global.is_global = 1; |
737
|
|
|
|
|
|
|
PLJSONSL_INIT_KSV(pjsn); |
738
|
7
|
|
|
|
|
|
MY_CXT.quick = pjsn; |
739
|
|
|
|
|
|
|
} |
740
|
|
|
|
|
|
|
|
741
|
32
|
|
|
|
|
|
pjsn = MY_CXT.quick; |
742
|
32
|
|
|
|
|
|
jsonsl_reset(pjsn->jsn); |
743
|
32
|
|
|
|
|
|
jsonsl_enable_all_callbacks(pjsn->jsn); |
744
|
32
|
|
|
|
|
|
pjsn->jsn->error_callback = error_callback; |
745
|
32
|
|
|
|
|
|
pjsn->jsn->action_callback_PUSH = initial_callback; |
746
|
32
|
|
|
|
|
|
pjsn->results = (AV*)sv_2mortal((SV*)newAV()); |
747
|
32
|
|
|
|
|
|
return pjsn; |
748
|
|
|
|
|
|
|
} |
749
|
|
|
|
|
|
|
|
750
|
|
|
|
|
|
|
#define pljsonsl_feed_oneshot(pjsn,str) pljsonsl_feed_oneshot_THX(aTHX_ pjsn,str) |
751
|
|
|
|
|
|
|
static void |
752
|
32
|
|
|
|
|
|
pljsonsl_feed_oneshot_THX(pTHX_ PLJSONSL* pjsn, SV *input) |
753
|
|
|
|
|
|
|
{ |
754
|
32
|
50
|
|
|
|
|
if (!SvPOK(input)) { |
755
|
0
|
|
|
|
|
|
die("Input is not a string"); |
756
|
|
|
|
|
|
|
} |
757
|
|
|
|
|
|
|
|
758
|
32
|
50
|
|
|
|
|
if (SvUTF8(input)) { |
759
|
0
|
|
|
|
|
|
pjsn->options.utf8 = 1; |
760
|
|
|
|
|
|
|
} |
761
|
32
|
50
|
|
|
|
|
CHECK_MAX_SIZE(pjsn, input); |
|
|
0
|
|
|
|
|
|
762
|
32
|
|
|
|
|
|
pjsn->buf = input; |
763
|
32
|
|
|
|
|
|
jsonsl_feed(pjsn->jsn, SvPVX_const(input), SvCUR(input)); |
764
|
30
|
|
|
|
|
|
pjsn->buf = NULL; |
765
|
30
|
|
|
|
|
|
pjsn->options.utf8 = 0; |
766
|
|
|
|
|
|
|
/* the current root is never in the result stack.. |
767
|
|
|
|
|
|
|
* so mortalizing it won't hurt anyone. |
768
|
|
|
|
|
|
|
*/ |
769
|
30
|
|
|
|
|
|
} |
770
|
|
|
|
|
|
|
|
771
|
|
|
|
|
|
|
/** |
772
|
|
|
|
|
|
|
* Takes an array ref (or list?) of JSONPointer strings and converts |
773
|
|
|
|
|
|
|
* them to JPR objects. Dies on error |
774
|
|
|
|
|
|
|
*/ |
775
|
|
|
|
|
|
|
#define pljsonsl_set_jsonpointer(pjsn,jprstr) \ |
776
|
|
|
|
|
|
|
pljsonsl_set_jsonpointer_THX(aTHX_ pjsn,jprstr) |
777
|
|
|
|
|
|
|
static void |
778
|
4
|
|
|
|
|
|
pljsonsl_set_jsonpointer_THX(pTHX_ PLJSONSL *pjsn, AV *paths) |
779
|
|
|
|
|
|
|
{ |
780
|
|
|
|
|
|
|
jsonsl_jpr_t *jprs; |
781
|
|
|
|
|
|
|
jsonsl_error_t err; |
782
|
|
|
|
|
|
|
int ii; |
783
|
4
|
|
|
|
|
|
int max = av_len(paths)+1; |
784
|
|
|
|
|
|
|
const char *diestr, *pathstr; |
785
|
|
|
|
|
|
|
|
786
|
4
|
50
|
|
|
|
|
if (!max) { |
787
|
0
|
|
|
|
|
|
die("No paths given!"); |
788
|
|
|
|
|
|
|
} |
789
|
|
|
|
|
|
|
|
790
|
4
|
50
|
|
|
|
|
Newxz(jprs, max, jsonsl_jpr_t); |
791
|
|
|
|
|
|
|
|
792
|
8
|
100
|
|
|
|
|
for (ii = 0; ii < max; ii++) { |
793
|
4
|
|
|
|
|
|
SV **tmpsv = av_fetch(paths, ii, 0); |
794
|
4
|
50
|
|
|
|
|
if (tmpsv == NULL || SvPOK(*tmpsv) == 0) { |
|
|
50
|
|
|
|
|
|
795
|
0
|
|
|
|
|
|
diestr = "Found empty path"; |
796
|
0
|
|
|
|
|
|
goto GT_ERR; |
797
|
|
|
|
|
|
|
} |
798
|
4
|
|
|
|
|
|
jprs[ii] = jsonsl_jpr_new(SvPVX_const(*tmpsv), &err); |
799
|
4
|
50
|
|
|
|
|
if (jprs[ii] == NULL) { |
800
|
0
|
|
|
|
|
|
pathstr = SvPVX_const(*tmpsv); |
801
|
0
|
|
|
|
|
|
goto GT_ERR; |
802
|
|
|
|
|
|
|
} |
803
|
|
|
|
|
|
|
} |
804
|
|
|
|
|
|
|
|
805
|
4
|
|
|
|
|
|
jsonsl_jpr_match_state_init(pjsn->jsn, jprs, max); |
806
|
4
|
|
|
|
|
|
pjsn->jprs = jprs; |
807
|
4
|
|
|
|
|
|
pjsn->njprs = max; |
808
|
4
|
|
|
|
|
|
return; |
809
|
|
|
|
|
|
|
|
810
|
|
|
|
|
|
|
GT_ERR: |
811
|
0
|
0
|
|
|
|
|
for (ii = 0; ii < max; ii++) { |
812
|
0
|
0
|
|
|
|
|
if (jprs[ii] == NULL) { |
813
|
0
|
|
|
|
|
|
break; |
814
|
|
|
|
|
|
|
} |
815
|
0
|
|
|
|
|
|
jsonsl_jpr_destroy(jprs[ii]); |
816
|
|
|
|
|
|
|
} |
817
|
0
|
|
|
|
|
|
Safefree(jprs); |
818
|
0
|
0
|
|
|
|
|
if (pathstr) { |
819
|
0
|
|
|
|
|
|
die("Couldn't convert %s to jsonpointer: %s", pathstr, jsonsl_strerror(err)); |
820
|
|
|
|
|
|
|
} else { |
821
|
4
|
|
|
|
|
|
die(diestr); |
822
|
|
|
|
|
|
|
} |
823
|
|
|
|
|
|
|
} |
824
|
|
|
|
|
|
|
|
825
|
|
|
|
|
|
|
|
826
|
|
|
|
|
|
|
|
827
|
|
|
|
|
|
|
|
828
|
|
|
|
|
|
|
|
829
|
|
|
|
|
|
|
/** |
830
|
|
|
|
|
|
|
* JSON::SL::Tuba functions. |
831
|
|
|
|
|
|
|
* In case you haven't wondered already, 'Tuba' is a play on 'SAX'. |
832
|
|
|
|
|
|
|
*/ |
833
|
|
|
|
|
|
|
|
834
|
|
|
|
|
|
|
|
835
|
|
|
|
|
|
|
/** |
836
|
|
|
|
|
|
|
* This is our quick version of MRO caching. Maybe I'll swap this out |
837
|
|
|
|
|
|
|
* for something which already exists (as I get the feeling I've reinveted |
838
|
|
|
|
|
|
|
* the wheel here. |
839
|
|
|
|
|
|
|
* namep is populated with the handler name, gvp is a pointer to the GV** |
840
|
|
|
|
|
|
|
* - or an offset into the PLTUBA's methgv structure. |
841
|
|
|
|
|
|
|
*/ |
842
|
|
|
|
|
|
|
static void |
843
|
57
|
|
|
|
|
|
pltuba_get_method_info(PLTUBA *tuba, |
844
|
|
|
|
|
|
|
jsonsl_action_t action, |
845
|
|
|
|
|
|
|
pltuba_callback_type cbtype, |
846
|
|
|
|
|
|
|
GV ***gvpp, |
847
|
|
|
|
|
|
|
const char **namep) |
848
|
|
|
|
|
|
|
{ |
849
|
57
|
|
|
|
|
|
GV **methgvp = NULL; |
850
|
57
|
|
|
|
|
|
const char *methname = NULL; |
851
|
57
|
|
|
|
|
|
cbtype &= 0x7f; |
852
|
|
|
|
|
|
|
|
853
|
57
|
50
|
|
|
|
|
if (tuba->options.cb_unified) { |
854
|
0
|
|
|
|
|
|
methname = "on_any"; |
855
|
0
|
|
|
|
|
|
methgvp = &tuba->methgv.on_any; |
856
|
0
|
|
|
|
|
|
goto GT_RETASGN; |
857
|
|
|
|
|
|
|
} |
858
|
|
|
|
|
|
|
#define PLTUBA_METH_GETMETH |
859
|
|
|
|
|
|
|
#include "srcout/tuba_dispatch_getmeth.h" |
860
|
|
|
|
|
|
|
#undef PLTUBA_METH_GETMETH |
861
|
|
|
|
|
|
|
GT_RETASGN: |
862
|
57
|
50
|
|
|
|
|
if (gvpp) { |
863
|
57
|
|
|
|
|
|
*gvpp = methgvp; |
864
|
|
|
|
|
|
|
} |
865
|
57
|
50
|
|
|
|
|
if (namep) { |
866
|
57
|
|
|
|
|
|
*namep = methname; |
867
|
|
|
|
|
|
|
} |
868
|
57
|
|
|
|
|
|
} |
869
|
|
|
|
|
|
|
|
870
|
|
|
|
|
|
|
static void |
871
|
4
|
|
|
|
|
|
pltuba_invalidate_gvs_THX(pTHX_ PLTUBA *tuba) |
872
|
|
|
|
|
|
|
{ |
873
|
|
|
|
|
|
|
|
874
|
|
|
|
|
|
|
#define X(action,type) \ |
875
|
|
|
|
|
|
|
REFDEC_FIELD(tuba, methgv.action## _ ##type) |
876
|
|
|
|
|
|
|
|
877
|
4
|
50
|
|
|
|
|
PLTUBA_XMETHGV |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
878
|
|
|
|
|
|
|
#undef X |
879
|
4
|
|
|
|
|
|
} |
880
|
|
|
|
|
|
|
|
881
|
|
|
|
|
|
|
/** |
882
|
|
|
|
|
|
|
* Maps a 'jsonsl' type to a tuba callback type. |
883
|
|
|
|
|
|
|
*/ |
884
|
|
|
|
|
|
|
static pltuba_callback_type |
885
|
58
|
|
|
|
|
|
convert_to_tuba_cbt(struct jsonsl_state_st *state) |
886
|
|
|
|
|
|
|
{ |
887
|
58
|
100
|
|
|
|
|
if (state->type != JSONSL_T_SPECIAL) { |
888
|
46
|
|
|
|
|
|
return state->type; |
889
|
|
|
|
|
|
|
} |
890
|
12
|
100
|
|
|
|
|
if (state->special_flags & JSONSL_SPECIALf_BOOLEAN) { |
891
|
4
|
|
|
|
|
|
return PLTUBA_CALLBACK_BOOLEAN; |
892
|
8
|
100
|
|
|
|
|
} else if (state->special_flags & (JSONSL_SPECIALf_NUMERIC | JSONSL_SPECIALf_DASH)) { |
893
|
6
|
|
|
|
|
|
return PLTUBA_CALLBACK_NUMBER; |
894
|
2
|
50
|
|
|
|
|
} else if (state->special_flags == JSONSL_SPECIALf_NULL) { |
895
|
2
|
|
|
|
|
|
return PLTUBA_CALLBACK_NULL; |
896
|
|
|
|
|
|
|
} |
897
|
0
|
|
|
|
|
|
warn("Special flag is %d", state->special_flags); |
898
|
0
|
|
|
|
|
|
die("wtf?"); |
899
|
|
|
|
|
|
|
return 0; |
900
|
|
|
|
|
|
|
} |
901
|
|
|
|
|
|
|
|
902
|
|
|
|
|
|
|
/** |
903
|
|
|
|
|
|
|
* This function invokes the selected callback (if it exists). |
904
|
|
|
|
|
|
|
*/ |
905
|
|
|
|
|
|
|
#define pltuba_invoke_callback(tb,a,cbt,sv) \ |
906
|
|
|
|
|
|
|
pltuba_invoke_callback_THX(aTHX_ tb,a,cbt,sv) |
907
|
|
|
|
|
|
|
static void |
908
|
26
|
|
|
|
|
|
pltuba_invoke_callback_THX(pTHX_ PLTUBA *tuba, |
909
|
|
|
|
|
|
|
int action, |
910
|
|
|
|
|
|
|
pltuba_callback_type cbtype, |
911
|
|
|
|
|
|
|
SV *mextrasv) |
912
|
|
|
|
|
|
|
{ |
913
|
26
|
|
|
|
|
|
dSP; |
914
|
26
|
|
|
|
|
|
GV **methp = NULL; |
915
|
26
|
|
|
|
|
|
GV *meth = NULL; |
916
|
26
|
|
|
|
|
|
const char *meth_name = NULL; |
917
|
26
|
|
|
|
|
|
int effective_type = cbtype; |
918
|
26
|
|
|
|
|
|
int effective_action = action; |
919
|
26
|
|
|
|
|
|
int stop_mro = 0; |
920
|
|
|
|
|
|
|
/** |
921
|
|
|
|
|
|
|
* If we are in a pop mode of a callback with the accumulator flag set, |
922
|
|
|
|
|
|
|
* then we provide the data in the SV as the argument (maybe with some |
923
|
|
|
|
|
|
|
* conversion into an appropriate object), otherwise, we just signal as |
924
|
|
|
|
|
|
|
* normal. |
925
|
|
|
|
|
|
|
*/ |
926
|
26
|
|
|
|
|
|
cbtype &= 0x7f; |
927
|
|
|
|
|
|
|
|
928
|
26
|
100
|
|
|
|
|
if (tuba->accum && action == JSONSL_ACTION_POP) { |
|
|
50
|
|
|
|
|
|
929
|
14
|
|
|
|
|
|
effective_action = PLTUBA_ACTION_ON; |
930
|
14
|
|
|
|
|
|
pltuba_get_method_info(tuba, PLTUBA_ACTION_ON, cbtype, &methp, &meth_name); |
931
|
|
|
|
|
|
|
assert(mextrasv == NULL); |
932
|
14
|
|
|
|
|
|
mextrasv = tuba->accum; |
933
|
14
|
|
|
|
|
|
tuba->accum = NULL; |
934
|
|
|
|
|
|
|
} else { |
935
|
12
|
|
|
|
|
|
pltuba_get_method_info(tuba, action, cbtype, &methp, &meth_name); |
936
|
|
|
|
|
|
|
} |
937
|
|
|
|
|
|
|
|
938
|
26
|
50
|
|
|
|
|
if (meth_name == NULL) { |
939
|
0
|
|
|
|
|
|
die("Can't find method name. Action=%c, Type=%c", action, cbtype); |
940
|
|
|
|
|
|
|
} |
941
|
|
|
|
|
|
|
|
942
|
26
|
100
|
|
|
|
|
if (!mextrasv) { |
943
|
12
|
|
|
|
|
|
mextrasv = &PL_sv_undef; |
944
|
|
|
|
|
|
|
} else { |
945
|
14
|
|
|
|
|
|
sv_2mortal(mextrasv); |
946
|
|
|
|
|
|
|
} |
947
|
|
|
|
|
|
|
|
948
|
|
|
|
|
|
|
assert(methp); |
949
|
|
|
|
|
|
|
|
950
|
26
|
100
|
|
|
|
|
if (tuba->last_stash != SvSTASH(SvRV(tuba->selfrv))) { |
951
|
2
|
|
|
|
|
|
pltuba_invalidate_gvs_THX(aTHX_ tuba); |
952
|
2
|
|
|
|
|
|
tuba->last_stash = SvSTASH(SvRV(tuba->selfrv)); |
953
|
|
|
|
|
|
|
} |
954
|
|
|
|
|
|
|
|
955
|
|
|
|
|
|
|
do { |
956
|
52
|
100
|
|
|
|
|
if (*methp == NULL) { |
957
|
32
|
|
|
|
|
|
meth = gv_fetchmethod_autoload(SvSTASH(SvRV(tuba->selfrv)), meth_name, 1); |
958
|
32
|
100
|
|
|
|
|
if (meth && GvCV(meth)) { |
|
|
50
|
|
|
|
|
|
959
|
1
|
50
|
|
|
|
|
if (tuba->options.no_cache_mro == 0) { |
960
|
1
|
|
|
|
|
|
*methp = meth; |
961
|
1
|
|
|
|
|
|
SvREFCNT_inc(meth); |
962
|
|
|
|
|
|
|
} |
963
|
1
|
|
|
|
|
|
break; |
964
|
|
|
|
|
|
|
} /* else */ |
965
|
31
|
|
|
|
|
|
pltuba_get_method_info(tuba, PLTUBA_ACTION_ON, |
966
|
|
|
|
|
|
|
PLTUBA_CALLBACK_ANY, &methp, &meth_name); |
967
|
|
|
|
|
|
|
assert(methp && meth_name); |
968
|
31
|
|
|
|
|
|
stop_mro++; |
969
|
|
|
|
|
|
|
} else { |
970
|
20
|
|
|
|
|
|
meth = *methp; |
971
|
20
|
|
|
|
|
|
break; |
972
|
|
|
|
|
|
|
} |
973
|
31
|
100
|
|
|
|
|
} while (stop_mro < 2); |
974
|
|
|
|
|
|
|
|
975
|
26
|
|
|
|
|
|
PLTUBA_SET_PARAMFIELDS_dv(tuba, Mode, effective_action); |
976
|
26
|
|
|
|
|
|
PLTUBA_SET_PARAMFIELDS_dv(tuba, Type, effective_type); |
977
|
|
|
|
|
|
|
|
978
|
|
|
|
|
|
|
/** |
979
|
|
|
|
|
|
|
* We still want a SAVETMPS/FREETMPS pair active before we decide |
980
|
|
|
|
|
|
|
* to call a function or not, as the contents mextrasv and possibly |
981
|
|
|
|
|
|
|
* some of the hash values are mortalized. |
982
|
|
|
|
|
|
|
*/ |
983
|
26
|
|
|
|
|
|
ENTER; SAVETMPS; |
984
|
26
|
100
|
|
|
|
|
if (meth && GvCV(meth)) { |
|
|
50
|
|
|
|
|
|
985
|
21
|
50
|
|
|
|
|
PUSHMARK(SP); |
986
|
21
|
50
|
|
|
|
|
EXTEND(SP, 2); |
987
|
21
|
|
|
|
|
|
PUSHs(tuba->selfrv); |
988
|
21
|
|
|
|
|
|
PUSHs(tuba->paramhvrv); |
989
|
21
|
100
|
|
|
|
|
if (mextrasv != &PL_sv_undef) { |
990
|
12
|
50
|
|
|
|
|
XPUSHs(mextrasv); |
991
|
|
|
|
|
|
|
} |
992
|
21
|
|
|
|
|
|
PUTBACK; |
993
|
21
|
|
|
|
|
|
call_sv((SV*)GvCV(meth), G_DISCARD); |
994
|
|
|
|
|
|
|
} else { |
995
|
5
|
50
|
|
|
|
|
if (!tuba->options.allow_unhandled) { |
996
|
0
|
|
|
|
|
|
die("Tuba: Cannot find handler for mode 0x%02x action 0x%02x", |
997
|
|
|
|
|
|
|
effective_action, effective_type); |
998
|
|
|
|
|
|
|
} |
999
|
|
|
|
|
|
|
} |
1000
|
26
|
50
|
|
|
|
|
FREETMPS; LEAVE; |
1001
|
26
|
|
|
|
|
|
} |
1002
|
|
|
|
|
|
|
|
1003
|
|
|
|
|
|
|
/** |
1004
|
|
|
|
|
|
|
* Flush characters between the invocation of the last callback |
1005
|
|
|
|
|
|
|
* and the current one. the until argument is the end position (inclusive) |
1006
|
|
|
|
|
|
|
* at which we should stop submitting 'character' data. |
1007
|
|
|
|
|
|
|
*/ |
1008
|
|
|
|
|
|
|
#define pltuba_flush_characters(tb,end) \ |
1009
|
|
|
|
|
|
|
pltuba_flush_characters_THX(aTHX_ tb,end) |
1010
|
|
|
|
|
|
|
static void |
1011
|
25
|
|
|
|
|
|
pltuba_flush_characters_THX(pTHX_ PLTUBA *tuba, size_t until) |
1012
|
|
|
|
|
|
|
{ |
1013
|
|
|
|
|
|
|
STRLEN toFlush; |
1014
|
|
|
|
|
|
|
const char *buf; |
1015
|
|
|
|
|
|
|
SV *chunksv; |
1016
|
|
|
|
|
|
|
|
1017
|
25
|
50
|
|
|
|
|
if (!tuba->keep_pos) { |
1018
|
0
|
|
|
|
|
|
return; |
1019
|
|
|
|
|
|
|
} |
1020
|
|
|
|
|
|
|
|
1021
|
25
|
|
|
|
|
|
toFlush = (until - tuba->keep_pos); |
1022
|
25
|
50
|
|
|
|
|
if (toFlush == 0) { |
1023
|
0
|
|
|
|
|
|
return; |
1024
|
|
|
|
|
|
|
} |
1025
|
25
|
|
|
|
|
|
buf = GET_STATE_BUFFER(tuba, tuba->keep_pos); |
1026
|
|
|
|
|
|
|
|
1027
|
25
|
100
|
|
|
|
|
if (tuba->shift_quote) { |
1028
|
19
|
|
|
|
|
|
buf++; |
1029
|
19
|
|
|
|
|
|
toFlush--; |
1030
|
|
|
|
|
|
|
} |
1031
|
|
|
|
|
|
|
|
1032
|
25
|
|
|
|
|
|
tuba->keep_pos = 0; |
1033
|
25
|
|
|
|
|
|
tuba->shift_quote = 0; |
1034
|
|
|
|
|
|
|
|
1035
|
25
|
50
|
|
|
|
|
if (toFlush == 0 && tuba->shift_quote == 0) { |
|
|
0
|
|
|
|
|
|
1036
|
|
|
|
|
|
|
/* if we have no data and the count was not artificially decremented, then |
1037
|
|
|
|
|
|
|
* don't invoke the callback |
1038
|
|
|
|
|
|
|
*/ |
1039
|
0
|
|
|
|
|
|
return; |
1040
|
|
|
|
|
|
|
} |
1041
|
|
|
|
|
|
|
|
1042
|
|
|
|
|
|
|
/* if accumulator mode is on, don't send the data right away. |
1043
|
|
|
|
|
|
|
* buffer it instead */ |
1044
|
25
|
50
|
|
|
|
|
if (tuba->accum) { |
1045
|
25
|
|
|
|
|
|
sv_catpvn(tuba->accum, buf, toFlush); |
1046
|
25
|
|
|
|
|
|
return; |
1047
|
|
|
|
|
|
|
} /* else, no accum for this state */ |
1048
|
|
|
|
|
|
|
|
1049
|
0
|
|
|
|
|
|
chunksv = newSVpvn(buf, toFlush); |
1050
|
0
|
|
|
|
|
|
pltuba_invoke_callback(tuba, |
1051
|
|
|
|
|
|
|
PLTUBA_ACTION_ON, |
1052
|
|
|
|
|
|
|
PLTUBA_CALLBACK_DATA, |
1053
|
|
|
|
|
|
|
chunksv); |
1054
|
|
|
|
|
|
|
/** |
1055
|
|
|
|
|
|
|
* SV has been mortalized by the invoke_callback function |
1056
|
|
|
|
|
|
|
*/ |
1057
|
|
|
|
|
|
|
} |
1058
|
|
|
|
|
|
|
/** |
1059
|
|
|
|
|
|
|
* Push callback. This is easy because we never actually do any character |
1060
|
|
|
|
|
|
|
* data here. |
1061
|
|
|
|
|
|
|
*/ |
1062
|
|
|
|
|
|
|
static void |
1063
|
29
|
|
|
|
|
|
pltuba_jsonsl_push_callback(jsonsl_t jsn, |
1064
|
|
|
|
|
|
|
jsonsl_action_t action, |
1065
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
1066
|
|
|
|
|
|
|
const char *at) |
1067
|
|
|
|
|
|
|
{ |
1068
|
29
|
|
|
|
|
|
PLTUBA *tuba = (PLTUBA*)jsn->data; |
1069
|
|
|
|
|
|
|
PLJSONSL_dTHX(tuba); |
1070
|
29
|
|
|
|
|
|
struct jsonsl_state_st *parent = jsonsl_last_state(jsn, state); |
1071
|
29
|
|
|
|
|
|
pltuba_callback_type cbt = convert_to_tuba_cbt(state); |
1072
|
29
|
100
|
|
|
|
|
if (state->level == 1) { |
1073
|
2
|
|
|
|
|
|
pltuba_invoke_callback(tuba, action, PLTUBA_CALLBACK_JSON, NULL); |
1074
|
|
|
|
|
|
|
} else { |
1075
|
|
|
|
|
|
|
assert(parent); |
1076
|
27
|
100
|
|
|
|
|
if (parent->type == JSONSL_T_LIST) { |
1077
|
5
|
|
|
|
|
|
PLTUBA_SET_PARAMFIELDS_iv(tuba, Index, parent->nelem-1); |
1078
|
|
|
|
|
|
|
} else { |
1079
|
22
|
|
|
|
|
|
PLTUBA_RESET_PARAMFIELD(tuba, Index); |
1080
|
|
|
|
|
|
|
} |
1081
|
|
|
|
|
|
|
} |
1082
|
|
|
|
|
|
|
|
1083
|
29
|
100
|
|
|
|
|
if (tuba->accum_options[cbt & 0x7f]) { |
1084
|
|
|
|
|
|
|
assert(tuba->accum == NULL); |
1085
|
|
|
|
|
|
|
/* accum is only ever valid for atomic types */ |
1086
|
25
|
|
|
|
|
|
tuba->accum = newSVpvn("", 0); |
1087
|
|
|
|
|
|
|
} else { |
1088
|
4
|
100
|
|
|
|
|
if (JSONSL_STATE_IS_CONTAINER(state) && tuba->kaccum) { |
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
1089
|
2
|
|
|
|
|
|
sv_2mortal(tuba->kaccum); |
1090
|
2
|
|
|
|
|
|
tuba->kaccum = NULL; |
1091
|
2
|
|
|
|
|
|
pltuba_invoke_callback(tuba, action, cbt, NULL); |
1092
|
2
|
|
|
|
|
|
PLTUBA_RESET_PARAMFIELD(tuba, Key); |
1093
|
|
|
|
|
|
|
} else { |
1094
|
2
|
|
|
|
|
|
pltuba_invoke_callback(tuba, action, cbt, NULL); |
1095
|
|
|
|
|
|
|
} |
1096
|
|
|
|
|
|
|
} |
1097
|
|
|
|
|
|
|
|
1098
|
|
|
|
|
|
|
/* This is a different branch and must get executed regardless |
1099
|
|
|
|
|
|
|
* of whether we invoke a callback or use the accumulator */ |
1100
|
29
|
100
|
|
|
|
|
if (!JSONSL_STATE_IS_CONTAINER(state)) { |
|
|
100
|
|
|
|
|
|
1101
|
25
|
|
|
|
|
|
tuba->keep_pos = state->pos_begin; |
1102
|
44
|
100
|
|
|
|
|
if (state->type & JSONSL_Tf_STRINGY) { |
1103
|
19
|
|
|
|
|
|
tuba->shift_quote = 1; |
1104
|
|
|
|
|
|
|
} |
1105
|
|
|
|
|
|
|
} else { |
1106
|
4
|
|
|
|
|
|
tuba->keep_pos = 0; |
1107
|
|
|
|
|
|
|
} |
1108
|
29
|
|
|
|
|
|
} |
1109
|
|
|
|
|
|
|
|
1110
|
|
|
|
|
|
|
/** |
1111
|
|
|
|
|
|
|
* If we're special, then convert all weird stuff to their |
1112
|
|
|
|
|
|
|
* proper perly form. Simple plain integers are not weird and |
1113
|
|
|
|
|
|
|
* can be stringified on demand. |
1114
|
|
|
|
|
|
|
* This is akin to JSON::SL's process_string and process_special |
1115
|
|
|
|
|
|
|
* functions. |
1116
|
|
|
|
|
|
|
*/ |
1117
|
|
|
|
|
|
|
#define pltuba_process_accum(tuba, state) \ |
1118
|
|
|
|
|
|
|
pltuba_process_accum_THX(aTHX_ tuba, state) |
1119
|
|
|
|
|
|
|
static void |
1120
|
25
|
|
|
|
|
|
pltuba_process_accum_THX(pTHX_ |
1121
|
|
|
|
|
|
|
PLTUBA *tuba, |
1122
|
|
|
|
|
|
|
struct jsonsl_state_st *state) |
1123
|
|
|
|
|
|
|
{ |
1124
|
25
|
|
|
|
|
|
size_t poscur = tuba->jsn->pos; |
1125
|
25
|
100
|
|
|
|
|
if (state->type == JSONSL_T_SPECIAL) { |
1126
|
|
|
|
|
|
|
SV *newsv; |
1127
|
|
|
|
|
|
|
|
1128
|
6
|
100
|
|
|
|
|
if ( (state->special_flags & JSONSL_SPECIALf_NUMERIC) && |
|
|
100
|
|
|
|
|
|
1129
|
3
|
|
|
|
|
|
(state->special_flags & JSONSL_SPECIALf_NUMNOINT) == 0) { |
1130
|
|
|
|
|
|
|
goto GT_NONEWSV; |
1131
|
|
|
|
|
|
|
|
1132
|
5
|
100
|
|
|
|
|
} else if (state->special_flags & JSONSL_SPECIALf_NUMNOINT) { |
1133
|
2
|
|
|
|
|
|
newsv = pljsonsl_common_mknumeric(state, |
1134
|
|
|
|
|
|
|
SvPVX_const(tuba->accum), |
1135
|
|
|
|
|
|
|
poscur - state->pos_begin); |
1136
|
3
|
100
|
|
|
|
|
} else if (state->special_flags & JSONSL_SPECIALf_BOOLEAN) { |
1137
|
2
|
|
|
|
|
|
newsv = pljsonsl_common_mkboolean((PLJSONSL*)tuba, |
1138
|
|
|
|
|
|
|
state->special_flags); |
1139
|
|
|
|
|
|
|
} else { |
1140
|
1
|
|
|
|
|
|
newsv = &PL_sv_undef; |
1141
|
|
|
|
|
|
|
} |
1142
|
5
|
|
|
|
|
|
SvREFCNT_dec(tuba->accum); |
1143
|
6
|
|
|
|
|
|
tuba->accum = newsv; |
1144
|
|
|
|
|
|
|
GT_NONEWSV: |
1145
|
|
|
|
|
|
|
; |
1146
|
|
|
|
|
|
|
} else { |
1147
|
19
|
50
|
|
|
|
|
if (tuba->options.utf8) { |
1148
|
0
|
|
|
|
|
|
SvUTF8_on(tuba->accum); |
1149
|
|
|
|
|
|
|
} |
1150
|
19
|
100
|
|
|
|
|
if (state->nescapes) { |
1151
|
|
|
|
|
|
|
jsonsl_error_t err; |
1152
|
|
|
|
|
|
|
jsonsl_special_t flags; |
1153
|
|
|
|
|
|
|
size_t newlen; |
1154
|
2
|
|
|
|
|
|
newlen = jsonsl_util_unescape_ex(SvPVX_const(tuba->accum), |
1155
|
2
|
|
|
|
|
|
SvPVX(tuba->accum), |
1156
|
2
|
|
|
|
|
|
SvCUR(tuba->accum), |
1157
|
2
|
|
|
|
|
|
tuba->escape_table, |
1158
|
|
|
|
|
|
|
&flags, |
1159
|
|
|
|
|
|
|
&err, |
1160
|
|
|
|
|
|
|
NULL); |
1161
|
2
|
50
|
|
|
|
|
if (newlen == 0) { |
1162
|
0
|
|
|
|
|
|
die("Could not unescape string: %s", jsonsl_strerror(err)); |
1163
|
|
|
|
|
|
|
} |
1164
|
2
|
|
|
|
|
|
SvCUR_set(tuba->accum, newlen); |
1165
|
2
|
50
|
|
|
|
|
if (flags & JSONSL_SPECIALf_NONASCII) { |
1166
|
2
|
|
|
|
|
|
SvUTF8_on(tuba->accum); |
1167
|
|
|
|
|
|
|
} |
1168
|
|
|
|
|
|
|
} |
1169
|
|
|
|
|
|
|
} |
1170
|
25
|
|
|
|
|
|
} |
1171
|
|
|
|
|
|
|
|
1172
|
|
|
|
|
|
|
static void |
1173
|
29
|
|
|
|
|
|
pltuba_jsonsl_pop_callback(jsonsl_t jsn, |
1174
|
|
|
|
|
|
|
jsonsl_action_t action, |
1175
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
1176
|
|
|
|
|
|
|
const char *at) |
1177
|
|
|
|
|
|
|
{ |
1178
|
29
|
|
|
|
|
|
PLTUBA *tuba = (PLTUBA*)jsn->data; |
1179
|
|
|
|
|
|
|
PLJSONSL_dTHX(tuba); |
1180
|
29
|
|
|
|
|
|
pltuba_callback_type cbt = convert_to_tuba_cbt(state); |
1181
|
29
|
|
|
|
|
|
size_t poscur = jsn->pos; |
1182
|
|
|
|
|
|
|
|
1183
|
29
|
100
|
|
|
|
|
if (!JSONSL_STATE_IS_CONTAINER(state)) { |
|
|
100
|
|
|
|
|
|
1184
|
|
|
|
|
|
|
/* Special handling for character crap.. */ |
1185
|
25
|
|
|
|
|
|
pltuba_flush_characters(tuba, poscur); |
1186
|
|
|
|
|
|
|
|
1187
|
|
|
|
|
|
|
|
1188
|
25
|
50
|
|
|
|
|
if (tuba->accum) { |
1189
|
25
|
|
|
|
|
|
pltuba_process_accum(tuba, state); |
1190
|
|
|
|
|
|
|
} else { |
1191
|
0
|
0
|
|
|
|
|
if (state->nescapes) { |
1192
|
0
|
|
|
|
|
|
PLTUBA_SET_PARAMFIELDS_sv(tuba, Escaped, &PL_sv_yes); |
1193
|
|
|
|
|
|
|
} |
1194
|
|
|
|
|
|
|
} |
1195
|
|
|
|
|
|
|
|
1196
|
25
|
100
|
|
|
|
|
if (state->type == JSONSL_T_HKEY && |
|
|
50
|
|
|
|
|
|
1197
|
11
|
|
|
|
|
|
tuba->options.accum_kv) { |
1198
|
|
|
|
|
|
|
/** |
1199
|
|
|
|
|
|
|
* If we are accumulating the key then don't flush characters under |
1200
|
|
|
|
|
|
|
* any circumstances. Just swap over the accumulator buffer |
1201
|
|
|
|
|
|
|
*/ |
1202
|
|
|
|
|
|
|
assert(tuba->accum); |
1203
|
11
|
|
|
|
|
|
tuba->kaccum = tuba->accum; |
1204
|
11
|
|
|
|
|
|
tuba->accum = NULL; |
1205
|
11
|
|
|
|
|
|
tuba->keep_pos = 0; |
1206
|
11
|
|
|
|
|
|
PLTUBA_SET_PARAMFIELDS_sv(tuba, Key, tuba->kaccum); |
1207
|
11
|
|
|
|
|
|
return; |
1208
|
|
|
|
|
|
|
} |
1209
|
|
|
|
|
|
|
} |
1210
|
|
|
|
|
|
|
|
1211
|
18
|
100
|
|
|
|
|
if (tuba->kaccum && state->type != JSONSL_T_HKEY) { |
|
|
50
|
|
|
|
|
|
1212
|
9
|
|
|
|
|
|
sv_2mortal(tuba->kaccum); |
1213
|
9
|
|
|
|
|
|
tuba->kaccum = NULL; |
1214
|
|
|
|
|
|
|
} |
1215
|
|
|
|
|
|
|
|
1216
|
18
|
|
|
|
|
|
pltuba_invoke_callback(tuba, action, cbt, NULL); |
1217
|
|
|
|
|
|
|
|
1218
|
|
|
|
|
|
|
/** |
1219
|
|
|
|
|
|
|
* Clear all fields |
1220
|
|
|
|
|
|
|
*/ |
1221
|
|
|
|
|
|
|
#define X(kname) \ |
1222
|
|
|
|
|
|
|
PLTUBA_RESET_PARAMFIELD(tuba, kname); |
1223
|
18
|
|
|
|
|
|
PLTUBA_XPARAMS; |
1224
|
|
|
|
|
|
|
#undef X |
1225
|
|
|
|
|
|
|
|
1226
|
18
|
100
|
|
|
|
|
if (state->level == 1) { |
1227
|
2
|
|
|
|
|
|
pltuba_invoke_callback(tuba, action, PLTUBA_CALLBACK_JSON, NULL); |
1228
|
|
|
|
|
|
|
} |
1229
|
18
|
|
|
|
|
|
tuba->keep_pos = 0; |
1230
|
|
|
|
|
|
|
} |
1231
|
|
|
|
|
|
|
|
1232
|
|
|
|
|
|
|
static int |
1233
|
0
|
|
|
|
|
|
pltuba_jsonsl_error_callback(jsonsl_t jsn, |
1234
|
|
|
|
|
|
|
jsonsl_error_t error, |
1235
|
|
|
|
|
|
|
struct jsonsl_state_st *state, |
1236
|
|
|
|
|
|
|
char *at) |
1237
|
|
|
|
|
|
|
{ |
1238
|
|
|
|
|
|
|
/** |
1239
|
|
|
|
|
|
|
* This needs special handling, as we will be receiving a return |
1240
|
|
|
|
|
|
|
* value from Perl for this.. |
1241
|
|
|
|
|
|
|
*/ |
1242
|
0
|
|
|
|
|
|
die ("Got error: %s", jsonsl_strerror(error)); |
1243
|
|
|
|
|
|
|
return 0; |
1244
|
|
|
|
|
|
|
} |
1245
|
|
|
|
|
|
|
|
1246
|
|
|
|
|
|
|
#define pltuba_feed(tb,str) pltuba_feed_THX(aTHX_ tb,str) |
1247
|
|
|
|
|
|
|
static void |
1248
|
2
|
|
|
|
|
|
pltuba_feed_THX(pTHX_ PLTUBA *tuba, SV *input) |
1249
|
|
|
|
|
|
|
{ |
1250
|
2
|
50
|
|
|
|
|
if (!SvPOK(input)) { |
1251
|
0
|
|
|
|
|
|
die("Input is not string!"); |
1252
|
|
|
|
|
|
|
} |
1253
|
2
|
|
|
|
|
|
tuba->buf = input; |
1254
|
2
|
|
|
|
|
|
tuba->pos_min_valid = tuba->jsn->pos; |
1255
|
|
|
|
|
|
|
|
1256
|
2
|
|
|
|
|
|
SvREADONLY_on(input); |
1257
|
2
|
|
|
|
|
|
jsonsl_feed(tuba->jsn, SvPVX_const(input), SvCUR(input)); |
1258
|
2
|
50
|
|
|
|
|
if (tuba->keep_pos) { |
1259
|
0
|
|
|
|
|
|
pltuba_flush_characters(tuba, tuba->jsn->pos); |
1260
|
0
|
|
|
|
|
|
tuba->keep_pos = tuba->jsn->pos; |
1261
|
|
|
|
|
|
|
} |
1262
|
2
|
|
|
|
|
|
SvREADONLY_off(input); |
1263
|
2
|
|
|
|
|
|
} |
1264
|
|
|
|
|
|
|
|
1265
|
|
|
|
|
|
|
static SV * |
1266
|
2
|
|
|
|
|
|
pltuba_initialize_THX(pTHX_ const char *pkg) |
1267
|
|
|
|
|
|
|
{ |
1268
|
|
|
|
|
|
|
SV *ptriv, *retrv; |
1269
|
|
|
|
|
|
|
HV *hvret; |
1270
|
|
|
|
|
|
|
HV *subclass; |
1271
|
|
|
|
|
|
|
dMY_CXT; |
1272
|
|
|
|
|
|
|
|
1273
|
|
|
|
|
|
|
/* Initialize our internal C data structures */ |
1274
|
|
|
|
|
|
|
PLTUBA *tuba; |
1275
|
2
|
|
|
|
|
|
Newxz(tuba, 1, PLTUBA); |
1276
|
2
|
|
|
|
|
|
pljsonsl_common_initialize(&MY_CXT, (PLJSONSL*)tuba, PLJSONSL_MAX_DEFAULT); |
1277
|
|
|
|
|
|
|
|
1278
|
2
|
|
|
|
|
|
tuba->jsn->action_callback_PUSH = pltuba_jsonsl_push_callback; |
1279
|
2
|
|
|
|
|
|
tuba->jsn->action_callback_POP = pltuba_jsonsl_pop_callback; |
1280
|
2
|
|
|
|
|
|
tuba->jsn->error_callback = pltuba_jsonsl_error_callback; |
1281
|
2
|
|
|
|
|
|
jsonsl_enable_all_callbacks(tuba->jsn); |
1282
|
|
|
|
|
|
|
|
1283
|
2
|
|
|
|
|
|
ptriv = newSViv(PTR2IV(tuba)); |
1284
|
2
|
|
|
|
|
|
SvREADONLY_on(ptriv); |
1285
|
|
|
|
|
|
|
|
1286
|
|
|
|
|
|
|
/* The Perl object .. */ |
1287
|
2
|
|
|
|
|
|
hvret = newHV(); |
1288
|
2
|
|
|
|
|
|
(void)hv_stores(hvret, PLTUBA_HKEY_NAME, ptriv); |
1289
|
2
|
|
|
|
|
|
tuba->selfrv = newRV_inc((SV*)hvret); |
1290
|
2
|
|
|
|
|
|
sv_rvweaken(tuba->selfrv); |
1291
|
2
|
|
|
|
|
|
retrv = newRV_noinc((SV*)hvret); |
1292
|
|
|
|
|
|
|
|
1293
|
2
|
|
|
|
|
|
subclass = gv_stashpv(pkg, GV_ADD); |
1294
|
2
|
|
|
|
|
|
sv_bless(retrv, subclass); |
1295
|
|
|
|
|
|
|
|
1296
|
2
|
|
|
|
|
|
tuba->paramhv = newHV(); |
1297
|
2
|
|
|
|
|
|
tuba->paramhvrv = newRV_noinc((SV*)tuba->paramhv); |
1298
|
|
|
|
|
|
|
{ |
1299
|
2
|
|
|
|
|
|
SV *ksv = newSV(0); |
1300
|
|
|
|
|
|
|
HE *tmphe; |
1301
|
|
|
|
|
|
|
|
1302
|
|
|
|
|
|
|
#define X(kname) \ |
1303
|
|
|
|
|
|
|
sv_setpvs(ksv, #kname); \ |
1304
|
|
|
|
|
|
|
tmphe = hv_store_ent(tuba->paramhv, ksv, &PL_sv_undef, 0); \ |
1305
|
|
|
|
|
|
|
HeVAL(tmphe) = &PL_sv_placeholder; \ |
1306
|
|
|
|
|
|
|
assert(tmphe); \ |
1307
|
|
|
|
|
|
|
tuba->p_ents.pe_##kname.he = tmphe; |
1308
|
|
|
|
|
|
|
|
1309
|
2
|
|
|
|
|
|
PLTUBA_XPARAMS; |
1310
|
|
|
|
|
|
|
#undef X |
1311
|
|
|
|
|
|
|
} |
1312
|
|
|
|
|
|
|
|
1313
|
|
|
|
|
|
|
#define initialize_param_iv(b) \ |
1314
|
|
|
|
|
|
|
PLTUBA_PARAM_FIELD(tuba, b).sv = newSViv(0); \ |
1315
|
|
|
|
|
|
|
SvREADONLY_on(PLTUBA_PARAM_FIELD(tuba,b).sv); |
1316
|
|
|
|
|
|
|
#define initialize_param_dualvar(b) \ |
1317
|
|
|
|
|
|
|
PLTUBA_PARAM_FIELD(tuba, b).sv = newSViv(0); \ |
1318
|
|
|
|
|
|
|
sv_setpv(PLTUBA_PARAM_FIELD(tuba, b).sv, " "); \ |
1319
|
|
|
|
|
|
|
SvIOK_on(PLTUBA_PARAM_FIELD(tuba,b).sv); \ |
1320
|
|
|
|
|
|
|
SvREADONLY_on(PLTUBA_PARAM_FIELD(tuba,b).sv); |
1321
|
|
|
|
|
|
|
|
1322
|
2
|
|
|
|
|
|
initialize_param_iv(Index); |
1323
|
2
|
|
|
|
|
|
initialize_param_dualvar(Mode); |
1324
|
2
|
|
|
|
|
|
initialize_param_dualvar(Type); |
1325
|
|
|
|
|
|
|
|
1326
|
|
|
|
|
|
|
#undef initialize_param_iv |
1327
|
|
|
|
|
|
|
#undef initialize_param_dualvar |
1328
|
|
|
|
|
|
|
|
1329
|
2
|
|
|
|
|
|
SvREADONLY_on(tuba->paramhv); |
1330
|
2
|
|
|
|
|
|
return retrv; |
1331
|
|
|
|
|
|
|
} |
1332
|
|
|
|
|
|
|
|
1333
|
|
|
|
|
|
|
/** |
1334
|
|
|
|
|
|
|
* Initialize our thread-local context |
1335
|
|
|
|
|
|
|
*/ |
1336
|
|
|
|
|
|
|
#define POPULATE_CXT \ |
1337
|
|
|
|
|
|
|
MY_CXT.stash_obj = gv_stashpv(PLJSONSL_CLASS_NAME, GV_ADD); \ |
1338
|
|
|
|
|
|
|
MY_CXT.stash_boolean = gv_stashpv(PLJSONSL_BOOLEAN_NAME, GV_ADD); \ |
1339
|
|
|
|
|
|
|
MY_CXT.stash_tuba = gv_stashpv(PLTUBA_CLASS_NAME, GV_ADD); \ |
1340
|
|
|
|
|
|
|
MY_CXT.quick = NULL; |
1341
|
|
|
|
|
|
|
|
1342
|
|
|
|
|
|
|
|
1343
|
|
|
|
|
|
|
/** |
1344
|
|
|
|
|
|
|
* These two macros arrange for the contents of the result stack to be returned |
1345
|
|
|
|
|
|
|
* to perlspace. |
1346
|
|
|
|
|
|
|
*/ |
1347
|
|
|
|
|
|
|
#define dRESULT_VARS \ |
1348
|
|
|
|
|
|
|
int result_count; \ |
1349
|
|
|
|
|
|
|
int result_iter; \ |
1350
|
|
|
|
|
|
|
SV *result_sv; |
1351
|
|
|
|
|
|
|
|
1352
|
|
|
|
|
|
|
#define RETURN_RESULTS(pjsn) \ |
1353
|
|
|
|
|
|
|
switch(GIMME_V) { \ |
1354
|
|
|
|
|
|
|
case G_VOID: \ |
1355
|
|
|
|
|
|
|
result_count = 0; \ |
1356
|
|
|
|
|
|
|
break; \ |
1357
|
|
|
|
|
|
|
case G_SCALAR: \ |
1358
|
|
|
|
|
|
|
result_sv = av_shift(pjsn->results); \ |
1359
|
|
|
|
|
|
|
if (result_sv == &PL_sv_undef) { \ |
1360
|
|
|
|
|
|
|
result_count = 0; \ |
1361
|
|
|
|
|
|
|
break; \ |
1362
|
|
|
|
|
|
|
} \ |
1363
|
|
|
|
|
|
|
XPUSHs(sv_2mortal(result_sv)); \ |
1364
|
|
|
|
|
|
|
result_count = 1; \ |
1365
|
|
|
|
|
|
|
break; \ |
1366
|
|
|
|
|
|
|
case G_ARRAY: \ |
1367
|
|
|
|
|
|
|
result_count = av_len(pjsn->results) + 1; \ |
1368
|
|
|
|
|
|
|
if (result_count == 0) { \ |
1369
|
|
|
|
|
|
|
break; \ |
1370
|
|
|
|
|
|
|
} \ |
1371
|
|
|
|
|
|
|
EXTEND(SP, result_count); \ |
1372
|
|
|
|
|
|
|
for (result_iter = 0; result_iter < result_count; result_iter++) { \ |
1373
|
|
|
|
|
|
|
result_sv = av_delete(pjsn->results, result_iter, 0); \ |
1374
|
|
|
|
|
|
|
/*already mortal according to av_delete*/ \ |
1375
|
|
|
|
|
|
|
PUSHs(result_sv); \ |
1376
|
|
|
|
|
|
|
} \ |
1377
|
|
|
|
|
|
|
av_clear(pjsn->results); \ |
1378
|
|
|
|
|
|
|
break; \ |
1379
|
|
|
|
|
|
|
default: \ |
1380
|
|
|
|
|
|
|
die("eh? (RETURN_RESULTS)"); \ |
1381
|
|
|
|
|
|
|
result_count = 0; \ |
1382
|
|
|
|
|
|
|
break; \ |
1383
|
|
|
|
|
|
|
} |
1384
|
|
|
|
|
|
|
|
1385
|
|
|
|
|
|
|
|
1386
|
|
|
|
|
|
|
|
1387
|
|
|
|
|
|
|
|
1388
|
|
|
|
|
|
|
MODULE = JSON::SL PACKAGE = JSON::SL PREFIX = PLJSONSL_ |
1389
|
|
|
|
|
|
|
|
1390
|
|
|
|
|
|
|
PROTOTYPES: DISABLED |
1391
|
|
|
|
|
|
|
|
1392
|
|
|
|
|
|
|
BOOT: |
1393
|
|
|
|
|
|
|
{ |
1394
|
|
|
|
|
|
|
MY_CXT_INIT; |
1395
|
19
|
|
|
|
|
|
POPULATE_CXT; |
1396
|
19
|
|
|
|
|
|
PLJSONSL_ESCTBL_INIT(ESCTBL); |
1397
|
|
|
|
|
|
|
} |
1398
|
|
|
|
|
|
|
|
1399
|
|
|
|
|
|
|
SV * |
1400
|
|
|
|
|
|
|
PLJSONSL_new(SV *pkg, ...) |
1401
|
|
|
|
|
|
|
PREINIT: |
1402
|
|
|
|
|
|
|
PLJSONSL *pjsn; |
1403
|
|
|
|
|
|
|
SV *ptriv, *retrv; |
1404
|
|
|
|
|
|
|
int levels; |
1405
|
|
|
|
|
|
|
dMY_CXT; |
1406
|
|
|
|
|
|
|
CODE: |
1407
|
|
|
|
|
|
|
(void)pkg; |
1408
|
24
|
100
|
|
|
|
|
if (items > 1) { |
1409
|
3
|
50
|
|
|
|
|
if (!SvIOK(ST(1))) { |
1410
|
0
|
|
|
|
|
|
die("Second argument (if provided) must be numeric"); |
1411
|
|
|
|
|
|
|
} |
1412
|
3
|
50
|
|
|
|
|
levels = SvIV(ST(1)); |
1413
|
3
|
50
|
|
|
|
|
if (levels < 2) { |
1414
|
0
|
|
|
|
|
|
die ("Levels must be at least 2"); |
1415
|
|
|
|
|
|
|
} |
1416
|
|
|
|
|
|
|
} else { |
1417
|
21
|
|
|
|
|
|
levels = PLJSONSL_MAX_DEFAULT; |
1418
|
|
|
|
|
|
|
} |
1419
|
|
|
|
|
|
|
|
1420
|
24
|
|
|
|
|
|
Newxz(pjsn, 1, PLJSONSL); |
1421
|
24
|
|
|
|
|
|
pljsonsl_common_initialize(&MY_CXT, pjsn, levels); |
1422
|
24
|
|
|
|
|
|
ptriv = newSViv(PTR2IV(pjsn)); |
1423
|
24
|
|
|
|
|
|
retrv = newRV_noinc(ptriv); |
1424
|
24
|
|
|
|
|
|
sv_bless(retrv, MY_CXT.stash_obj); |
1425
|
24
|
|
|
|
|
|
pjsn->buf = newSVpvn("", 0); |
1426
|
|
|
|
|
|
|
|
1427
|
24
|
|
|
|
|
|
jsonsl_enable_all_callbacks(pjsn->jsn); |
1428
|
24
|
|
|
|
|
|
pjsn->jsn->action_callback = initial_callback; |
1429
|
24
|
|
|
|
|
|
pjsn->jsn->error_callback = error_callback; |
1430
|
|
|
|
|
|
|
|
1431
|
24
|
|
|
|
|
|
pjsn->results = newAV(); |
1432
|
|
|
|
|
|
|
PLJSONSL_INIT_KSV(pjsn); |
1433
|
24
|
|
|
|
|
|
RETVAL = retrv; |
1434
|
|
|
|
|
|
|
|
1435
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1436
|
|
|
|
|
|
|
|
1437
|
|
|
|
|
|
|
|
1438
|
|
|
|
|
|
|
void |
1439
|
|
|
|
|
|
|
PLJSONSL_set_jsonpointer(PLJSONSL *pjsn, AV *paths) |
1440
|
|
|
|
|
|
|
PPCODE: |
1441
|
4
|
|
|
|
|
|
pljsonsl_set_jsonpointer(pjsn, paths); |
1442
|
|
|
|
|
|
|
|
1443
|
|
|
|
|
|
|
SV * |
1444
|
|
|
|
|
|
|
PLJSONSL_root(PLJSONSL *pjsn) |
1445
|
|
|
|
|
|
|
CODE: |
1446
|
3
|
50
|
|
|
|
|
if (pjsn->root) { |
1447
|
3
|
|
|
|
|
|
RETVAL = newRV_inc(pjsn->root); |
1448
|
|
|
|
|
|
|
} else { |
1449
|
0
|
|
|
|
|
|
RETVAL = &PL_sv_undef; |
1450
|
|
|
|
|
|
|
} |
1451
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1452
|
|
|
|
|
|
|
|
1453
|
|
|
|
|
|
|
void |
1454
|
|
|
|
|
|
|
PLJSONSL__modify_readonly(PLJSONSL *pjsn, SV *ref) |
1455
|
|
|
|
|
|
|
ALIAS: |
1456
|
|
|
|
|
|
|
make_referrent_writeable = 1 |
1457
|
|
|
|
|
|
|
make_referrent_readonly = 2 |
1458
|
|
|
|
|
|
|
CODE: |
1459
|
3
|
50
|
|
|
|
|
if (!SvROK(ref)) { |
1460
|
0
|
|
|
|
|
|
die("Variable is not a reference!"); |
1461
|
|
|
|
|
|
|
} |
1462
|
3
|
50
|
|
|
|
|
if (ix == 0) { |
1463
|
0
|
|
|
|
|
|
PLJSONSL_CROAK_USAGE("use make_referrent_writeable or make_referrent_readonly"); |
1464
|
3
|
100
|
|
|
|
|
} else if (ix == 1) { |
1465
|
1
|
|
|
|
|
|
SvREADONLY_off(SvRV(ref)); |
1466
|
2
|
50
|
|
|
|
|
} else if (ix == 2) { |
1467
|
2
|
|
|
|
|
|
SvREADONLY_on(SvRV(ref)); |
1468
|
|
|
|
|
|
|
} |
1469
|
|
|
|
|
|
|
|
1470
|
|
|
|
|
|
|
int |
1471
|
|
|
|
|
|
|
PLJSONSL_referrent_is_writeable(PLJSONSL *pjsn, SV *ref) |
1472
|
|
|
|
|
|
|
CODE: |
1473
|
5
|
50
|
|
|
|
|
if (!SvROK(ref)) { |
1474
|
0
|
|
|
|
|
|
die("Variable is not a reference!"); |
1475
|
|
|
|
|
|
|
} |
1476
|
5
|
|
|
|
|
|
RETVAL = SvREADONLY(SvRV(ref)) == 0; |
1477
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1478
|
|
|
|
|
|
|
|
1479
|
|
|
|
|
|
|
|
1480
|
|
|
|
|
|
|
void |
1481
|
|
|
|
|
|
|
PLJSONSL_feed(PLJSONSL *pjsn, SV *input) |
1482
|
|
|
|
|
|
|
ALIAS: |
1483
|
|
|
|
|
|
|
incr_parse =1 |
1484
|
|
|
|
|
|
|
|
1485
|
|
|
|
|
|
|
PPCODE: |
1486
|
|
|
|
|
|
|
{ |
1487
|
|
|
|
|
|
|
dRESULT_VARS; |
1488
|
24
|
|
|
|
|
|
pljsonsl_feed_incr(pjsn, input); |
1489
|
20
|
50
|
|
|
|
|
RETURN_RESULTS(pjsn); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
1490
|
|
|
|
|
|
|
} |
1491
|
|
|
|
|
|
|
|
1492
|
|
|
|
|
|
|
void |
1493
|
|
|
|
|
|
|
PLJSONSL_fetch(PLJSONSL *pjsn) |
1494
|
|
|
|
|
|
|
PPCODE: |
1495
|
|
|
|
|
|
|
{ |
1496
|
|
|
|
|
|
|
dRESULT_VARS; |
1497
|
5
|
50
|
|
|
|
|
RETURN_RESULTS(pjsn); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
1498
|
|
|
|
|
|
|
} |
1499
|
|
|
|
|
|
|
|
1500
|
|
|
|
|
|
|
int |
1501
|
|
|
|
|
|
|
PLJSONSL__escape_table_chr(PLJSONSL *pjsn, U8 chrc, ...) |
1502
|
|
|
|
|
|
|
CODE: |
1503
|
2
|
50
|
|
|
|
|
if (chrc > 0x7f) { |
1504
|
0
|
|
|
|
|
|
warn("Attempted to set non-ASCII escape preference"); |
1505
|
0
|
|
|
|
|
|
RETVAL = -1; |
1506
|
|
|
|
|
|
|
} else { |
1507
|
2
|
|
|
|
|
|
RETVAL = pjsn->escape_table[chrc]; |
1508
|
2
|
50
|
|
|
|
|
if (items == 3) { |
1509
|
2
|
50
|
|
|
|
|
pjsn->escape_table[chrc] = SvIV(ST(2)); |
1510
|
|
|
|
|
|
|
} |
1511
|
|
|
|
|
|
|
} |
1512
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1513
|
|
|
|
|
|
|
|
1514
|
|
|
|
|
|
|
void |
1515
|
|
|
|
|
|
|
PLJSONSL_reset(PLJSONSL *pjsn) |
1516
|
|
|
|
|
|
|
CODE: |
1517
|
2
|
50
|
|
|
|
|
REFDEC_FIELD(pjsn, root); |
1518
|
|
|
|
|
|
|
|
1519
|
2
|
50
|
|
|
|
|
if (pjsn->results) { |
1520
|
2
|
|
|
|
|
|
av_clear(pjsn->results); |
1521
|
|
|
|
|
|
|
} |
1522
|
2
|
50
|
|
|
|
|
if (pjsn->buf) { |
1523
|
2
|
|
|
|
|
|
SvCUR_set(pjsn->buf, 0); |
1524
|
|
|
|
|
|
|
} |
1525
|
|
|
|
|
|
|
|
1526
|
2
|
|
|
|
|
|
jsonsl_reset(pjsn->jsn); |
1527
|
2
|
|
|
|
|
|
pjsn->pos_min_valid = 0; |
1528
|
2
|
|
|
|
|
|
pjsn->keep_pos = 0; |
1529
|
2
|
|
|
|
|
|
pjsn->curhk = NULL; |
1530
|
2
|
|
|
|
|
|
pjsn->jsn->action_callback_PUSH = initial_callback; |
1531
|
|
|
|
|
|
|
|
1532
|
|
|
|
|
|
|
SV* |
1533
|
|
|
|
|
|
|
PLJSONSL_root_callback(PLJSONSL *pjsn, SV *callback) |
1534
|
|
|
|
|
|
|
CODE: |
1535
|
4
|
|
|
|
|
|
RETVAL = pjsn->options.root_callback; |
1536
|
4
|
100
|
|
|
|
|
if (RETVAL) { |
1537
|
1
|
|
|
|
|
|
SvREFCNT_inc(RETVAL); |
1538
|
|
|
|
|
|
|
} else { |
1539
|
3
|
|
|
|
|
|
RETVAL = &PL_sv_undef; |
1540
|
|
|
|
|
|
|
} |
1541
|
|
|
|
|
|
|
|
1542
|
4
|
100
|
|
|
|
|
if (SvTYPE(callback) == SVt_NULL) { |
1543
|
1
|
50
|
|
|
|
|
if (pjsn->options.root_callback) { |
1544
|
1
|
|
|
|
|
|
SvREFCNT_dec(pjsn->options.root_callback); |
1545
|
1
|
|
|
|
|
|
pjsn->options.root_callback = NULL; |
1546
|
|
|
|
|
|
|
} |
1547
|
|
|
|
|
|
|
} else { |
1548
|
3
|
100
|
|
|
|
|
if (SvTYPE(callback) != SVt_RV || |
|
|
50
|
|
|
|
|
|
1549
|
2
|
|
|
|
|
|
SvTYPE(SvRV(callback)) != SVt_PVCV) { |
1550
|
1
|
|
|
|
|
|
die("Second argument must be undef or a CODE ref"); |
1551
|
|
|
|
|
|
|
} |
1552
|
2
|
50
|
|
|
|
|
if (pjsn->options.root_callback) { |
1553
|
0
|
|
|
|
|
|
SvREFCNT_dec(pjsn->options.root_callback); |
1554
|
|
|
|
|
|
|
} |
1555
|
2
|
|
|
|
|
|
pjsn->options.root_callback = newRV_inc(SvRV(callback)); |
1556
|
|
|
|
|
|
|
} |
1557
|
|
|
|
|
|
|
|
1558
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1559
|
|
|
|
|
|
|
|
1560
|
|
|
|
|
|
|
void |
1561
|
|
|
|
|
|
|
PLJSONSL_DESTROY(PLJSONSL *pjsn) |
1562
|
|
|
|
|
|
|
PREINIT: |
1563
|
|
|
|
|
|
|
int ii; |
1564
|
|
|
|
|
|
|
|
1565
|
|
|
|
|
|
|
CODE: |
1566
|
24
|
50
|
|
|
|
|
if (pjsn->priv_global.is_global == 0) { |
1567
|
24
|
100
|
|
|
|
|
REFDEC_FIELD(pjsn, root); |
1568
|
24
|
50
|
|
|
|
|
REFDEC_FIELD(pjsn, results); |
1569
|
24
|
50
|
|
|
|
|
REFDEC_FIELD(pjsn, buf); |
1570
|
24
|
100
|
|
|
|
|
REFDEC_FIELD(pjsn, options.root_callback); |
1571
|
|
|
|
|
|
|
} /* else, it's a mortal and shouldn't be freed */ |
1572
|
24
|
|
|
|
|
|
jsonsl_jpr_match_state_cleanup(pjsn->jsn); |
1573
|
24
|
100
|
|
|
|
|
if (pjsn->jprs) { |
1574
|
8
|
100
|
|
|
|
|
for ( ii = 0; ii < pjsn->njprs; ii++) { |
1575
|
4
|
50
|
|
|
|
|
if (pjsn->jprs[ii] == NULL) { |
1576
|
0
|
|
|
|
|
|
break; |
1577
|
|
|
|
|
|
|
} |
1578
|
4
|
|
|
|
|
|
jsonsl_jpr_destroy(pjsn->jprs[ii]); |
1579
|
|
|
|
|
|
|
} |
1580
|
4
|
|
|
|
|
|
Safefree(pjsn->jprs); |
1581
|
4
|
|
|
|
|
|
pjsn->jprs = NULL; |
1582
|
|
|
|
|
|
|
} |
1583
|
24
|
50
|
|
|
|
|
if (pjsn->jsn) { |
1584
|
24
|
|
|
|
|
|
jsonsl_destroy(pjsn->jsn); |
1585
|
24
|
|
|
|
|
|
pjsn->jsn = NULL; |
1586
|
|
|
|
|
|
|
} |
1587
|
|
|
|
|
|
|
PLJSONSL_DESTROY_KSV(pjsn); |
1588
|
24
|
|
|
|
|
|
Safefree(pjsn); |
1589
|
|
|
|
|
|
|
|
1590
|
|
|
|
|
|
|
void |
1591
|
|
|
|
|
|
|
PLJSONSL_decode_json(SV *input) |
1592
|
|
|
|
|
|
|
PREINIT: |
1593
|
|
|
|
|
|
|
PLJSONSL* pjsn; |
1594
|
|
|
|
|
|
|
dRESULT_VARS; |
1595
|
|
|
|
|
|
|
|
1596
|
|
|
|
|
|
|
PPCODE: |
1597
|
32
|
|
|
|
|
|
pjsn = pljsonsl_get_and_initialize_global(aTHX); |
1598
|
32
|
|
|
|
|
|
pljsonsl_feed_oneshot(pjsn, input); |
1599
|
|
|
|
|
|
|
|
1600
|
30
|
|
|
|
|
|
pjsn->curhk = NULL; |
1601
|
30
|
|
|
|
|
|
pjsn->keep_pos = 0; |
1602
|
30
|
|
|
|
|
|
pjsn->pos_min_valid = 0; |
1603
|
30
|
|
|
|
|
|
pjsn->jsn->action_callback_PUSH = initial_callback; |
1604
|
|
|
|
|
|
|
|
1605
|
30
|
100
|
|
|
|
|
RETURN_RESULTS(pjsn); |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
1606
|
30
|
100
|
|
|
|
|
if (result_count == 0 && av_len(pjsn->results) == -1) { |
|
|
50
|
|
|
|
|
|
1607
|
1
|
|
|
|
|
|
die("Incomplete JSON string?"); |
1608
|
|
|
|
|
|
|
} |
1609
|
|
|
|
|
|
|
|
1610
|
|
|
|
|
|
|
SV * |
1611
|
|
|
|
|
|
|
PLJSONSL_unescape_json_string(SV *input) |
1612
|
|
|
|
|
|
|
PREINIT: |
1613
|
|
|
|
|
|
|
size_t origlen, newlen; |
1614
|
3
|
|
|
|
|
|
SV *retsv = NULL; |
1615
|
|
|
|
|
|
|
char *errpos; |
1616
|
|
|
|
|
|
|
jsonsl_error_t err; |
1617
|
|
|
|
|
|
|
jsonsl_special_t flags; |
1618
|
|
|
|
|
|
|
|
1619
|
|
|
|
|
|
|
CODE: |
1620
|
3
|
50
|
|
|
|
|
if (!SvPOK(input)) { |
1621
|
0
|
|
|
|
|
|
die("Input is not a valid string"); |
1622
|
|
|
|
|
|
|
} |
1623
|
3
|
|
|
|
|
|
origlen = SvCUR(input); |
1624
|
3
|
50
|
|
|
|
|
if (origlen) { |
1625
|
3
|
|
|
|
|
|
retsv = newSV(origlen); |
1626
|
3
|
|
|
|
|
|
newlen = jsonsl_util_unescape_ex(SvPVX_const(input), SvPVX(retsv), |
1627
|
3
|
|
|
|
|
|
SvCUR(input), ESCTBL, &flags, |
1628
|
|
|
|
|
|
|
&err, (const char**)&errpos); |
1629
|
3
|
100
|
|
|
|
|
if (newlen == 0) { |
1630
|
1
|
|
|
|
|
|
SvREFCNT_dec(retsv); |
1631
|
1
|
|
|
|
|
|
die("Could not unescape: %s at pos %lu ('%c'..)", |
1632
|
|
|
|
|
|
|
jsonsl_strerror(err), |
1633
|
1
|
|
|
|
|
|
errpos - SvPVX_const(input), |
1634
|
1
|
|
|
|
|
|
*errpos |
1635
|
|
|
|
|
|
|
); |
1636
|
|
|
|
|
|
|
} |
1637
|
|
|
|
|
|
|
|
1638
|
2
|
|
|
|
|
|
SvCUR_set(retsv, newlen); |
1639
|
2
|
|
|
|
|
|
SvPOK_only(retsv); |
1640
|
2
|
50
|
|
|
|
|
if (SvUTF8(input) || (flags & JSONSL_SPECIALf_NONASCII)) { |
|
|
100
|
|
|
|
|
|
1641
|
2
|
|
|
|
|
|
SvUTF8_on(retsv); |
1642
|
|
|
|
|
|
|
} |
1643
|
|
|
|
|
|
|
} else { |
1644
|
0
|
|
|
|
|
|
retsv = &PL_sv_undef; |
1645
|
|
|
|
|
|
|
} |
1646
|
2
|
|
|
|
|
|
RETVAL = retsv; |
1647
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1648
|
|
|
|
|
|
|
|
1649
|
|
|
|
|
|
|
|
1650
|
|
|
|
|
|
|
|
1651
|
|
|
|
|
|
|
void |
1652
|
|
|
|
|
|
|
PLJSONSL_CLONE(PLJSONSL *pjsn) |
1653
|
|
|
|
|
|
|
CODE: |
1654
|
|
|
|
|
|
|
{ |
1655
|
|
|
|
|
|
|
MY_CXT_CLONE; |
1656
|
0
|
|
|
|
|
|
POPULATE_CXT; |
1657
|
|
|
|
|
|
|
} |
1658
|
|
|
|
|
|
|
|
1659
|
|
|
|
|
|
|
MODULE = JSON::SL PACKAGE = JSON::SL::Tuba PREFIX = PLTUBA_ |
1660
|
|
|
|
|
|
|
|
1661
|
|
|
|
|
|
|
SV * |
1662
|
|
|
|
|
|
|
PLTUBA__initialize(const char *pkg) |
1663
|
|
|
|
|
|
|
CODE: |
1664
|
2
|
|
|
|
|
|
RETVAL = pltuba_initialize_THX(aTHX_ pkg); |
1665
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1666
|
|
|
|
|
|
|
|
1667
|
|
|
|
|
|
|
void |
1668
|
|
|
|
|
|
|
PLTUBA_DESTROY(PLTUBA* tuba) |
1669
|
|
|
|
|
|
|
CODE: |
1670
|
2
|
|
|
|
|
|
jsonsl_destroy(tuba->jsn); |
1671
|
2
|
|
|
|
|
|
tuba->jsn = NULL; |
1672
|
|
|
|
|
|
|
|
1673
|
2
|
50
|
|
|
|
|
REFDEC_FIELD(tuba, accum); |
1674
|
2
|
50
|
|
|
|
|
REFDEC_FIELD(tuba, kaccum); |
1675
|
2
|
50
|
|
|
|
|
REFDEC_FIELD(tuba, selfrv); |
1676
|
|
|
|
|
|
|
#define X(kname) \ |
1677
|
|
|
|
|
|
|
PLTUBA_RESET_PARAMFIELD(tuba, kname); \ |
1678
|
|
|
|
|
|
|
REFDEC_FIELD(tuba, p_ents.pe_##kname.sv); |
1679
|
2
|
50
|
|
|
|
|
PLTUBA_XPARAMS; |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
1680
|
|
|
|
|
|
|
#undef X |
1681
|
2
|
50
|
|
|
|
|
REFDEC_FIELD(tuba, paramhvrv); |
1682
|
|
|
|
|
|
|
/* Implicit that the hash has been decrementas as well. |
1683
|
|
|
|
|
|
|
* Don't do another dec |
1684
|
|
|
|
|
|
|
*/ |
1685
|
2
|
|
|
|
|
|
tuba->paramhv = NULL; |
1686
|
2
|
|
|
|
|
|
pltuba_invalidate_gvs_THX(aTHX_ tuba); |
1687
|
2
|
|
|
|
|
|
Safefree(tuba); |
1688
|
|
|
|
|
|
|
|
1689
|
|
|
|
|
|
|
int |
1690
|
|
|
|
|
|
|
PLTUBA__ax_opt(PLTUBA *tuba, int mode, ...) |
1691
|
|
|
|
|
|
|
CODE: |
1692
|
20
|
|
|
|
|
|
RETVAL = tuba->accum_options[mode & 0xff]; |
1693
|
20
|
100
|
|
|
|
|
if (items > 2) { |
1694
|
15
|
50
|
|
|
|
|
tuba->accum_options[mode & 0xff] = SvIV(ST(2)); |
1695
|
|
|
|
|
|
|
} |
1696
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1697
|
|
|
|
|
|
|
|
1698
|
|
|
|
|
|
|
int |
1699
|
|
|
|
|
|
|
PLTUBA_accum_kv(PLTUBA *tuba, ...) |
1700
|
|
|
|
|
|
|
CODE: |
1701
|
4
|
50
|
|
|
|
|
if (items > 2) { |
1702
|
0
|
|
|
|
|
|
die("accum_kv(..boolean)"); |
1703
|
|
|
|
|
|
|
} |
1704
|
4
|
|
|
|
|
|
RETVAL = tuba->options.accum_kv; |
1705
|
4
|
100
|
|
|
|
|
if (items == 2) { |
1706
|
3
|
50
|
|
|
|
|
int newval = SvIV(ST(1)); |
1707
|
3
|
50
|
|
|
|
|
if (newval) { |
1708
|
3
|
|
|
|
|
|
tuba->accum_options['#'] = 1; |
1709
|
|
|
|
|
|
|
} |
1710
|
3
|
|
|
|
|
|
tuba->options.accum_kv = newval; |
1711
|
|
|
|
|
|
|
} |
1712
|
|
|
|
|
|
|
OUTPUT: RETVAL |
1713
|
|
|
|
|
|
|
|
1714
|
|
|
|
|
|
|
|
1715
|
|
|
|
|
|
|
void |
1716
|
|
|
|
|
|
|
PLTUBA__parse(PLTUBA* tuba, SV *input) |
1717
|
|
|
|
|
|
|
CODE: |
1718
|
2
|
|
|
|
|
|
pltuba_feed(tuba, input); |
1719
|
|
|
|
|
|
|
|
1720
|
|
|
|
|
|
|
|
1721
|
|
|
|
|
|
|
INCLUDE: srcout/option_accessors.xs |