| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Statistics::R::IO::Rserve; |
|
2
|
|
|
|
|
|
|
# ABSTRACT: Supply object methods for Rserve communication |
|
3
|
|
|
|
|
|
|
$Statistics::R::IO::Rserve::VERSION = '1.0'; |
|
4
|
9
|
|
|
9
|
|
139991
|
use 5.010; |
|
|
9
|
|
|
|
|
21
|
|
|
5
|
|
|
|
|
|
|
|
|
6
|
9
|
|
|
9
|
|
2439
|
use Class::Tiny::Antlers; |
|
|
9
|
|
|
|
|
20667
|
|
|
|
9
|
|
|
|
|
44
|
|
|
7
|
|
|
|
|
|
|
|
|
8
|
9
|
|
|
9
|
|
2905
|
use Statistics::R::IO::REXPFactory; |
|
|
9
|
|
|
|
|
15
|
|
|
|
9
|
|
|
|
|
339
|
|
|
9
|
9
|
|
|
9
|
|
3907
|
use Statistics::R::IO::QapEncoding; |
|
|
9
|
|
|
|
|
19
|
|
|
|
9
|
|
|
|
|
355
|
|
|
10
|
|
|
|
|
|
|
|
|
11
|
9
|
|
|
9
|
|
2570
|
use Socket; |
|
|
9
|
|
|
|
|
12974
|
|
|
|
9
|
|
|
|
|
3539
|
|
|
12
|
9
|
|
|
9
|
|
2226
|
use IO::Socket::INET (); |
|
|
9
|
|
|
|
|
63498
|
|
|
|
9
|
|
|
|
|
188
|
|
|
13
|
9
|
|
|
9
|
|
41
|
use Scalar::Util qw(blessed looks_like_number openhandle); |
|
|
9
|
|
|
|
|
13
|
|
|
|
9
|
|
|
|
|
406
|
|
|
14
|
9
|
|
|
9
|
|
32
|
use Carp; |
|
|
9
|
|
|
|
|
8
|
|
|
|
9
|
|
|
|
|
324
|
|
|
15
|
|
|
|
|
|
|
|
|
16
|
9
|
|
|
9
|
|
32
|
use namespace::clean; |
|
|
9
|
|
|
|
|
11
|
|
|
|
9
|
|
|
|
|
47
|
|
|
17
|
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
has fh => ( |
|
20
|
|
|
|
|
|
|
is => 'ro', |
|
21
|
|
|
|
|
|
|
default => sub { |
|
22
|
|
|
|
|
|
|
my $self = shift; |
|
23
|
|
|
|
|
|
|
my $fh; |
|
24
|
|
|
|
|
|
|
if ($self->_usesocket) { |
|
25
|
|
|
|
|
|
|
socket($fh, PF_INET, SOCK_STREAM, getprotobyname('tcp')) || |
|
26
|
|
|
|
|
|
|
croak "socket: $!"; |
|
27
|
|
|
|
|
|
|
connect($fh, sockaddr_in($self->port, inet_aton($self->server))) || |
|
28
|
|
|
|
|
|
|
croak "connect: $!"; |
|
29
|
|
|
|
|
|
|
bless $fh, 'IO::Handle' |
|
30
|
|
|
|
|
|
|
} |
|
31
|
|
|
|
|
|
|
else { |
|
32
|
|
|
|
|
|
|
$fh = IO::Socket::INET->new(PeerAddr => $self->server, |
|
33
|
|
|
|
|
|
|
PeerPort => $self->port) or |
|
34
|
|
|
|
|
|
|
croak $! |
|
35
|
|
|
|
|
|
|
} |
|
36
|
|
|
|
|
|
|
$self->_set_autoclose(1) unless defined($self->_autoclose); |
|
37
|
|
|
|
|
|
|
my ($response, $rc) = ''; |
|
38
|
|
|
|
|
|
|
while ($rc = $fh->read($response, 32 - length $response, |
|
39
|
|
|
|
|
|
|
length $response)) {} |
|
40
|
|
|
|
|
|
|
croak $! unless defined $rc; |
|
41
|
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
croak "Unrecognized server ID" unless |
|
43
|
|
|
|
|
|
|
substr($response, 0, 12) eq 'Rsrv0103QAP1'; |
|
44
|
|
|
|
|
|
|
$fh |
|
45
|
|
|
|
|
|
|
}, |
|
46
|
|
|
|
|
|
|
); |
|
47
|
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
has server => ( |
|
49
|
|
|
|
|
|
|
is => 'ro', |
|
50
|
|
|
|
|
|
|
default => 'localhost', |
|
51
|
|
|
|
|
|
|
); |
|
52
|
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
has port => ( |
|
54
|
|
|
|
|
|
|
is => 'ro', |
|
55
|
|
|
|
|
|
|
default => 6311, |
|
56
|
|
|
|
|
|
|
); |
|
57
|
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
has _autoclose => ( |
|
60
|
|
|
|
|
|
|
is => 'ro', |
|
61
|
|
|
|
|
|
|
); |
|
62
|
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
|
|
64
|
|
|
|
|
|
|
has _autoflush => ( |
|
65
|
|
|
|
|
|
|
is => 'ro', |
|
66
|
|
|
|
|
|
|
default => sub { |
|
67
|
|
|
|
|
|
|
my $self = shift; |
|
68
|
|
|
|
|
|
|
$self->_usesocket ? 1 : 0 |
|
69
|
|
|
|
|
|
|
}, |
|
70
|
|
|
|
|
|
|
); |
|
71
|
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
has _usesocket => ( |
|
73
|
|
|
|
|
|
|
is => 'ro', |
|
74
|
|
|
|
|
|
|
default => 0 |
|
75
|
|
|
|
|
|
|
); |
|
76
|
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
use constant { |
|
79
|
9
|
|
|
|
|
7893
|
CMD_login => 0x001, # "name\npwd" : - |
|
80
|
|
|
|
|
|
|
CMD_voidEval => 0x002, # string : - |
|
81
|
|
|
|
|
|
|
CMD_eval => 0x003, # string | encoded SEXP : encoded SEXP |
|
82
|
|
|
|
|
|
|
CMD_shutdown => 0x004, # [admin-pwd] : - |
|
83
|
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
# security/encryption - all since 1.7-0 |
|
85
|
|
|
|
|
|
|
CMD_switch => 0x005, # string (protocol) : - |
|
86
|
|
|
|
|
|
|
CMD_keyReq => 0x006, # string (request) : bytestream (key) |
|
87
|
|
|
|
|
|
|
CMD_secLogin => 0x007, # bytestream (encrypted auth) : - |
|
88
|
|
|
|
|
|
|
CMD_OCcall => 0x00f, # SEXP : SEXP -- it is the only command |
|
89
|
|
|
|
|
|
|
# supported in object-capability mode and it |
|
90
|
|
|
|
|
|
|
# requires that the SEXP is a language |
|
91
|
|
|
|
|
|
|
# construct with OC reference in the first |
|
92
|
|
|
|
|
|
|
# position |
|
93
|
|
|
|
|
|
|
CMD_OCinit => 0x434f7352, # SEXP -- 'RsOC' - command sent from the |
|
94
|
|
|
|
|
|
|
# server in OC mode with the packet of |
|
95
|
|
|
|
|
|
|
# initial capabilities. file I/O |
|
96
|
|
|
|
|
|
|
# routines. server may answe |
|
97
|
|
|
|
|
|
|
CMD_openFile => 0x010, # fn : - |
|
98
|
|
|
|
|
|
|
CMD_createFile => 0x011, # fn : - |
|
99
|
|
|
|
|
|
|
CMD_closeFile => 0x012, # - : - |
|
100
|
|
|
|
|
|
|
CMD_readFile => 0x013, # [int size] : data... ; if size not |
|
101
|
|
|
|
|
|
|
# present, server is free to choose any |
|
102
|
|
|
|
|
|
|
# value - usually it uses the size of its |
|
103
|
|
|
|
|
|
|
# static buffer |
|
104
|
|
|
|
|
|
|
CMD_writeFile => 0x014, # data : - |
|
105
|
|
|
|
|
|
|
CMD_removeFile => 0x015, # fn : - |
|
106
|
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
# object manipulation |
|
108
|
|
|
|
|
|
|
CMD_setSEXP => 0x020, # string(name), REXP : - |
|
109
|
|
|
|
|
|
|
CMD_assignSEXP => 0x021, # string(name), REXP : - ; same as |
|
110
|
|
|
|
|
|
|
# setSEXP except that the name is parsed |
|
111
|
|
|
|
|
|
|
|
|
112
|
|
|
|
|
|
|
# session management (since 0.4-0) |
|
113
|
|
|
|
|
|
|
CMD_detachSession => 0x030, # : session key |
|
114
|
|
|
|
|
|
|
CMD_detachedVoidEval => 0x031, # string : session key; doesn't |
|
115
|
|
|
|
|
|
|
CMD_attachSession => 0x032, # session key : - |
|
116
|
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
# control commands (since 0.6-0) - passed on to the master process */ |
|
118
|
|
|
|
|
|
|
# Note: currently all control commands are asychronous, i.e. RESP_OK |
|
119
|
|
|
|
|
|
|
# indicates that the command was enqueued in the master pipe, but there |
|
120
|
|
|
|
|
|
|
# is no guarantee that it will be processed. Moreover non-forked |
|
121
|
|
|
|
|
|
|
# connections (e.g. the default debug setup) don't process any |
|
122
|
|
|
|
|
|
|
# control commands until the current client connection is closed so |
|
123
|
|
|
|
|
|
|
# the connection issuing the control command will never see its |
|
124
|
|
|
|
|
|
|
# result. |
|
125
|
|
|
|
|
|
|
CMD_ctrl => 0x40, # -- not a command - just a constant -- |
|
126
|
|
|
|
|
|
|
CMD_ctrlEval => 0x42, # string : - |
|
127
|
|
|
|
|
|
|
CMD_ctrlSource => 0x45, # string : - |
|
128
|
|
|
|
|
|
|
CMD_ctrlShutdown => 0x44, # - : - |
|
129
|
|
|
|
|
|
|
|
|
130
|
|
|
|
|
|
|
# 'internal' commands (since 0.1-9) |
|
131
|
|
|
|
|
|
|
CMD_setBufferSize => 0x081, # [int sendBufSize] this commad allow |
|
132
|
|
|
|
|
|
|
# clients to request bigger buffer |
|
133
|
|
|
|
|
|
|
# sizes if large data is to be |
|
134
|
|
|
|
|
|
|
# transported from Rserve to the |
|
135
|
|
|
|
|
|
|
# client. (incoming buffer is resized |
|
136
|
|
|
|
|
|
|
# automatically) |
|
137
|
|
|
|
|
|
|
CMD_setEncoding => 0x082, # string (one of "native","latin1","utf8") : -; since 0.5-3 |
|
138
|
|
|
|
|
|
|
|
|
139
|
|
|
|
|
|
|
# special commands - the payload of packages with this mask does not contain defined parameters |
|
140
|
|
|
|
|
|
|
CMD_SPECIAL_MASK => 0xf0, |
|
141
|
|
|
|
|
|
|
CMD_serEval => 0xf5, # serialized eval - the packets are raw |
|
142
|
|
|
|
|
|
|
# serialized data without data header |
|
143
|
|
|
|
|
|
|
CMD_serAssign => 0xf6, # serialized assign - serialized list with |
|
144
|
|
|
|
|
|
|
# [[1]]=name, [[2]]=value |
|
145
|
|
|
|
|
|
|
CMD_serEEval => 0xf7, # serialized expression eval - like serEval |
|
146
|
|
|
|
|
|
|
# with one additional evaluation round |
|
147
|
9
|
|
|
9
|
|
13326
|
}; |
|
|
9
|
|
|
|
|
14
|
|
|
148
|
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
|
|
150
|
|
|
|
|
|
|
sub BUILDARGS { |
|
151
|
155
|
|
|
155
|
0
|
135513
|
my $class = shift; |
|
152
|
|
|
|
|
|
|
|
|
153
|
155
|
50
|
|
|
|
689
|
if ( scalar @_ == 0 ) { |
|
|
|
100
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
return { } |
|
155
|
0
|
|
|
|
|
0
|
} elsif ( scalar @_ == 1 ) { |
|
156
|
51
|
50
|
|
|
|
170
|
if ( ref $_[0] eq 'HASH' ) { |
|
|
|
100
|
|
|
|
|
|
|
157
|
0
|
|
|
|
|
0
|
my $args = { %{ $_[0] } }; |
|
|
0
|
|
|
|
|
0
|
|
|
158
|
0
|
0
|
|
|
|
0
|
if (my $fh = $args->{fh}) { |
|
159
|
0
|
|
|
|
|
0
|
($args->{server}, $args->{port}) = _fh_host_port($fh); |
|
160
|
|
|
|
|
|
|
} |
|
161
|
0
|
|
|
|
|
0
|
return $args |
|
162
|
|
|
|
|
|
|
} elsif (ref $_[0] eq '') { |
|
163
|
1
|
|
|
|
|
2
|
my $server = shift; |
|
164
|
1
|
|
|
|
|
3
|
return { server => $server } |
|
165
|
|
|
|
|
|
|
} else { |
|
166
|
50
|
|
|
|
|
51
|
my $fh = shift; |
|
167
|
50
|
|
|
|
|
93
|
my ($server, $port) = _fh_host_port($fh); |
|
168
|
50
|
|
|
|
|
4218
|
return { fh => $fh, |
|
169
|
|
|
|
|
|
|
server => $server, |
|
170
|
|
|
|
|
|
|
port => $port, |
|
171
|
|
|
|
|
|
|
_autoclose => 0, |
|
172
|
|
|
|
|
|
|
_autoflush => ref($fh) eq 'GLOB' } |
|
173
|
|
|
|
|
|
|
} |
|
174
|
|
|
|
|
|
|
} |
|
175
|
|
|
|
|
|
|
elsif ( @_ % 2 ) { |
|
176
|
0
|
|
|
|
|
0
|
die "The new() method for $class expects a hash reference or a key/value list." |
|
177
|
|
|
|
|
|
|
. " You passed an odd number of arguments\n"; |
|
178
|
|
|
|
|
|
|
} |
|
179
|
|
|
|
|
|
|
else { |
|
180
|
104
|
|
|
|
|
268
|
my $args = { @_ }; |
|
181
|
104
|
100
|
|
|
|
758
|
if (my $fh = $args->{fh}) { |
|
182
|
102
|
|
|
|
|
275
|
($args->{server}, $args->{port}) = _fh_host_port($fh); |
|
183
|
|
|
|
|
|
|
} |
|
184
|
104
|
|
|
|
|
8386
|
return $args |
|
185
|
|
|
|
|
|
|
} |
|
186
|
|
|
|
|
|
|
} |
|
187
|
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
|
|
189
|
|
|
|
|
|
|
sub BUILD { |
|
190
|
155
|
|
|
155
|
0
|
2740
|
my ($self, $args) = @_; |
|
191
|
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
# Required attribute types |
|
193
|
|
|
|
|
|
|
die "Attribute 'fh' must be an instance of IO::Handle or an open filehandle" if |
|
194
|
|
|
|
|
|
|
defined($args->{fh}) && |
|
195
|
|
|
|
|
|
|
!((ref($args->{fh}) eq "GLOB" && Scalar::Util::openhandle($args->{fh})) || |
|
196
|
155
|
50
|
33
|
|
|
1507
|
(blessed($args->{fh}) && $args->{fh}->isa("IO::Handle"))); |
|
|
|
|
66
|
|
|
|
|
|
197
|
|
|
|
|
|
|
die "Attribute 'server' must be scalar value" if |
|
198
|
155
|
100
|
66
|
|
|
2372
|
exists($args->{server}) && (!defined($args->{server}) || ref($args->{server})); |
|
|
|
|
33
|
|
|
|
|
|
199
|
152
|
50
|
33
|
|
|
3622
|
die "Attribute 'port' must be an integer" unless |
|
200
|
|
|
|
|
|
|
looks_like_number($self->port) && (int($self->port) == $self->port); |
|
201
|
|
|
|
|
|
|
} |
|
202
|
|
|
|
|
|
|
|
|
203
|
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
## Extracts host address and port from the given socket handle (either |
|
205
|
|
|
|
|
|
|
## as an object or a "classic" socket) |
|
206
|
|
|
|
|
|
|
sub _fh_host_port { |
|
207
|
152
|
50
|
|
152
|
|
362
|
my $fh = shift or return; |
|
208
|
152
|
50
|
33
|
|
|
971
|
if (ref($fh) eq 'GLOB') { |
|
|
|
50
|
|
|
|
|
|
|
209
|
0
|
0
|
|
|
|
0
|
my ($port, $host) = unpack_sockaddr_in(getpeername($fh)) or return; |
|
210
|
0
|
|
|
|
|
0
|
my $name = gethostbyaddr($host, AF_INET); |
|
211
|
0
|
|
0
|
|
|
0
|
return ($name // inet_ntoa($host), $port) |
|
212
|
|
|
|
|
|
|
} elsif (blessed($fh) && $fh->isa('IO::Socket')){ |
|
213
|
152
|
|
|
|
|
2094
|
return ($fh->peerhost, $fh->peerport) |
|
214
|
|
|
|
|
|
|
} |
|
215
|
|
|
|
|
|
|
return undef |
|
216
|
0
|
|
|
|
|
0
|
} |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
## Private setter for autoclose used in the default handler of 'fh' |
|
220
|
|
|
|
|
|
|
sub _set_autoclose { |
|
221
|
0
|
|
|
0
|
|
0
|
my $self = shift; |
|
222
|
|
|
|
|
|
|
$self->{_autoclose} = shift |
|
223
|
0
|
|
|
|
|
0
|
} |
|
224
|
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
sub eval { |
|
227
|
101
|
|
|
101
|
1
|
1877
|
my ($self, $expr) = (shift, shift); |
|
228
|
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
# Encode $expr as DT_STRING |
|
230
|
101
|
|
|
|
|
413
|
my $parameter = pack('VZ*', |
|
231
|
|
|
|
|
|
|
((length($expr)+1) << 8) + 4, |
|
232
|
|
|
|
|
|
|
$expr); |
|
233
|
|
|
|
|
|
|
|
|
234
|
101
|
|
|
|
|
265
|
my $data = $self->_send_command(CMD_eval, $parameter); |
|
235
|
|
|
|
|
|
|
|
|
236
|
99
|
|
|
|
|
120
|
my ($value, $state) = @{Statistics::R::IO::QapEncoding::decode($data)}; |
|
|
99
|
|
|
|
|
308
|
|
|
237
|
99
|
50
|
|
|
|
222
|
croak 'Could not parse Rserve value' unless $state; |
|
238
|
99
|
50
|
|
|
|
214
|
croak 'Unread data remaining in the Rserve response' unless $state->eof; |
|
239
|
99
|
|
|
|
|
619
|
$value |
|
240
|
|
|
|
|
|
|
} |
|
241
|
|
|
|
|
|
|
|
|
242
|
|
|
|
|
|
|
|
|
243
|
|
|
|
|
|
|
sub ser_eval { |
|
244
|
51
|
|
|
51
|
1
|
1646
|
my ($self, $rexp) = (shift, shift); |
|
245
|
|
|
|
|
|
|
|
|
246
|
|
|
|
|
|
|
## simulate the request parameter as constructed by: |
|
247
|
|
|
|
|
|
|
## > serialize(quote(parse(text="{$rexp}")[[1]]), NULL) |
|
248
|
51
|
|
|
|
|
193
|
my $parameter = |
|
249
|
|
|
|
|
|
|
"\x58\x0a\0\0\0\2\0\3\0\3\0\2\3\0\0\0\0\6\0\0\0\1\0\4\0" . |
|
250
|
|
|
|
|
|
|
"\x09\0\0\0\2\x5b\x5b\0\0\0\2\0\0\0\6\0\0\0\1\0\4\0\x09\0\0" . |
|
251
|
|
|
|
|
|
|
"\0\5\x70\x61\x72\x73\x65\0\0\4\2\0\0\0\1\0\4\0\x09\0\0\0\4\x74\x65" . |
|
252
|
|
|
|
|
|
|
"\x78\x74\0\0\0\x10\0\0\0\1\0\4\0\x09" . |
|
253
|
|
|
|
|
|
|
pack('N', length($rexp)+2) . |
|
254
|
|
|
|
|
|
|
"\x7b" . $rexp . "\x7d" . |
|
255
|
|
|
|
|
|
|
"\0\0\0\xfe\0\0\0\2\0\0\0\x0e\0\0\0\1\x3f\xf0\0\0\0\0\0\0" . |
|
256
|
|
|
|
|
|
|
"\0\0\0\xfe"; |
|
257
|
|
|
|
|
|
|
## request is: |
|
258
|
|
|
|
|
|
|
## - command (0xf5, CMD_serEval, |
|
259
|
|
|
|
|
|
|
## means raw serialized data without data header) |
|
260
|
51
|
|
|
|
|
109
|
my $data = $self->_send_command(CMD_serEval, $parameter); |
|
261
|
|
|
|
|
|
|
|
|
262
|
50
|
|
|
|
|
54
|
my ($value, $state) = @{Statistics::R::IO::REXPFactory::unserialize($data)}; |
|
|
50
|
|
|
|
|
156
|
|
|
263
|
50
|
50
|
|
|
|
134
|
croak 'Could not parse Rserve value' unless $state; |
|
264
|
50
|
50
|
|
|
|
118
|
croak 'Unread data remaining in the Rserve response' unless $state->eof; |
|
265
|
50
|
|
|
|
|
252
|
$value |
|
266
|
|
|
|
|
|
|
} |
|
267
|
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
|
|
269
|
|
|
|
|
|
|
sub get_file { |
|
270
|
0
|
|
|
0
|
1
|
0
|
my ($self, $remote, $local) = (shift, shift, shift); |
|
271
|
|
|
|
|
|
|
|
|
272
|
0
|
|
|
|
|
0
|
my $data = pack 'C*', @{$self->eval("readBin('$remote', what='raw', n=file.info('$remote')[['size']])")->to_pl}; |
|
|
0
|
|
|
|
|
0
|
|
|
273
|
|
|
|
|
|
|
|
|
274
|
0
|
0
|
|
|
|
0
|
if ($local) { |
|
275
|
0
|
0
|
|
|
|
0
|
open my $local_file, '>:raw', $local or |
|
276
|
|
|
|
|
|
|
croak "Cannot open $!"; |
|
277
|
|
|
|
|
|
|
|
|
278
|
0
|
|
|
|
|
0
|
print $local_file $data; |
|
279
|
|
|
|
|
|
|
|
|
280
|
0
|
|
|
|
|
0
|
close $local_file; |
|
281
|
|
|
|
|
|
|
} |
|
282
|
|
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
$data |
|
284
|
0
|
|
|
|
|
0
|
} |
|
285
|
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
|
|
287
|
|
|
|
|
|
|
use constant { |
|
288
|
9
|
|
|
|
|
670
|
CMD_RESP => 0x10000, # all responses have this flag set |
|
289
|
|
|
|
|
|
|
CMD_OOB => 0x20000, # out-of-band data - i.e. unsolicited messages |
|
290
|
9
|
|
|
9
|
|
39
|
}; |
|
|
9
|
|
|
|
|
12
|
|
|
291
|
|
|
|
|
|
|
|
|
292
|
|
|
|
|
|
|
use constant { |
|
293
|
9
|
|
|
|
|
2594
|
RESP_OK => (CMD_RESP|0x0001), # command succeeded; returned |
|
294
|
|
|
|
|
|
|
# parameters depend on the command |
|
295
|
|
|
|
|
|
|
# issued |
|
296
|
|
|
|
|
|
|
RESP_ERR => (CMD_RESP|0x0002), # command failed, check stats code |
|
297
|
|
|
|
|
|
|
# attached string may describe the |
|
298
|
|
|
|
|
|
|
# error |
|
299
|
|
|
|
|
|
|
OOB_SEND => (CMD_OOB | 0x1000), # OOB send - unsolicited SEXP sent |
|
300
|
|
|
|
|
|
|
# from the R instance to the |
|
301
|
|
|
|
|
|
|
# client. 12 LSB are reserved for |
|
302
|
|
|
|
|
|
|
# application-specific code |
|
303
|
|
|
|
|
|
|
OOB_MSG => (CMD_OOB | 0x2000), # OOB message - unsolicited message |
|
304
|
|
|
|
|
|
|
# sent from the R instance to the |
|
305
|
|
|
|
|
|
|
# client requiring a response. 12 |
|
306
|
|
|
|
|
|
|
# LSB are reserved for |
|
307
|
|
|
|
|
|
|
# application-specific code |
|
308
|
9
|
|
|
9
|
|
35
|
}; |
|
|
9
|
|
|
|
|
10
|
|
|
309
|
|
|
|
|
|
|
|
|
310
|
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
## Sends a request to Rserve and receives the response, checking for |
|
312
|
|
|
|
|
|
|
## any errors. |
|
313
|
|
|
|
|
|
|
## |
|
314
|
|
|
|
|
|
|
## Returns the data portion of the server response |
|
315
|
|
|
|
|
|
|
sub _send_command { |
|
316
|
152
|
|
50
|
152
|
|
659
|
my ($self, $command, $parameters) = (shift, shift, shift || ''); |
|
317
|
|
|
|
|
|
|
|
|
318
|
|
|
|
|
|
|
## request is (byte order is low-endian): |
|
319
|
|
|
|
|
|
|
## - command (4 bytes) |
|
320
|
|
|
|
|
|
|
## - length of the message (low 32 bits) |
|
321
|
|
|
|
|
|
|
## - offset of the data part (normally 0) |
|
322
|
|
|
|
|
|
|
## - high 32 bits of the length of the message (0 if < 4GB) |
|
323
|
152
|
|
|
|
|
2545
|
$self->fh->print(pack('V4', $command, length($parameters), 0, 0) . |
|
324
|
|
|
|
|
|
|
$parameters); |
|
325
|
152
|
50
|
|
|
|
10040
|
$self->fh->flush if $self->_autoflush; |
|
326
|
|
|
|
|
|
|
|
|
327
|
152
|
|
|
|
|
2550
|
my $response = $self->_receive_response(16); |
|
328
|
|
|
|
|
|
|
## Of the next four long-ints: |
|
329
|
|
|
|
|
|
|
## - the first one is status and should be 65537 (bytes \1, \0, \1, \0) |
|
330
|
|
|
|
|
|
|
## - the second one is length |
|
331
|
|
|
|
|
|
|
## - the third and fourth are ?? |
|
332
|
152
|
|
|
|
|
790
|
my ($status, $length) = unpack VV => substr($response, 0, 8); |
|
333
|
152
|
50
|
|
|
|
357
|
if ($status & CMD_RESP) { |
|
|
|
0
|
|
|
|
|
|
|
334
|
152
|
100
|
|
|
|
324
|
unless ($status == RESP_OK) { |
|
335
|
3
|
|
|
|
|
523
|
croak 'R server returned an error: ' . sprintf("0x%X", $status) |
|
336
|
|
|
|
|
|
|
} |
|
337
|
|
|
|
|
|
|
} |
|
338
|
|
|
|
|
|
|
elsif ($status & CMD_OOB) { |
|
339
|
0
|
|
|
|
|
0
|
croak 'OOB messages are not supported yet' |
|
340
|
|
|
|
|
|
|
} |
|
341
|
|
|
|
|
|
|
else { |
|
342
|
0
|
|
|
|
|
0
|
croak 'Unrecognized response type: ' . $status |
|
343
|
|
|
|
|
|
|
} |
|
344
|
|
|
|
|
|
|
|
|
345
|
149
|
|
|
|
|
299
|
$self->_receive_response($length) |
|
346
|
|
|
|
|
|
|
} |
|
347
|
|
|
|
|
|
|
|
|
348
|
|
|
|
|
|
|
|
|
349
|
|
|
|
|
|
|
sub _receive_response { |
|
350
|
301
|
|
|
301
|
|
337
|
my ($self, $length) = (shift, shift); |
|
351
|
|
|
|
|
|
|
|
|
352
|
301
|
|
|
|
|
394
|
my ($response, $offset, $rc) = ('', 0); |
|
353
|
301
|
|
|
|
|
4676
|
while ($rc = $self->fh->read($response, $length - $offset, $offset)) { |
|
354
|
298
|
|
|
|
|
13814
|
$offset += $rc; |
|
355
|
298
|
50
|
|
|
|
621
|
last if $length == $offset; |
|
356
|
|
|
|
|
|
|
} |
|
357
|
301
|
50
|
|
|
|
620
|
croak $! unless defined $rc; |
|
358
|
301
|
|
|
|
|
494
|
$response |
|
359
|
|
|
|
|
|
|
} |
|
360
|
|
|
|
|
|
|
|
|
361
|
|
|
|
|
|
|
|
|
362
|
|
|
|
|
|
|
sub close { |
|
363
|
0
|
|
|
0
|
1
|
0
|
my $self = shift; |
|
364
|
0
|
|
|
|
|
0
|
$self->fh->close |
|
365
|
|
|
|
|
|
|
} |
|
366
|
|
|
|
|
|
|
|
|
367
|
|
|
|
|
|
|
|
|
368
|
|
|
|
|
|
|
sub DEMOLISH { |
|
369
|
155
|
|
|
155
|
0
|
392396
|
my $self = shift; |
|
370
|
155
|
50
|
|
|
|
3202
|
$self->close if $self->_autoclose |
|
371
|
|
|
|
|
|
|
} |
|
372
|
|
|
|
|
|
|
|
|
373
|
|
|
|
|
|
|
|
|
374
|
|
|
|
|
|
|
1; |
|
375
|
|
|
|
|
|
|
|
|
376
|
|
|
|
|
|
|
__END__ |