2021-07-16 15:38:53 +00:00
#!/usr/bin/perl
#
# Manages VNC ports for server VMs that have VNC enabled.
#
use strict;
use warnings;
use Anvil::Tools;
2023-07-18 07:09:42 +00:00
use Data::Dumper;
2021-07-16 15:38:53 +00:00
$| = 1;
my $THIS_FILE = ($0 =~ /^.*\/(.*)$/)[0];
my $running_directory = ($0 =~ /^(.*?)\/$THIS_FILE$/)[0];
if (($running_directory =~ /^\./) && ($ENV{PWD}))
{
$running_directory =~ s/^\./$ENV{PWD}/;
}
my $anvil = Anvil::Tools->new();
2023-07-17 11:03:30 +00:00
my $echo = $anvil->data->{path}{exe}{'echo'};
2023-07-17 21:28:05 +00:00
my $grep = $anvil->data->{path}{exe}{'grep'};
2023-07-17 11:03:30 +00:00
my $kill = $anvil->data->{path}{exe}{'kill'};
my $pgrep = $anvil->data->{path}{exe}{'pgrep'};
my $ss = $anvil->data->{path}{exe}{'ss'};
my $sed = $anvil->data->{path}{exe}{'sed'};
my $websockify = $anvil->data->{path}{exe}{'websockify'};
2023-07-14 21:45:22 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Get->switches;
2023-06-08 06:24:02 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Database->connect;
$anvil->Log->entry({ source => $THIS_FILE, line => __LINE__, level => 2, secure => 0, key => "log_0132" });
if (not $anvil->data->{sys}{database}{connections})
{
# No databases, exit.
$anvil->Log->entry({ source => $THIS_FILE, line => __LINE__, level => 0, 'print' => 1, priority => "err", key => "error_0003" });
$anvil->nice_exit({ exit_code => 1 });
}
2023-06-08 06:24:02 +00:00
2023-07-17 11:03:30 +00:00
my $switch_debug = $anvil->data->{switches}{'debug'};
my $open = $anvil->data->{switches}{'open'};
my $server = $anvil->data->{switches}{'server'};
my $server_uuid = $anvil->data->{switches}{'server-uuid'};
my $server_vnc_port = $anvil->data->{switches}{'server-vnc-port'};
2023-06-08 06:24:02 +00:00
2023-07-17 11:03:30 +00:00
if (defined $server)
{
$server_uuid //= is_uuid_v4($server) ? $server : $anvil->Get->server_uuid_from_name({ server_name => $server });
2023-06-08 06:24:02 +00:00
}
2023-07-17 21:30:42 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $switch_debug, list => {
2023-07-17 11:03:30 +00:00
open => $open,
server => $server,
server_uuid => $server_uuid,
server_vnc_port => $server_vnc_port
} });
2023-06-22 20:06:02 +00:00
2023-07-17 11:03:30 +00:00
my $map_to_operation = { start => \&start_pipe, stop => \&stop_pipe };
2023-06-15 21:12:27 +00:00
2023-07-17 21:53:16 +00:00
if ($server_uuid)
2023-07-17 11:03:30 +00:00
{
my $operation = $open ? "start" : "stop";
2023-06-22 20:06:02 +00:00
2023-07-17 21:30:42 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $switch_debug, list => { operation => $operation } });
2023-07-21 22:09:42 +00:00
my ($rcode, $err_msg) = $map_to_operation->{$operation}({
2023-07-17 11:03:30 +00:00
debug => $switch_debug,
svr_uuid => $server_uuid,
svr_vnc_port => $server_vnc_port,
});
2023-06-22 20:06:02 +00:00
2023-07-21 22:09:42 +00:00
if ($rcode)
{
$anvil->Log->entry({
source => $THIS_FILE,
line => __LINE__,
level => $switch_debug || 2,
raw => "[ Error ] - Operation $operation failed; CAUSE: $err_msg",
});
}
2023-07-17 11:03:30 +00:00
$anvil->nice_exit({ exit_code => $rcode });
}
2023-06-22 20:06:02 +00:00
2023-07-17 11:03:30 +00:00
$anvil->nice_exit({ exit_code => 0 });
2023-06-15 21:12:27 +00:00
2023-07-17 11:03:30 +00:00
#
# Functions
#
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
sub build_find_available_port_call
{
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $start = $parameters->{start};
my $step_operator = $parameters->{step_operator} // "+";
my $step_size = $parameters->{step_size} || 1;
2023-06-19 14:11:35 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "build_find_available_port_call" });
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return (1) if ( (not $step_operator =~ /^[+-]$/)
|| (not is_int($step_size)) || ($step_size < 1) );
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
my $call = "ss_output=\$($ss -ant) && port=${start} && while $grep -Eq \":\${port}[[:space:]]+[^[:space:]]+\" <<<\$ss_output; do (( port ${step_operator}= $step_size )); done && $echo \$port";
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return (0, $call);
2021-07-16 15:38:53 +00:00
}
2023-07-18 19:49:21 +00:00
sub build_vncinfo_variable_name
2021-07-16 15:38:53 +00:00
{
2023-07-18 19:08:27 +00:00
my ($svr_uuid) = @_;
2021-07-16 15:38:53 +00:00
2023-07-18 22:15:19 +00:00
return "server::${svr_uuid}::vncinfo";
2021-07-16 15:38:53 +00:00
}
2023-07-17 11:03:30 +00:00
sub call
2021-07-20 18:34:15 +00:00
{
2023-07-17 11:03:30 +00:00
my $parameters = shift;
2023-07-18 07:10:32 +00:00
my $background = $parameters->{background} || 0;
2023-07-17 11:03:30 +00:00
my $call = $parameters->{call};
my $debug = $parameters->{debug} || 3;
2023-06-08 06:24:02 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "call" });
2021-07-20 18:34:15 +00:00
2023-07-17 11:03:30 +00:00
return (1) if ( (not defined $call) || ($call eq "") );
2021-07-20 18:34:15 +00:00
2023-07-18 07:10:32 +00:00
my ($output, $rcode) = $anvil->System->call({ background => $background, shell_call => $call });
2021-07-20 18:34:15 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => {
output => $output,
rcode => $rcode
} });
2021-07-20 18:34:15 +00:00
2023-07-18 07:10:32 +00:00
# Output order reversed keep returns consistent.
2023-07-17 11:03:30 +00:00
return ($rcode, $output);
2021-07-20 18:34:15 +00:00
}
2023-07-17 11:03:30 +00:00
sub find_available_port
2021-07-16 15:38:53 +00:00
{
my $parameters = shift;
2023-07-17 11:03:30 +00:00
my $debug = $parameters->{debug} || 3;
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "find_available_port" });
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
my ($build_rcode, $call) = build_find_available_port_call($parameters);
2023-06-08 06:24:02 +00:00
2023-07-17 11:03:30 +00:00
return (1) if ($build_rcode);
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return call({ call => $call, debug => $debug });
2021-07-16 15:38:53 +00:00
}
2023-07-20 08:42:36 +00:00
sub find_end_port
{
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $svr_host_name = $parameters->{svr_host_name} // $anvil->data->{sys}{host_name};
my $svr_uuid = $parameters->{svr_uuid};
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "find_end_port" });
return (1) if (not defined $svr_uuid);
my $variable_name = build_vncinfo_variable_name($svr_uuid);
my $variable_value_pattern = "$svr_host_name:%";
# Look in the history variables table because libvirt hook operation
# 'stopped' gets triggered **after** the 'started' operation on the
# peer host during server migration.
my $query = "
SELECT
SUBSTRING(variable_value, ".$anvil->Database->quote("(\\d+)\$").")
FROM
history.variables
WHERE
variable_name = ".$anvil->Database->quote($variable_name)."
AND
variable_value LIKE ".$anvil->Database->quote($variable_value_pattern)."
ORDER BY
modified_date DESC
LIMIT 1
;";
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => { query => $query } });
my $results = $anvil->Database->query({ source => $THIS_FILE, line => __LINE__, query => $query });
$anvil->Log->entry({ source => $THIS_FILE, line => __LINE__, level => $debug, raw => prettify($results, "results") });
return (1) if (not @{$results});
return (0, int($results->[0]->[0]));
}
2023-07-18 18:21:18 +00:00
sub find_server_vnc_port
{
my $parameters = shift;
2023-07-20 08:42:36 +00:00
my $debug = $parameters->{debug} || 3;
2023-07-18 18:21:18 +00:00
my $svr_uuid = $parameters->{svr_uuid};
my $svr_vnc_port = $parameters->{svr_vnc_port};
2023-07-20 08:42:36 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "find_server_vnc_port" });
return (1) if (not defined $svr_uuid);
2023-07-18 19:08:27 +00:00
return (0, $svr_vnc_port) if (is_int($svr_vnc_port));
2023-07-18 18:21:18 +00:00
# If we don't have the server's VNC port, find it in its qemu-kvm process.
2023-07-18 19:08:27 +00:00
my ($rcode, $svr_processes) = $anvil->Server->find_processes({ debug => $debug });
return (1) if ($rcode);
2023-07-18 18:21:18 +00:00
my $svr_process = $svr_processes->{uuids}{$svr_uuid};
my $svr_vnc_alive = $svr_process->{vnc_alive};
return (1) if (not $svr_vnc_alive);
return (0, $svr_process->{vnc_port});
}
2023-07-18 19:08:27 +00:00
sub find_ws_processes
2021-07-16 15:38:53 +00:00
{
2023-07-17 11:03:30 +00:00
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
2023-07-18 19:08:27 +00:00
my $ps_name = $parameters->{ps_name} // "websockify";
2023-07-17 11:03:30 +00:00
2023-07-18 19:08:27 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "find_ws_processes" });
2023-07-17 11:03:30 +00:00
2024-03-23 02:21:17 +00:00
my $ps_call = "$pgrep -a '$ps_name' | $sed -En 's/^([[:digit:]]+).*${ps_name}.*[[:space:]:]+([[:digit:]]+)[[:space:]:]+([[:digit:]]+).*\$/\\1,\\2,\\3/p'";
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
my ($rcode, $output) = call({ call => $ps_call, debug => $debug });
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return (1) if ($rcode);
2021-07-16 15:38:53 +00:00
2023-07-18 19:08:27 +00:00
my $result = { pids => {}, sources => {}, targets => {} };
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
foreach my $line (split(/\n/, $output))
2023-06-19 14:11:35 +00:00
{
2023-07-17 11:03:30 +00:00
chomp($line);
2021-07-16 15:38:53 +00:00
2023-07-18 19:08:27 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => { ws_line => $line } });
2023-07-18 09:36:40 +00:00
2023-07-18 19:08:27 +00:00
my ($pid, $sport, $tport) = split(/,/, $line);
2023-07-18 09:36:40 +00:00
2023-07-18 19:08:27 +00:00
my $process = { pid => $pid, sport => $sport, tport => $tport };
2021-07-20 18:34:15 +00:00
2023-07-18 19:08:27 +00:00
set_ws_process({ debug => $debug, entry => $process, entries => $result });
2023-07-18 04:33:16 +00:00
}
2021-07-16 15:38:53 +00:00
2023-07-18 19:08:27 +00:00
$anvil->Log->entry({ source => $THIS_FILE, line => __LINE__, level => $debug, raw => prettify($result, "ws_processes") });
2023-07-18 07:09:42 +00:00
2023-07-18 04:33:16 +00:00
return (0, $result);
2023-07-17 11:03:30 +00:00
}
2023-06-09 05:43:32 +00:00
2023-07-17 11:03:30 +00:00
sub is_int
2021-07-16 15:38:53 +00:00
{
2023-07-18 19:08:27 +00:00
return defined $_[0] && $_[0] =~ /^\d+$/;
2023-07-17 11:03:30 +00:00
}
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
sub is_uuid_v4
{
2023-07-18 19:08:27 +00:00
return defined $_[0] && $_[0] =~ /[a-f0-9]{8}-[a-f0-9]{4}-[1-5][a-f0-9]{3}-[89ab][a-f0-9]{3}-[a-f0-9]{12}/;
2023-07-18 09:02:15 +00:00
}
2023-07-18 07:09:42 +00:00
sub prettify
{
my $var_value = shift;
my $var_name = shift;
local $Data::Dumper::Indent = 1;
local $Data::Dumper::Varname = $var_name;
2023-07-21 22:09:42 +00:00
local $Data::Dumper::Terse = (defined $var_name) ? 0 : 1;
2023-07-18 07:09:42 +00:00
return Dumper($var_value);
}
2023-07-18 04:33:16 +00:00
sub set_entry
2021-07-16 15:38:53 +00:00
{
2023-07-17 11:03:30 +00:00
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $handle_delete = $parameters->{handle_delete};
my $handle_set = $parameters->{handle_set};
2023-07-18 04:33:16 +00:00
my $id = $parameters->{id};
my $entry = $parameters->{entry};
my $entries = $parameters->{entries};
2023-07-17 11:03:30 +00:00
2023-07-18 07:09:42 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => {
%$parameters,
p_entry => prettify($entry),
p_entries => prettify($entries),
}, prefix => "set_entry" });
2021-07-16 15:38:53 +00:00
2023-07-18 04:33:16 +00:00
return (1) if (not defined $entries);
2023-07-17 11:03:30 +00:00
2023-07-18 04:33:16 +00:00
if (defined $entry)
2021-07-16 15:38:53 +00:00
{
2023-07-18 04:33:16 +00:00
$handle_set->($id, $entry, $entries);
2021-07-16 15:38:53 +00:00
}
2023-07-18 04:33:16 +00:00
elsif (defined $id)
2021-07-16 15:38:53 +00:00
{
2023-07-18 04:33:16 +00:00
$handle_delete->($id, $entry, $entries);
2023-07-17 11:03:30 +00:00
}
2021-07-20 18:34:15 +00:00
2023-07-18 07:09:42 +00:00
$anvil->Log->entry({ source => $THIS_FILE, line => __LINE__, level => $debug, raw => prettify($entries, "entries") });
2023-07-17 11:03:30 +00:00
return (0);
}
2021-07-20 18:34:15 +00:00
2023-07-18 19:49:21 +00:00
sub set_vncinfo_variable
2021-07-16 15:38:53 +00:00
{
2023-07-17 11:03:30 +00:00
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $end_port = $parameters->{end_port};
my $svr_uuid = $parameters->{svr_uuid};
2023-07-18 19:49:21 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "set_vncinfo_variable" });
my $local_host_name = $anvil->data->{sys}{host_name};
2023-07-17 11:03:30 +00:00
my ($variable_uuid) = $anvil->Database->insert_or_update_variables({
2023-07-18 19:08:27 +00:00
file => $THIS_FILE,
line => __LINE__,
2023-07-18 19:49:21 +00:00
variable_name => build_vncinfo_variable_name($svr_uuid),
variable_value => "${local_host_name}:${end_port}",
2023-07-17 11:03:30 +00:00
});
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return (1) if (not is_uuid_v4($variable_uuid));
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return (0);
2021-07-16 15:38:53 +00:00
}
2023-07-17 11:03:30 +00:00
sub set_ws_process
2021-07-16 15:38:53 +00:00
{
2023-07-17 11:03:30 +00:00
my $parameters = shift;
2021-07-21 14:58:36 +00:00
2023-07-17 11:03:30 +00:00
$parameters->{handle_delete} = sub {
my ($pid, $process, $processes) = @_;
2023-06-09 05:43:32 +00:00
2023-07-18 04:33:16 +00:00
$process = $processes->{pids}{$pid};
2023-07-17 11:03:30 +00:00
my $sport = $process->{sport};
my $tport = $process->{tport};
2023-06-09 05:43:32 +00:00
2023-07-17 22:14:54 +00:00
delete $processes->{pids}{$pid};
delete $processes->{sources}{$sport};
delete $processes->{targets}{$tport};
2023-07-17 11:03:30 +00:00
};
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
$parameters->{handle_set} = sub {
my ($pid, $process, $processes) = @_;
2021-07-16 15:38:53 +00:00
2023-07-18 04:33:16 +00:00
$pid = $process->{pid};
2023-07-17 11:03:30 +00:00
my $sport = $process->{sport};
my $tport = $process->{tport};
2023-06-19 14:11:35 +00:00
2023-07-20 07:10:44 +00:00
# The websockify daemon wrapper may remain alive, hence each
# port can map to mutiple pids.
my $spids = $processes->{sources}{$sport} // [];
my $tpids = $processes->{targets}{$tport} // [];
2023-07-17 22:14:54 +00:00
$processes->{pids}{$pid} = $process;
2023-07-20 07:10:44 +00:00
# Process identifiers are already ordered by pgrep, record them
# in ascending order.
$processes->{sources}{$sport} = [@{$spids}, $pid];
$processes->{targets}{$tport} = [@{$tpids}, $pid];
2023-07-17 11:03:30 +00:00
};
2023-07-18 04:33:16 +00:00
return set_entry($parameters);
2023-06-19 14:11:35 +00:00
}
2023-07-17 11:03:30 +00:00
sub start_pipe
2023-06-19 14:11:35 +00:00
{
my $parameters = shift;
2023-07-17 11:03:30 +00:00
my $debug = $parameters->{debug} || 3;
my $svr_uuid = $parameters->{svr_uuid};
my $svr_vnc_port = $parameters->{svr_vnc_port};
2023-06-19 14:11:35 +00:00
2023-07-21 22:09:42 +00:00
return (1, __LINE__.": [$svr_uuid]") if (not is_uuid_v4($svr_uuid));
2023-06-19 14:11:35 +00:00
2023-07-17 11:03:30 +00:00
my $common_params = { debug => $debug };
2023-06-19 14:11:35 +00:00
2023-07-17 11:03:30 +00:00
my $rcode;
2023-06-19 14:11:35 +00:00
2023-07-18 18:21:18 +00:00
($rcode, $svr_vnc_port) = find_server_vnc_port($parameters);
2023-06-09 05:43:32 +00:00
2023-07-21 22:09:42 +00:00
return ($rcode, __LINE__.": $rcode,[$svr_vnc_port]") if ($rcode);
2023-06-09 05:43:32 +00:00
2023-07-17 11:03:30 +00:00
($rcode, my $ws_processes) = find_ws_processes($common_params);
2021-07-16 15:38:53 +00:00
2023-07-21 22:09:42 +00:00
return ($rcode, __LINE__.": $rcode,[".prettify($ws_processes)."]") if ($rcode);
2023-06-09 05:43:32 +00:00
2023-07-17 11:03:30 +00:00
($rcode, my $ws_pid) = start_ws({ svr_vnc_port => $svr_vnc_port, ws_processes => $ws_processes, %$common_params });
2021-07-16 15:38:53 +00:00
2023-07-21 22:09:42 +00:00
return ($rcode, __LINE__.": $rcode,[$ws_pid]") if ($rcode);
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
my $ws_process = $ws_processes->{pids}{$ws_pid};
2023-07-21 22:09:42 +00:00
my $ws_sport = $ws_process->{sport};
($rcode) = set_vncinfo_variable({ end_port => $ws_sport, svr_uuid => $svr_uuid, %$common_params });
2021-07-16 15:38:53 +00:00
2023-07-21 22:09:42 +00:00
return ($rcode, __LINE__.": [$svr_uuid:$ws_sport],$rcode") if ($rcode);
2021-07-16 15:38:53 +00:00
2023-07-17 11:03:30 +00:00
return (0);
2023-06-09 05:43:32 +00:00
}
2023-07-17 11:03:30 +00:00
sub start_ws
2023-06-09 05:43:32 +00:00
{
2023-07-17 11:03:30 +00:00
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $svr_vnc_port = $parameters->{svr_vnc_port};
my $ws_processes = $parameters->{ws_processes};
my $ws_sport_offset = $parameters->{ws_sport_offset} || 10000;
2023-06-09 05:43:32 +00:00
2023-07-17 11:03:30 +00:00
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "start_ws" });
2023-06-09 05:43:32 +00:00
2023-07-17 11:03:30 +00:00
return (1) if ( (not defined $ws_processes)
2023-07-18 19:08:27 +00:00
|| (not is_int($svr_vnc_port))
2023-07-17 11:03:30 +00:00
|| (not is_int($ws_sport_offset)) );
2023-06-09 05:43:32 +00:00
2023-07-20 07:10:44 +00:00
my $existing_ws_pids = $ws_processes->{targets}{$svr_vnc_port};
2023-06-09 05:43:32 +00:00
2023-07-20 07:10:44 +00:00
$anvil->Log->entry({ source => $THIS_FILE, line => __LINE__, level => $debug, raw => prettify($existing_ws_pids, "existing_ws_pids") });
2023-07-18 00:25:48 +00:00
2023-07-20 07:10:44 +00:00
return (0, $existing_ws_pids->[0]) if (defined $existing_ws_pids);
2023-06-09 05:43:32 +00:00
2023-07-18 21:06:45 +00:00
my $rcode;
($rcode, my $ws_sport) = find_available_port({ debug => $debug, start => int($svr_vnc_port) + int($ws_sport_offset) });
return ($rcode) if ($rcode);
2023-06-09 05:43:32 +00:00
2024-03-23 02:21:17 +00:00
my $ws_log = $anvil->data->{path}{directories}{log}."/anvil-ws-".$ws_sport."-".$svr_vnc_port.".log";
2023-07-20 07:10:44 +00:00
# The daemon wrapper can tell us whether the daemon started correctly;
# we won't know this if the process is started in the background.
2024-03-23 02:21:17 +00:00
my $ws_call = $websockify." -D --log-file '".$ws_log."' ".$ws_sport." :".$svr_vnc_port;
2023-06-09 05:43:32 +00:00
2023-07-18 21:06:45 +00:00
($rcode) = call({ call => $ws_call, debug => $debug });
2021-07-16 15:38:53 +00:00
2023-07-18 21:06:45 +00:00
return ($rcode) if ($rcode);
2021-07-16 15:38:53 +00:00
2023-07-18 21:06:45 +00:00
# Re-find to locate the new daemon.
($rcode, my $re_ws_processes) = find_ws_processes({ debug => $debug });
2021-07-16 17:49:19 +00:00
2023-07-18 21:06:45 +00:00
return ($rcode) if ($rcode);
2021-07-16 17:49:19 +00:00
2023-07-20 07:10:44 +00:00
my $ws_pid = $re_ws_processes->{targets}{$svr_vnc_port}->[0];
2023-07-18 21:06:45 +00:00
my $ws_process = $re_ws_processes->{pids}{$ws_pid};
2021-07-16 17:49:19 +00:00
2023-07-18 21:06:45 +00:00
# Remember the started daemon.
2023-07-18 04:33:16 +00:00
set_ws_process({ debug => $debug, entry => $ws_process, entries => $ws_processes });
2021-07-16 17:49:19 +00:00
2023-07-17 11:03:30 +00:00
return (0, $ws_pid);
2021-07-16 17:49:19 +00:00
}
2023-07-17 11:03:30 +00:00
sub stop_pipe
{
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $svr_uuid = $parameters->{svr_uuid};
my $svr_vnc_port = $parameters->{svr_vnc_port};
2021-07-16 15:38:53 +00:00
2023-07-21 22:09:42 +00:00
return (1, __LINE__.": [$svr_uuid]") if (not is_uuid_v4($svr_uuid));
2023-06-15 21:12:27 +00:00
2023-07-17 11:03:30 +00:00
my $common_params = { debug => $debug };
2023-06-22 20:06:02 +00:00
2023-07-17 11:03:30 +00:00
my $rcode;
2023-07-20 08:42:36 +00:00
($rcode, my $ws_processes) = find_ws_processes($common_params);
2021-07-16 15:38:53 +00:00
2023-07-21 22:09:42 +00:00
return ($rcode, __LINE__.": $rcode,[".prettify($ws_processes)."]") if ($rcode);
2021-07-16 15:38:53 +00:00
2023-07-20 08:42:36 +00:00
($rcode, $svr_vnc_port) = find_server_vnc_port($parameters);
2023-06-09 05:43:32 +00:00
2023-07-20 08:42:36 +00:00
my $ws_pids;
if ($rcode)
{
# The server VNC port is not available during the libvirt hook
# operation 'stopped'. Try to locate the websockify daemon by
# its source port.
($rcode, my $end_port) = find_end_port({ svr_uuid => $svr_uuid, %$common_params });
2023-07-21 22:09:42 +00:00
return ($rcode, __LINE__.": $rcode,[$end_port]") if ($rcode);
2023-06-09 05:43:32 +00:00
2023-07-20 08:42:36 +00:00
$ws_pids = $ws_processes->{sources}{$end_port};
}
else
{
$ws_pids = $ws_processes->{targets}{$svr_vnc_port};
}
2023-06-09 05:43:32 +00:00
2023-07-20 07:10:44 +00:00
foreach my $ws_pid (@{$ws_pids})
{
2023-07-21 22:09:42 +00:00
($rcode) = stop_ws({ ws_pid => $ws_pid, ws_processes => $ws_processes, %$common_params });
return ($rcode, __LINE__.": [$ws_pid],$rcode") if ($rcode);
2023-07-20 07:10:44 +00:00
}
2023-07-17 11:03:30 +00:00
return (0);
2021-07-16 20:32:08 +00:00
}
2023-06-09 05:43:32 +00:00
2023-07-17 11:03:30 +00:00
sub stop_ws
{
my $parameters = shift;
my $debug = $parameters->{debug} || 3;
my $ws_pid = $parameters->{ws_pid};
my $ws_processes = $parameters->{ws_processes};
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "stop_ws" });
2023-07-18 19:08:27 +00:00
return (1) if ( (not is_int($ws_pid)) || (not defined $ws_processes) );
2023-07-17 11:03:30 +00:00
call({ debug => $debug, call => "$kill $ws_pid || $kill -9 $ws_pid" });
2023-07-18 04:33:16 +00:00
set_ws_process({ debug => $debug, id => $ws_pid, entries => $ws_processes });
2023-07-18 19:08:27 +00:00
return (0);
2023-07-17 11:03:30 +00:00
}