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 126 127 128 129
|
# 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, $length = undef, $offset = undef) { syswrite($to_fd, $json, $length, $offset) }
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 = 0;
my $bytes_to_write = length($json);
while ($written_bytes < $bytes_to_write) {
$written_bytes += _syswrite($to_fd, $json, $bytes_to_write - $written_bytes, $written_bytes) // 0;
if ($!) {
die('myjsonrpc: remote end terminated connection, stopping') if !DEBUG_JSON && $! =~ qr/Broken pipe/;
confess sprintf "syswrite failed: err: '%s'; written_bytes: %d/%d; JSON: '%s'", $!, $written_bytes, $bytes_to_write, $json;
}
}
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;
|