forked from os-autoinst/os-autoinst
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmyjsonrpc.pm
125 lines (103 loc) · 4.21 KB
/
myjsonrpc.pm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
# Copyright 2012-2021 SUSE LLC
# SPDX-License-Identifier: GPL-2.0-or-later
package myjsonrpc;
use Mojo::Base -strict, -signatures;
use Carp qw(cluck confess);
use IO::Select;
use Errno;
use Mojo::JSON; # booleans
use Cpanel::JSON::XS ();
use bmwqemu ();
use constant DEBUG_JSON => $ENV{PERL_MYJSONRPC_DEBUG} || 0;
use constant READ_BUFFER => $ENV{PERL_MYJSONRPC_BYTES} || 8_000_000;
# hash for keeping state
our $sockets;
sub _syswrite($to_fd, $json) { syswrite($to_fd, $json) }
sub is_debug () { DEBUG_JSON || $bmwqemu::vars{DEBUG_JSON_RPC} }
sub handle_read_error ($fd) {
# throw an error except can_read has been interrupted
my $error = $!;
confess "ERROR: unable to wait for JSON reply: $error\n" unless $!{EINTR};
# try again if can_read's underlying system call has been interrupted as suggested by the perlipc documentation
bmwqemu::diag("read_json($fd): can_read's underlying system call has been interrupted, trying again\n") if is_debug; # uncoverable statement
}
sub send_json ($to_fd, $cmd) {
# allow regular expressions to be automatically converted into
# strings, using the Regex::TO_JSON function as defined at the end
# of this file.
# The resulting JSON should be in a single line, otherwise
# read_json won't work
my $cjx = Cpanel::JSON::XS->new->canonical->utf8->convert_blessed();
# deep copy to add a random string
my %cmdcopy = %$cmd;
# The hash might already contain a json_cmd_token
$cmdcopy{json_cmd_token} ||= bmwqemu::random_string(8);
my $json = $cjx->encode(\%cmdcopy);
bmwqemu::diag(sprintf("send_json(%d) JSON=%s", fileno($to_fd), $json =~ s/"([^"]{30})[^"]+"/"$1"/gr)) if is_debug();
$json .= "\n";
confess 'myjsonrpc: called on undefined file descriptor' unless defined $to_fd;
my $written_bytes = _syswrite($to_fd, $json);
if (!$written_bytes || $written_bytes != length($json)) {
die('myjsonrpc: remote end terminated connection, stopping') if !DEBUG_JSON && $! =~ qr/Broken pipe/;
confess "syswrite failed: $!";
}
return $cmdcopy{json_cmd_token};
}
# utility function
sub read_json ($socket, $cmd_token = undef, $multi = undef) {
my $cjx = Cpanel::JSON::XS->new->utf8;
my $fd = fileno($socket);
bmwqemu::diag("read_json($fd)") if is_debug();
if (exists $sockets->{$fd}) {
# start with the trailing text from previous call
my $buffer = delete $sockets->{$fd};
$cjx->incr_parse($buffer);
}
my $s = IO::Select->new();
$s->add($socket);
my @results;
# the goal here is to find the end of the next valid JSON - and don't
# add more data to it. As the backend sends things unasked, we might
# run into the next message otherwise
while (1) {
my $hash = $cjx->incr_parse();
# remember the trailing text
if ($hash) {
$sockets->{$fd} = $cjx->incr_text();
bmwqemu::diag(sprintf("read_json(%d) json_cmd_token=%s", $fd, $hash->{json_cmd_token} // 'no-token')) if is_debug();
if ($hash->{QUIT}) {
bmwqemu::diag("received magic close");
push @results, undef;
last;
}
confess "ERROR: the token does not match - questions and answers not in the right order" if $cmd_token && ($hash->{json_cmd_token} || '') ne $cmd_token; # uncoverable statement
push @results, $hash;
# parse all lines from buffer
next if $multi;
last;
}
elsif ($multi and @results) {
# read at least one item in list context
last;
}
# wait for next read
handle_read_error($fd) until (my @res = $s->can_read);
my $qbuffer;
if (!sysread($socket, $qbuffer, READ_BUFFER)) { bmwqemu::fctwarn("sysread failed: $!") if is_debug(); return }
$cjx->incr_parse($qbuffer);
}
return $multi ? @results : $results[0];
}
###################################################################
# enable send_json to send regular expressions
#<<< perltidy off
# this has to be on two lines so other tools don't believe this file
# exports package Regexp
package
Regexp;
#>>> perltidy on
sub TO_JSON ($regex) {
$regex = "$regex";
return $regex;
}
1;