line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
38
|
|
|
38
|
|
71847
|
use v5.10.1; |
|
38
|
|
|
|
|
141
|
|
2
|
|
|
|
|
|
|
package REST::Neo4p::ParseStream; |
3
|
38
|
|
|
38
|
|
240
|
use base Exporter; |
|
38
|
|
|
|
|
72
|
|
|
38
|
|
|
|
|
3000
|
|
4
|
38
|
|
|
38
|
|
21302
|
use HOP::Stream qw/node promise/; |
|
38
|
|
|
|
|
64164
|
|
|
38
|
|
|
|
|
2909
|
|
5
|
|
|
|
|
|
|
|
6
|
38
|
|
|
38
|
|
290
|
use strict; |
|
38
|
|
|
|
|
79
|
|
|
38
|
|
|
|
|
776
|
|
7
|
38
|
|
|
38
|
|
178
|
use warnings; |
|
38
|
|
|
|
|
86
|
|
|
38
|
|
|
|
|
1282
|
|
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
BEGIN { |
10
|
38
|
|
|
38
|
|
5656
|
$REST::Neo4p::ParseStream::VERSION = '0.4003'; |
11
|
|
|
|
|
|
|
} |
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
our @EXPORT = qw/j_parse/;# j_parse_object j_parse_array /; |
14
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
# lazy linked lists |
16
|
|
|
|
|
|
|
# qry response in txn - "data":[{ "row" : [...]},...] |
17
|
|
|
|
|
|
|
# qry response for qry - "data":[ [{...},...],... ] |
18
|
|
|
|
|
|
|
# so - for txn response, j_parse_query_response returns hashes, |
19
|
|
|
|
|
|
|
# for qry response, j_parse_query_response returns arrays |
20
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
sub j_parse { |
22
|
15
|
|
|
15
|
0
|
8032
|
my $j = shift; |
23
|
15
|
|
|
|
|
30
|
state $state; # undef == first call? |
24
|
15
|
100
|
|
|
|
485
|
if ($j->incr_text =~ s/^\s*\[\s*//) { |
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
25
|
|
|
|
|
|
|
# batch (simple array of objects) |
26
|
1
|
|
|
|
|
6
|
return ['BATCH', j_parse_array($j)]; |
27
|
|
|
|
|
|
|
} |
28
|
|
|
|
|
|
|
elsif ($j->incr_text =~ s/^\s*{\s*//) { |
29
|
|
|
|
|
|
|
# object |
30
|
14
|
|
|
|
|
31
|
my $type; |
31
|
38
|
|
|
38
|
|
1462
|
use experimental 'smartmatch'; |
|
38
|
|
|
|
|
7927
|
|
|
38
|
|
|
|
|
278
|
|
32
|
14
|
|
|
|
|
39
|
given ($j->incr_text) { |
33
|
14
|
|
|
|
|
62
|
when (/^\s*"commit"/i) { |
34
|
8
|
|
|
|
|
27
|
$type = 'TXN'; |
35
|
|
|
|
|
|
|
} |
36
|
6
|
|
|
|
|
20
|
when (/^\s*"columns"/i) { |
37
|
5
|
|
|
|
|
14
|
$type = 'QUERY'; |
38
|
|
|
|
|
|
|
} |
39
|
1
|
|
|
|
|
3
|
default { |
40
|
1
|
|
|
|
|
4
|
$type = 'OBJECT'; |
41
|
|
|
|
|
|
|
} |
42
|
|
|
|
|
|
|
} |
43
|
14
|
|
|
|
|
45
|
return [$type, j_parse_object($j)] |
44
|
|
|
|
|
|
|
} |
45
|
|
|
|
|
|
|
elsif ($j->incr_text =~ m/^\s*"[a-zA-Z_]+"/) { |
46
|
|
|
|
|
|
|
# after a stream, next key of object |
47
|
0
|
|
|
|
|
0
|
return ['OBJECT', j_parse_object($j)]; |
48
|
|
|
|
|
|
|
} |
49
|
|
|
|
|
|
|
else { |
50
|
|
|
|
|
|
|
# problem |
51
|
0
|
|
|
|
|
0
|
return; |
52
|
|
|
|
|
|
|
} |
53
|
|
|
|
|
|
|
} |
54
|
|
|
|
|
|
|
|
55
|
|
|
|
|
|
|
# generic parse array stream |
56
|
|
|
|
|
|
|
# return decoded json entities at the top level |
57
|
|
|
|
|
|
|
# opening [ must be removed |
58
|
|
|
|
|
|
|
# handle empty array too |
59
|
|
|
|
|
|
|
sub j_parse_array { |
60
|
17
|
|
|
17
|
0
|
34
|
my $j = shift; |
61
|
17
|
|
|
|
|
28
|
my $po; |
62
|
|
|
|
|
|
|
$po = sub { |
63
|
474
|
|
|
474
|
|
6361
|
my $elt; |
64
|
474
|
|
|
|
|
679
|
my $int_po = $po; |
65
|
474
|
|
|
|
|
641
|
state $last_text; |
66
|
474
|
|
|
|
|
668
|
my $done = eval {$j->incr_text =~ s/^\s*\]\s*//}; |
|
474
|
|
|
|
|
15483
|
|
67
|
474
|
100
|
|
|
|
1228
|
return node(undef,undef) if $done; |
68
|
470
|
|
|
|
|
654
|
eval { |
69
|
470
|
|
|
|
|
6471
|
$elt = $j->incr_parse; |
70
|
|
|
|
|
|
|
}; |
71
|
470
|
100
|
|
|
|
1123
|
if (defined $elt) { |
|
|
100
|
|
|
|
|
|
72
|
412
|
|
|
|
|
1345
|
$last_text = $j->incr_text; |
73
|
412
|
100
|
|
|
|
1587
|
if ($j->incr_text =~ m/^\]}\],/) { # JSON::XS <=3.04 transaction kludge |
|
|
100
|
|
|
|
|
|
74
|
5
|
|
|
|
|
44
|
$j->incr_text =~ s/(\])(}\],)//; |
75
|
5
|
|
|
|
|
14
|
$done=1; |
76
|
|
|
|
|
|
|
} |
77
|
|
|
|
|
|
|
elsif ($elt eq 'transaction') { # JSON::XS 4.02 transaction kludge |
78
|
4
|
|
|
|
|
20
|
$j->incr_text = '"transaction"'.$j->incr_text; |
79
|
4
|
|
|
|
|
15
|
return node(undef,undef); |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
else { |
82
|
403
|
|
|
|
|
10017
|
$j->incr_text =~ s/^(\s*,\s*)|(\s*\]\s*)//; |
83
|
403
|
|
|
|
|
1269
|
$done = !!$2; |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
} |
87
|
|
|
|
|
|
|
elsif ($@) { |
88
|
1
|
50
|
33
|
|
|
15
|
if ($@ =~ /already started parsing/) { |
|
|
50
|
|
|
|
|
|
89
|
0
|
|
|
|
|
0
|
$elt = 'PENDING'; |
90
|
|
|
|
|
|
|
} |
91
|
|
|
|
|
|
|
elsif ($@ =~ /must be an object or array/ && |
92
|
|
|
|
|
|
|
$last_text =~ /("[^"]+")/) { |
93
|
|
|
|
|
|
|
# txn kludge |
94
|
0
|
|
|
|
|
0
|
$j->incr_skip; |
95
|
0
|
|
|
|
|
0
|
$j->incr_text = $1.$j->incr_text; |
96
|
0
|
|
|
|
|
0
|
return node(undef,undef); |
97
|
|
|
|
|
|
|
} |
98
|
|
|
|
|
|
|
else { |
99
|
1
|
|
|
|
|
7
|
die "j_parse: $@"; |
100
|
|
|
|
|
|
|
} |
101
|
|
|
|
|
|
|
} |
102
|
|
|
|
|
|
|
else { |
103
|
57
|
|
|
|
|
92
|
$elt = 'PENDING'; |
104
|
|
|
|
|
|
|
} |
105
|
465
|
100
|
|
|
|
2613
|
node(['ARELT'=>$elt], $done ? undef : promise { $po->() }); |
|
429
|
|
|
|
|
215182
|
|
106
|
17
|
|
|
|
|
114
|
}; |
107
|
17
|
|
|
|
|
63
|
return $po; |
108
|
|
|
|
|
|
|
} |
109
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
# generic parse object to one level |
111
|
|
|
|
|
|
|
# opening { must be removed |
112
|
|
|
|
|
|
|
# when another stream is returned as the value, |
113
|
|
|
|
|
|
|
# another call to j_parse_object has to wait until that stream is |
114
|
|
|
|
|
|
|
# exhausted... |
115
|
|
|
|
|
|
|
sub j_parse_object { |
116
|
23
|
|
|
23
|
0
|
43
|
my $j = shift; |
117
|
23
|
|
|
|
|
34
|
my $po; |
118
|
|
|
|
|
|
|
$po = sub { |
119
|
84
|
|
|
84
|
|
1268
|
state $key; |
120
|
84
|
|
|
|
|
132
|
state $current = ''; |
121
|
84
|
|
|
|
|
196
|
my ($text, $head,$obj); |
122
|
84
|
|
|
|
|
0
|
my $done; |
123
|
84
|
50
|
|
|
|
204
|
unless ($current eq 'PENDING') { |
124
|
84
|
|
|
|
|
119
|
my $m; |
125
|
38
|
|
|
38
|
|
29303
|
use experimental 'smartmatch'; |
|
38
|
|
|
|
|
95
|
|
|
38
|
|
|
|
|
240
|
|
126
|
84
|
|
|
|
|
131
|
eval { |
127
|
84
|
|
|
|
|
599
|
$j->incr_text =~ m/^(?:(\s*"([^"]+)"\s*:\s*)|(\s*}\s*))/; # look ahead |
128
|
84
|
|
100
|
|
|
341
|
$m = $2||$3; |
129
|
84
|
100
|
100
|
|
|
392
|
if ($m && ($m eq 'columns') && ($current eq 'RESULTS_STREAM')) { |
|
|
|
100
|
|
|
|
|
130
|
|
|
|
|
|
|
# if this is a 'results' item, don't eat the 'columns', |
131
|
|
|
|
|
|
|
# let the results stream do it |
132
|
8
|
|
|
|
|
27
|
$m = undef; |
133
|
|
|
|
|
|
|
} |
134
|
|
|
|
|
|
|
else { |
135
|
76
|
|
|
|
|
886
|
$j->incr_text =~ s/^(?:(\s*"([^"]+)"\s*:\s*)|(\s*}\s*))//; # consume |
136
|
|
|
|
|
|
|
} |
137
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
}; |
139
|
|
|
|
|
|
|
|
140
|
84
|
100
|
66
|
|
|
358
|
if ($@ =~ /already started parsing/ || !$m) { |
|
|
50
|
|
|
|
|
|
141
|
|
|
|
|
|
|
# either another function instance is in the middle of parsing, |
142
|
|
|
|
|
|
|
# or no key was found (which is the same thing) |
143
|
|
|
|
|
|
|
# so report where this instance is: |
144
|
25
|
|
|
|
|
154
|
return node([$key => $current], promise { $po->() }); |
|
10
|
|
|
|
|
7426
|
|
145
|
|
|
|
|
|
|
} |
146
|
|
|
|
|
|
|
elsif ($@) { |
147
|
0
|
|
|
|
|
0
|
die "j_parse: incr parser error: $@"; |
148
|
|
|
|
|
|
|
} |
149
|
59
|
|
|
|
|
152
|
$key = $m; |
150
|
|
|
|
|
|
|
} |
151
|
38
|
|
|
38
|
|
11394
|
use experimental 'smartmatch'; |
|
38
|
|
|
|
|
98
|
|
|
38
|
|
|
|
|
212
|
|
152
|
59
|
|
|
|
|
97
|
given ($key) { |
153
|
59
|
|
|
|
|
183
|
when ('columns') { |
154
|
14
|
|
|
|
|
26
|
eval { |
155
|
14
|
|
|
|
|
81
|
$obj = $j->incr_parse; |
156
|
|
|
|
|
|
|
}; |
157
|
14
|
50
|
|
|
|
51
|
die "j_parse: incr parser error: $@" if $@; |
158
|
14
|
|
|
|
|
36
|
$current = 'COMPLETE'; |
159
|
|
|
|
|
|
|
} |
160
|
45
|
|
|
|
|
96
|
when (/commit/) { |
161
|
8
|
|
|
|
|
99
|
$j->incr_text =~ s/^"([^"]+)"\s*,?\s*//; |
162
|
8
|
|
|
|
|
23
|
$obj = $1; |
163
|
8
|
|
|
|
|
23
|
$current = 'COMPLETE'; |
164
|
|
|
|
|
|
|
} |
165
|
37
|
|
|
|
|
68
|
when (/transaction/) { |
166
|
6
|
|
|
|
|
14
|
eval { |
167
|
6
|
|
|
|
|
39
|
$obj = $j->incr_parse; # get txn info obj |
168
|
|
|
|
|
|
|
}; |
169
|
6
|
50
|
|
|
|
22
|
die "j_parse: incr parser error: $@" if $@; |
170
|
6
|
|
|
|
|
18
|
$current = 'COMPLETE'; |
171
|
|
|
|
|
|
|
} |
172
|
31
|
|
|
|
|
111
|
when ('data') { |
173
|
12
|
100
|
|
|
|
195
|
if ($j->incr_text =~ s/^\[\s*//) { |
174
|
10
|
|
|
|
|
35
|
$obj = j_parse_array($j); |
175
|
10
|
|
|
|
|
32
|
$current = 'DATA_STREAM'; |
176
|
|
|
|
|
|
|
} |
177
|
|
|
|
|
|
|
else { |
178
|
2
|
|
|
|
|
21
|
die "j_parse: expecting an array value for 'data' key"; |
179
|
|
|
|
|
|
|
} |
180
|
|
|
|
|
|
|
} |
181
|
19
|
|
|
|
|
36
|
when ('results') { |
182
|
8
|
50
|
|
|
|
95
|
if ($j->incr_text =~ s/^\[\s*//) { |
|
|
0
|
|
|
|
|
|
183
|
8
|
|
|
|
|
96
|
$j->incr_text =~ s/^\s*{\s*//; |
184
|
8
|
|
|
|
|
21
|
eval { |
185
|
8
|
|
|
|
|
20
|
$obj = j_parse_object($j); |
186
|
|
|
|
|
|
|
}; |
187
|
8
|
50
|
|
|
|
25
|
die "j_parse: incr parser error: $@" if $@; |
188
|
8
|
|
|
|
|
43
|
$current = 'RESULTS_STREAM'; |
189
|
|
|
|
|
|
|
} |
190
|
|
|
|
|
|
|
elsif ($j->incr_text eq '') { |
191
|
0
|
|
|
|
|
0
|
$current = 'DONE'; |
192
|
0
|
|
|
|
|
0
|
$done=1; |
193
|
|
|
|
|
|
|
} |
194
|
|
|
|
|
|
|
} |
195
|
11
|
|
|
|
|
26
|
when ('errors') { |
196
|
6
|
50
|
|
|
|
37
|
if ($j->incr_text=~ s/^\[\s*//) { |
|
|
0
|
|
|
|
|
|
197
|
6
|
|
|
|
|
17
|
$obj = j_parse_array($j); |
198
|
6
|
|
|
|
|
20
|
$current = 'ERRORS_STREAM'; |
199
|
|
|
|
|
|
|
} |
200
|
|
|
|
|
|
|
elsif ($j->incr_text =~ s/^\s*\]\s*,?\s*//) { |
201
|
0
|
|
|
|
|
0
|
$current = 'DONE'; |
202
|
0
|
|
|
|
|
0
|
$done=1; |
203
|
|
|
|
|
|
|
} |
204
|
|
|
|
|
|
|
} |
205
|
5
|
|
|
|
|
17
|
when (/}/) { |
206
|
3
|
50
|
|
|
|
10
|
if ($current eq 'DATA_STREAM') { |
207
|
3
|
100
|
|
|
|
68
|
if ($j->incr_text =~ s/^\s*,\s*{//) { |
|
|
100
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
208
|
|
|
|
|
|
|
# prepared for next results object |
209
|
1
|
|
|
|
|
6
|
$obj = j_parse_object($j); |
210
|
1
|
|
|
|
|
3
|
$key = 'results'; |
211
|
1
|
|
|
|
|
4
|
$current = 'RESULTS_STREAM'; |
212
|
|
|
|
|
|
|
} |
213
|
|
|
|
|
|
|
elsif ($j->incr_text =~ s/^\s*\]\s*,?\s*//) { |
214
|
1
|
|
|
|
|
3
|
$current = 'DONE'; |
215
|
1
|
|
|
|
|
4
|
$done=1; |
216
|
|
|
|
|
|
|
} |
217
|
|
|
|
|
|
|
elsif ($j->incr_text eq '') { |
218
|
1
|
|
|
|
|
3
|
$current = 'DONE'; |
219
|
1
|
|
|
|
|
5
|
$done=1; |
220
|
|
|
|
|
|
|
} |
221
|
|
|
|
|
|
|
} |
222
|
|
|
|
|
|
|
else { |
223
|
0
|
|
|
|
|
0
|
$current = 'DONE'; |
224
|
0
|
|
|
|
|
0
|
$done=1; |
225
|
|
|
|
|
|
|
} |
226
|
|
|
|
|
|
|
} |
227
|
2
|
|
|
|
|
7
|
when (undef) { |
228
|
0
|
|
|
|
|
0
|
die "j_parse: No key found"; |
229
|
|
|
|
|
|
|
} |
230
|
2
|
|
|
|
|
5
|
default { |
231
|
|
|
|
|
|
|
# why am I here? |
232
|
2
|
|
|
|
|
23
|
die "j_parse: Unexpected key '$key' in stream"; |
233
|
|
|
|
|
|
|
} |
234
|
|
|
|
|
|
|
} |
235
|
55
|
100
|
|
|
|
117
|
if (defined $obj) { |
236
|
53
|
|
|
|
|
111
|
$head = [$key => $obj]; |
237
|
53
|
|
|
|
|
2041
|
$j->incr_text =~ s/^(?:(\s*,\s*)|(\s*}\s*))//; |
238
|
53
|
|
|
|
|
138
|
$done = !!$2; |
239
|
|
|
|
|
|
|
} |
240
|
|
|
|
|
|
|
else { |
241
|
2
|
50
|
|
|
|
9
|
$head = $done ? undef : [$key => $current = 'PENDING']; |
242
|
|
|
|
|
|
|
} |
243
|
55
|
50
|
|
|
|
326
|
return node($head, $done ? undef : promise { $po->() }) if $head; |
|
52
|
100
|
|
|
|
13389
|
|
244
|
2
|
|
|
|
|
11
|
return node(undef, undef); |
245
|
23
|
|
|
|
|
145
|
}; |
246
|
23
|
|
|
|
|
129
|
return $po; |
247
|
|
|
|
|
|
|
} |
248
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
=head1 NAME |
250
|
|
|
|
|
|
|
|
251
|
|
|
|
|
|
|
REST::Neo4p::ParseStream - Parse Neo4j REST responses on the fly |
252
|
|
|
|
|
|
|
|
253
|
|
|
|
|
|
|
=head1 SYNOPSIS |
254
|
|
|
|
|
|
|
|
255
|
|
|
|
|
|
|
Not for human consumption. |
256
|
|
|
|
|
|
|
This module is ignored by the Neo4j::Driver-based agent. |
257
|
|
|
|
|
|
|
|
258
|
|
|
|
|
|
|
=head1 DESCRIPTION |
259
|
|
|
|
|
|
|
|
260
|
|
|
|
|
|
|
This module helps L exploit the L |
261
|
|
|
|
|
|
|
server's chunked transfer encoding of its JSON REST responses. It is |
262
|
|
|
|
|
|
|
based on the fast L incremental parser and |
263
|
|
|
|
|
|
|
L's L
|
264
|
|
|
|
|
|
|
Perl|http://hop.perl.plover.com> ideas as implemented in |
265
|
|
|
|
|
|
|
L. |
266
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
The goal is to be able to pull in objects from the server stream as |
268
|
|
|
|
|
|
|
soon as they are available. In practice, this means specifically |
269
|
|
|
|
|
|
|
finding and incrementally processing the potentially large arrays of |
270
|
|
|
|
|
|
|
objects that are returned from cypher queries, transaction queries, |
271
|
|
|
|
|
|
|
and batch requests. |
272
|
|
|
|
|
|
|
|
273
|
|
|
|
|
|
|
Because of inconsistencies among the Neo4j response formats for each |
274
|
|
|
|
|
|
|
of these functions, this module does a significant amount of |
275
|
|
|
|
|
|
|
"hand-parsing". Currently the code will not be very robust to changes |
276
|
|
|
|
|
|
|
in those response formats. If you find your query handling is breaking |
277
|
|
|
|
|
|
|
with a new server version, L
|
278
|
|
|
|
|
|
|
ticket|https://rt.cpan.org/Public/Bug/Report.html?Queue=REST-Neo4p>. In |
279
|
|
|
|
|
|
|
the meantime, you should be able to keep things going (albeit more |
280
|
|
|
|
|
|
|
slowly) by turning off streaming at the agent: |
281
|
|
|
|
|
|
|
|
282
|
|
|
|
|
|
|
use REST::Neo4p; |
283
|
|
|
|
|
|
|
REST::Neo4p->agent->no_stream; |
284
|
|
|
|
|
|
|
... |
285
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
=head1 SEE ALSO |
287
|
|
|
|
|
|
|
|
288
|
|
|
|
|
|
|
L, L, L, |
289
|
|
|
|
|
|
|
L, L. |
290
|
|
|
|
|
|
|
|
291
|
|
|
|
|
|
|
=head1 AUTHOR |
292
|
|
|
|
|
|
|
|
293
|
|
|
|
|
|
|
Mark A. Jensen |
294
|
|
|
|
|
|
|
CPAN ID: MAJENSEN |
295
|
|
|
|
|
|
|
majensen -at- cpan -dot- org |
296
|
|
|
|
|
|
|
|
297
|
|
|
|
|
|
|
=head1 LICENSE |
298
|
|
|
|
|
|
|
|
299
|
|
|
|
|
|
|
Copyright (c) 2012-2022 Mark A. Jensen. This program is free software; you |
300
|
|
|
|
|
|
|
can redistribute it and/or modify it under the same terms as Perl |
301
|
|
|
|
|
|
|
itself. |
302
|
|
|
|
|
|
|
|
303
|
|
|
|
|
|
|
=cut |
304
|
|
|
|
|
|
|
|
305
|
|
|
|
|
|
|
1; |