line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
#{ |
2
|
|
|
|
|
|
|
package JRPC; |
3
|
1
|
|
|
1
|
|
272820
|
use JSON::XS; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
54
|
|
4
|
1
|
|
|
1
|
|
686
|
use Data::Dumper; |
|
1
|
|
|
|
|
5072
|
|
|
1
|
|
|
|
|
64
|
|
5
|
1
|
|
|
1
|
|
18
|
use strict; |
|
1
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
30
|
|
6
|
1
|
|
|
1
|
|
3
|
use warnings; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
727
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
#UNUSED:my $rstub = {'id' => 66666, 'jsonrpc' => '2.0'}; |
9
|
|
|
|
|
|
|
our $VERSION = '0.60'; |
10
|
|
|
|
|
|
|
# 0 = No validation (trust client, any exceptions thrown because of errors will |
11
|
|
|
|
|
|
|
# be much lower level.) |
12
|
|
|
|
|
|
|
# 1 = Validate method,params |
13
|
|
|
|
|
|
|
# 2 = Require 'id','jsonrpc', 3 |
14
|
|
|
|
|
|
|
our $msgvalid = 1; |
15
|
|
|
|
|
|
|
# This is prelogger callback. MUST be a _hard_ CODE ref to be used (not symbolic reference). |
16
|
|
|
|
|
|
|
our $prelogger; |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
=head1 NAME |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
JRPC - Create JSON-RPC Services focusing on app logic, not worrying about the details of JSON-RPC Processing. |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
=head1 SYNOPSIS |
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
use JRPC; |
25
|
|
|
|
|
|
|
# Load one of the Service modules JRPC::CGI, JRPC::Apache2 or JRPC::Nginx |
26
|
|
|
|
|
|
|
# See particular submodule documentation for the details. |
27
|
|
|
|
|
|
|
use JRPC::CGI; |
28
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
=head1 DESCRIPTION |
30
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
JRPC Module bundle consists of Server and Client pieces for creating JSON-RPC services. |
32
|
|
|
|
|
|
|
For the server piece it takes a slightly different approach than many other "API Heavy" CPAN modules. |
33
|
|
|
|
|
|
|
Instead of assembing your service out of API calls, JRPC forms a framework on top of your implementation and |
34
|
|
|
|
|
|
|
allows you to write a (single) callback: |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
=over 4 |
37
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
=item * receiving parameters (JSON-RPC "params") of the of JSON-RPC call pre-parsed, ready-to be worked with by your app code |
39
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
=item * returning the "result" data (to framework taking care of JSON-RPC) |
41
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
=back |
43
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
The callback should be wrapped into a class package. One package can host multiple service methods. |
45
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
When any exception is thrown (by die()) during the processing by callback, the framework takes care of turning this to an appropriate JSON-RPC fault. |
47
|
|
|
|
|
|
|
The framework will also take care of dealing with JSON-RPC "envelope" (term borrowed from SOAP lingo) of both request and response, "unwrapping" it |
48
|
|
|
|
|
|
|
on request and wrapping the result with it on response. |
49
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
package MyEchoService; |
51
|
|
|
|
|
|
|
our $VERSION = '0.01'; |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
# Respond with whatever was sent |
54
|
|
|
|
|
|
|
sub echo { |
55
|
|
|
|
|
|
|
my ($param) = @_; |
56
|
|
|
|
|
|
|
# Pass-through - Just send the "params" as "result" |
57
|
|
|
|
|
|
|
return($param); |
58
|
|
|
|
|
|
|
} |
59
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
=head1 DISPATCHING OF SERVICE REQUEST |
61
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
Dispatching of service request can use 2 methods: |
63
|
|
|
|
|
|
|
|
64
|
|
|
|
|
|
|
=over 4 |
65
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
=item * URL based dispatching, where relative URL (after server name and port) defines the package and method name ("method" in JSON-RPC envelope) defines the runtime method |
67
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
=item * URL independent dispatching where method name with dot-notation defines the method name |
69
|
|
|
|
|
|
|
|
70
|
|
|
|
|
|
|
=back |
71
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
Currently the dispatching method is automatically chosen based on what is found in "method" member of JSON-RPC envelope. |
73
|
|
|
|
|
|
|
Examples highlighting the (automatically) chosen dispatching method: |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
=over 4 |
76
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
=item * "method": "echo", URL "/MyEchoService" - Choose URL based dispatching, map relative URL to package and echo() method ( MyEchoService::echo() ) |
78
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
=item * "method": "MyEchoService.echo" - Derive both Class and method from dot-notation ( MyEchoService::echo() ) |
80
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
=back |
82
|
|
|
|
|
|
|
|
83
|
|
|
|
|
|
|
I'd recommend the latter as a more modern way of dispatching. Additionally (because of URL independence and need to "map" URL:s) it is less likely to require config changes in your web server. |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
=head1 METHODS |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
=head2 createfault($req, $msg, $errcode) |
88
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
Internal method to create a JSON-RPC Fault message. |
90
|
|
|
|
|
|
|
As these parameters are coming from the server side code, they are trusted |
91
|
|
|
|
|
|
|
(i.e. not validated) here. Parameters: |
92
|
|
|
|
|
|
|
|
93
|
|
|
|
|
|
|
=over 4 |
94
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
=item * $msg - Message (typically originating from exceptions). Placed to member "message" of |
96
|
|
|
|
|
|
|
"error" branch of fault (See JSON-RPC 2.0 spec for details). |
97
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
=item * $errcode - Numeric error code (must be given) |
99
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
=back |
101
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
Notice that the service methods should not be using this directly, but only be throwing exceptions. |
103
|
|
|
|
|
|
|
As a current shortcoming, the service methods cannot set $errcode (Only basic string based exceptions are |
104
|
|
|
|
|
|
|
currently allowed / accepted). |
105
|
|
|
|
|
|
|
|
106
|
|
|
|
|
|
|
This should not be called explicitly by service developer. Throw execptions in your service handler to have them |
107
|
|
|
|
|
|
|
automatically converted to valid JSON-RPC faults by createfault(). |
108
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
=cut |
110
|
|
|
|
|
|
|
# =item * $data - ANY data to be attached to 'data' member of error/fault Object |
111
|
|
|
|
|
|
|
sub createfault { |
112
|
0
|
|
|
0
|
1
|
|
my ($req, $msg, $errcode) = @_; # , $data |
113
|
|
|
|
|
|
|
# Create response stub HERE ???? |
114
|
|
|
|
|
|
|
# TODO: We could clone original or just pick 'id', 'jsonrpc' from it. |
115
|
0
|
|
|
|
|
|
my $resp = {'jsonrpc' => '2.0'}; # $req ? $req : Storable::dclone($rstub); |
116
|
0
|
|
|
|
|
|
$resp->{'id'} = $req->{'id'}; |
117
|
|
|
|
|
|
|
#$req->{'id'} = $msg->{'id'}; |
118
|
0
|
|
|
|
|
|
my $fault = $resp->{'error'} = {'message' => $msg, 'code' => $errcode, }; |
119
|
|
|
|
|
|
|
#if ($data) {$fault->{'data'} = $data;} |
120
|
|
|
|
|
|
|
# Return data (structure) or serialized JSON ? |
121
|
|
|
|
|
|
|
#if (1) {} |
122
|
0
|
|
|
|
|
|
return(encode_json($resp)); |
123
|
|
|
|
|
|
|
# Return apache return values, such as Apache2::Const::OK ? |
124
|
|
|
|
|
|
|
} |
125
|
|
|
|
|
|
|
# Note - these package global lazy-cached tables have different formats. |
126
|
|
|
|
|
|
|
# Single level dot-notation to service method (CODE) mapping. |
127
|
|
|
|
|
|
|
our %dotn2func = (); |
128
|
|
|
|
|
|
|
# Two level URL => method => service method (CODE) mapping. |
129
|
|
|
|
|
|
|
our %urlm2func = (); |
130
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
#=head1 METHOD RESOLVER METHODS |
132
|
|
|
|
|
|
|
|
133
|
|
|
|
|
|
|
#Both resolvers (Explained earlier in doc) are able to cache package+method combos in lookup tables for accelerated resolution. |
134
|
|
|
|
|
|
|
# Both have their own cache / mapping table (containing re-resolved methods) for this purpose. |
135
|
|
|
|
|
|
|
#Both resolver methods return a hard (CODE) reference to service for the server to execute. |
136
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
# DONE: Build a pre-resolved method mapping table. |
138
|
|
|
|
|
|
|
# TODO: Allow 'lazyload' for lazily loading modules on-demand. |
139
|
|
|
|
|
|
|
# Should we do package AND method resolution in single method ? |
140
|
|
|
|
|
|
|
sub methresolve_dotnot { |
141
|
0
|
|
|
0
|
0
|
|
my ($r, $m) = @_; |
142
|
|
|
|
|
|
|
# Support dot-notation (resolve_dotnot()) |
143
|
0
|
0
|
|
|
|
|
if ($m !~ /\./) {die("No dot-notation in method");} # Redundant check |
|
0
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
# Resolved earlier, Pre-cached ? |
145
|
0
|
0
|
|
|
|
|
if (my $f = $dotn2func{$m}) {return($f);} |
|
0
|
|
|
|
|
|
|
146
|
0
|
|
|
|
|
|
my @pp = split(/\./, $m); |
147
|
0
|
|
|
|
|
|
my $mcp = pop(@pp); # pop() (trailing) method |
148
|
|
|
|
|
|
|
|
149
|
0
|
0
|
|
|
|
|
if (!$mcp) {die("No method remaining for dotnot method resolution ($m)".Dumper(\@pp));} |
|
0
|
|
|
|
|
|
|
150
|
0
|
0
|
|
|
|
|
if (!@pp) {die("No package path comps for dotnot method resolution ($m)".Dumper(\@pp));} |
|
0
|
|
|
|
|
|
|
151
|
|
|
|
|
|
|
|
152
|
0
|
0
|
|
|
|
|
if (my $f = join('::', @pp)->can($mcp)) {$dotn2func{$m} = $f;return($f);} |
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
153
|
0
|
|
|
|
|
|
return(undef); |
154
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
} |
156
|
|
|
|
|
|
|
# URL2package based Service Class/Method resolver |
157
|
|
|
|
|
|
|
sub methresolve { |
158
|
0
|
|
|
0
|
0
|
|
my ($r, $m) = @_; |
159
|
|
|
|
|
|
|
# Extract Package from URL: |
160
|
|
|
|
|
|
|
# get the global request object (requires PerlOptions +GlobalRequest) |
161
|
|
|
|
|
|
|
#my $r = Apache2::RequestUtil->request; |
162
|
|
|
|
|
|
|
# Thankfully both Apache2 and Nginx have this method |
163
|
0
|
|
|
|
|
|
my $uri = $r->uri(); |
164
|
0
|
0
|
|
|
|
|
if (my $f = $urlm2func{$uri}->{$m}) {return($f);} |
|
0
|
|
|
|
|
|
|
165
|
0
|
|
|
|
|
|
my @pp = split(/\//, $uri); |
166
|
|
|
|
|
|
|
# Normalize components |
167
|
0
|
0
|
|
|
|
|
if (!$pp[0]) {shift(@pp);} |
|
0
|
|
|
|
|
|
|
168
|
0
|
0
|
|
|
|
|
if (!$pp[$#pp]) {pop(@pp);} |
|
0
|
|
|
|
|
|
|
169
|
|
|
|
|
|
|
# $ENV{'SCRIPT_NAME'} |
170
|
|
|
|
|
|
|
#my $dump = Dumper(\%ENV); # $dump |
171
|
0
|
0
|
0
|
|
|
|
if (!@pp || !$pp[0]) { |
172
|
0
|
|
|
|
|
|
die("No package comps for method resolution (uri=$uri)"); |
173
|
|
|
|
|
|
|
} |
174
|
0
|
|
|
|
|
|
my $mcp = join('::', @pp); |
175
|
0
|
|
|
|
|
|
my $f = $mcp->can($m); |
176
|
0
|
0
|
|
|
|
|
if (!$f) {die("Tried meth '$m' from package: '$mcp'");return(undef);} |
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
177
|
|
|
|
|
|
|
# Cache to a URL-to-method map (NOT methname-to-func) |
178
|
0
|
|
|
|
|
|
$urlm2func{$uri}->{$m} = $f; |
179
|
0
|
|
|
|
|
|
return($f); |
180
|
|
|
|
|
|
|
#return("qmp"->can($m)); |
181
|
|
|
|
|
|
|
} |
182
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
=head2 parse($jsontext) |
184
|
|
|
|
|
|
|
|
185
|
|
|
|
|
|
|
Parse JSON-RPC Message and validate the essential parts of it. What is validated (per JSON-RPC spec): |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
=over 4 |
188
|
|
|
|
|
|
|
|
189
|
|
|
|
|
|
|
=item * method - must be non-empty |
190
|
|
|
|
|
|
|
|
191
|
|
|
|
|
|
|
=item * params - presence (of key) - even null (value) is okay. |
192
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
=item * id - JSON-RPC ID of message - must be present (format not strictly constrained) |
194
|
|
|
|
|
|
|
|
195
|
|
|
|
|
|
|
=item * jsonrpc - JSON-RPC protocol version (must be '2.0') |
196
|
|
|
|
|
|
|
|
197
|
|
|
|
|
|
|
=back |
198
|
|
|
|
|
|
|
|
199
|
|
|
|
|
|
|
The particular format of "params" (Object/Array/scalar) or individual parameter |
200
|
|
|
|
|
|
|
validation in case of most common case "Object" is not in the scope here. |
201
|
|
|
|
|
|
|
|
202
|
|
|
|
|
|
|
=cut |
203
|
|
|
|
|
|
|
# TODO: Allow application level constraining of "params" to certain type (e.g. HASH/Object) |
204
|
|
|
|
|
|
|
sub parse { # JRPC::Msg:: |
205
|
|
|
|
|
|
|
#my ($buffer) = @_; |
206
|
0
|
|
|
0
|
1
|
|
my $j = eval { decode_json($_[0]); }; # $buffer / $_[0] |
|
0
|
|
|
|
|
|
|
207
|
0
|
0
|
|
|
|
|
if ($@) {die("Error Parsing JSON(-32700): $@");} |
|
0
|
|
|
|
|
|
|
208
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
# Error on batch requests |
210
|
0
|
0
|
|
|
|
|
if (ref($j) eq 'ARRAY') {die("JSON-RPC Batch request not (yet) supported");} |
|
0
|
|
|
|
|
|
|
211
|
|
|
|
|
|
|
# These validation steps have a slight cost (3800 => 3600 for simple |
212
|
|
|
|
|
|
|
# method processing where relative framework overhead is major). |
213
|
|
|
|
|
|
|
# Allow to skip them with a config |
214
|
|
|
|
|
|
|
#if (!$msgvalid) { |
215
|
0
|
|
|
|
|
|
return($j); |
216
|
|
|
|
|
|
|
#} |
217
|
|
|
|
|
|
|
#eval { |
218
|
|
|
|
|
|
|
# In order of importance method and params are necessary. |
219
|
0
|
0
|
|
|
|
|
if (!$j->{'method'}) {die("No 'method' found");} |
|
0
|
|
|
|
|
|
|
220
|
0
|
0
|
|
|
|
|
if (!exists($j->{'params'})) {die("No 'params' found");} # !(not) enough ? |
|
0
|
|
|
|
|
|
|
221
|
0
|
0
|
|
|
|
|
if ($msgvalid < 2) {return($j);} |
|
0
|
|
|
|
|
|
|
222
|
0
|
0
|
|
|
|
|
if (!$j->{'id'}) {die("No 'id' found");} |
|
0
|
|
|
|
|
|
|
223
|
0
|
0
|
|
|
|
|
if ($j->{'jsonrpc'} ne '2.0') {die("No jsonrpc version (2.0)found");} |
|
0
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
# Still validate envelope and param top level format |
225
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
# Additional params format constraint validation (fmtvalidator func ?) |
227
|
|
|
|
|
|
|
#if (my $fmt = $serv->{'pfmt'}) {} |
228
|
|
|
|
|
|
|
#}; |
229
|
0
|
0
|
|
|
|
|
if ($@) {die("Invalid JSON-RPC Message (-32600): $@");} |
|
0
|
|
|
|
|
|
|
230
|
0
|
|
|
|
|
|
return($j); |
231
|
|
|
|
|
|
|
} |
232
|
|
|
|
|
|
|
|
233
|
|
|
|
|
|
|
#}; # END package JRPC; |
234
|
|
|
|
|
|
|
|
235
|
|
|
|
|
|
|
=head2 respond_async($client, $url, $meth, $p, %opts); |
236
|
|
|
|
|
|
|
|
237
|
|
|
|
|
|
|
After async processing, acknowledge the original client tier (or any URL) of the completion of the asynchronous part. |
238
|
|
|
|
|
|
|
This method is experimental and the whole concept of using asynchronous processing at service is an unofficial extension |
239
|
|
|
|
|
|
|
to standard JSON-RPC 2.0 protocol spec. |
240
|
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
Parameters: |
242
|
|
|
|
|
|
|
|
243
|
|
|
|
|
|
|
=over 4 |
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
=item * $client - Instance of JRPC::Client. If not provided, a new client will be instantiated here. |
246
|
|
|
|
|
|
|
|
247
|
|
|
|
|
|
|
=item * $url - URL of the async callback - Must be provided |
248
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
=item * $meth - JSON-RPC Method to callback to on the server (default: "oncomplete") |
250
|
|
|
|
|
|
|
|
251
|
|
|
|
|
|
|
=item * $p - JSON-RPC "params" to send in completion acknowledgement (must be supplied, likely to be Object/Hash) |
252
|
|
|
|
|
|
|
|
253
|
|
|
|
|
|
|
=back |
254
|
|
|
|
|
|
|
|
255
|
|
|
|
|
|
|
If optional KW params in have param 'cb' set, it is used to process the response from callback service. The "result" of JSON-RPC response is passed to this callback. |
256
|
|
|
|
|
|
|
|
257
|
|
|
|
|
|
|
Return "result" of response (likely to be Object/Hash). |
258
|
|
|
|
|
|
|
|
259
|
|
|
|
|
|
|
=cut |
260
|
|
|
|
|
|
|
|
261
|
|
|
|
|
|
|
#Minor optimization for avoiding overhead of JRPC::Client instantiation in respond_async() (or during request) is to initialize it in the service package init() phase. |
262
|
|
|
|
|
|
|
|
263
|
|
|
|
|
|
|
# TODO: Example of combination of init and a call to respond_async() |
264
|
|
|
|
|
|
|
# package MyPack; |
265
|
|
|
|
|
|
|
# ... |
266
|
|
|
|
|
|
|
#our $client; |
267
|
|
|
|
|
|
|
#sub init { |
268
|
|
|
|
|
|
|
# $client = JRPC::Client->new(); |
269
|
|
|
|
|
|
|
#} |
270
|
|
|
|
|
|
|
#sub do_long_and_hard_work { |
271
|
|
|
|
|
|
|
# my ($p) = @_; |
272
|
|
|
|
|
|
|
# |
273
|
|
|
|
|
|
|
#} |
274
|
|
|
|
|
|
|
#TODO: Consider callback to handle specific response. |
275
|
|
|
|
|
|
|
sub respond_async { |
276
|
0
|
|
|
0
|
1
|
|
my ($client, $url, $meth, $p, %c) = @_; |
277
|
|
|
|
|
|
|
#my $client = $opts{'client'}; # Allow passing client as optional ? |
278
|
|
|
|
|
|
|
# Create a full JRPC::Client instance if not passed. |
279
|
0
|
0
|
|
|
|
|
if (!$client) {$client = JRPC::Client->new();} |
|
0
|
|
|
|
|
|
|
280
|
0
|
0
|
|
|
|
|
if (!$url) {die("No Callback URL passed");} |
|
0
|
|
|
|
|
|
|
281
|
0
|
0
|
|
|
|
|
if (!$meth) {$meth = 'oncomplete';} |
|
0
|
|
|
|
|
|
|
282
|
0
|
0
|
|
|
|
|
if (!$p) {die("No Parameters passed");} |
|
0
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
# Always create a new request |
284
|
0
|
|
|
|
|
|
my $req = $client->new_request($url); # Client does not know URL, request does. |
285
|
0
|
0
|
|
|
|
|
if (!$req) {die("JSON-RPC Request not instantiated");} |
|
0
|
|
|
|
|
|
|
286
|
0
|
|
|
|
|
|
my $resp = $req->call($meth, $p, 'notify' => 1); # Need eval ? |
287
|
0
|
0
|
|
|
|
|
if (!$resp->is_success()) {die("HTTP Error: ".$resp->status_line());} # Status code ? |
|
0
|
|
|
|
|
|
|
288
|
|
|
|
|
|
|
#DEBUG:print($fh "Resp from '$url': ".$resp->content()."\n"); |
289
|
|
|
|
|
|
|
# Server side may or may not care about this. |
290
|
|
|
|
|
|
|
# By default consider response as non-important as handling various specific responses here would |
291
|
|
|
|
|
|
|
# be hard. |
292
|
0
|
|
|
|
|
|
my $result = $resp->result(); |
293
|
|
|
|
|
|
|
# Consider:Expect still a valid JSON response ? Parse it ? |
294
|
1
|
0
|
|
1
|
|
5
|
if (my $f = $c{'cb'}) {no strict 'refs';$f->($result);} |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
507
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
295
|
0
|
|
|
|
|
|
return($result); |
296
|
|
|
|
|
|
|
} |
297
|
|
|
|
|
|
|
|
298
|
|
|
|
|
|
|
#setup_pkg_as_server($classname) |
299
|
|
|
|
|
|
|
# Setup a Service package as independent, runnable server w.o. hard-wiring any |
300
|
|
|
|
|
|
|
# code into a server package. Handy for testing a serice package. |
301
|
|
|
|
|
|
|
# Loads HTTP::Server::Simple::CGI, JRPC::CGI, Attaches the "handle_request" callback method as request handler. |
302
|
|
|
|
|
|
|
# After this setup all that remains to be done is to run the server (not done here). |
303
|
|
|
|
|
|
|
# Complete example of making "MyServPkg" run. |
304
|
|
|
|
|
|
|
# use MyServPkg; |
305
|
|
|
|
|
|
|
# my $port = $ENV{'JSONRPC_SERVICE_PORT'} || 8080; |
306
|
|
|
|
|
|
|
# # Run in the same process |
307
|
|
|
|
|
|
|
# MyServPkg->new($port)->run(); |
308
|
|
|
|
|
|
|
sub setup_pkg_as_server { |
309
|
0
|
|
|
0
|
0
|
|
my ($class) = @_; |
310
|
|
|
|
|
|
|
# Bootstrapping boilerplate. We are (almost completely) n control of of $boot string here, |
311
|
|
|
|
|
|
|
# so eval is acceptable. Especially with validation of $class. |
312
|
0
|
0
|
|
|
|
|
if ($class !~ /^[\w:]+$/) {die("Class does not look right");} # No spaces |
|
0
|
|
|
|
|
|
|
313
|
0
|
|
|
|
|
|
my $boot = |
314
|
|
|
|
|
|
|
"use HTTP::Server::Simple::CGI;\npush(\@$class\:\:ISA, 'HTTP::Server::Simple::CGI');\nuse JRPC::CGI;\n"; |
315
|
0
|
|
|
|
|
|
$boot .= "*$class\:\:handle_request = \\&JRPC::CGI::handle_simple_server_cgi;"; |
316
|
|
|
|
|
|
|
#print(STDERR "$boot"); |
317
|
0
|
|
|
|
|
|
eval("$boot"); |
318
|
|
|
|
|
|
|
} |
319
|
|
|
|
|
|
|
1; |
320
|
|
|
|
|
|
|
__END__ |