You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1097 lines
32 KiB
1097 lines
32 KiB
#!/usr/bin/perl |
|
# |
|
# Manages VNC ports for server VMs that have VNC enabled. |
|
# |
|
|
|
use strict; |
|
use warnings; |
|
use Anvil::Tools; |
|
|
|
$| = 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(); |
|
|
|
sub call |
|
{ |
|
my $parameters = shift; |
|
my $call = $parameters->{call}; |
|
my $debug = $parameters->{debug} // 3; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => $parameters, prefix => "call" }); |
|
|
|
return (1) if ( (not defined $call) || ($call eq "") ); |
|
|
|
my $shell_output; |
|
my $shell_return_code; |
|
|
|
my ($output, $rcode) = $anvil->System->call({ shell_call => $call }); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => $debug, list => { |
|
output => $output, |
|
rcode => $rcode |
|
}, prefix => "call" }); |
|
|
|
return ($rcode, $output); |
|
} |
|
|
|
sub get_server_info |
|
{ |
|
my $parameters = shift; |
|
my $server_host_name = $parameters->{server_host_name}; |
|
my $server_host_uuid = $parameters->{server_host_uuid}; |
|
my $server_name = $parameters->{server_name}; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
return if (not defined $server_uuid); |
|
|
|
my $query; |
|
my $server_info; |
|
|
|
# When all required server info are provided, i.e., extracted from |
|
# libvirt domain XML, simply return the values. |
|
if (defined $server_host_name and defined $server_host_uuid and defined $server_name) |
|
{ |
|
$server_info = { |
|
host_name => $server_host_name, |
|
host_uuid => $server_host_uuid, |
|
server_name => $server_name |
|
}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $server_info, prefix => "get_server_info" }); |
|
|
|
return $server_info; |
|
} |
|
|
|
if (defined $server_host_uuid) |
|
{ |
|
$query = " |
|
SELECT a.server_name, b.host_name, b.host_uuid |
|
FROM servers AS a, hosts AS b |
|
WHERE server_uuid = ".$anvil->Database->quote($server_uuid)." |
|
AND host_uuid = ".$anvil->Database->quote($server_host_uuid)." |
|
;"; |
|
} |
|
else |
|
{ |
|
$query = " |
|
SELECT a.server_name, b.host_name, b.host_uuid |
|
FROM servers AS a |
|
JOIN hosts AS b ON a.server_host_uuid = b.host_uuid |
|
WHERE server_uuid = ".$anvil->Database->quote($server_uuid)." |
|
;"; |
|
} |
|
|
|
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
my $count = @{$results}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { query => $query, count => $count }, prefix => "get_server_info" }); |
|
|
|
if ($count == 1) |
|
{ |
|
my $row = $results->[0]; |
|
|
|
$server_info = {}; |
|
$server_info->{server_name} = $row->[0]; |
|
$server_info->{host_name} = $row->[1]; |
|
$server_info->{host_uuid} = $row->[2]; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $server_info, prefix => "get_server_info" }); |
|
} |
|
|
|
return $server_info; |
|
} |
|
|
|
sub get_vnc_info |
|
{ |
|
my $parameters = shift; |
|
my $host_name = $parameters->{host_name}; |
|
my $port = $parameters->{port}; |
|
my $port_base = $parameters->{port_base} // 5900; |
|
my $server_name = $parameters->{server_name}; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
my $port_offset; |
|
my $vnc_info; |
|
|
|
if ( (not defined $port) or (not $port =~ /^\d+$/) ) |
|
{ |
|
# Requires root to access VM information. |
|
my $shell_call = "virsh vncdisplay \"$server_name\""; |
|
|
|
my ($shell_return_code, $shell_output) = call({ |
|
call => $shell_call, |
|
host_name => $host_name, |
|
user => "root", |
|
}); |
|
|
|
return if ($shell_return_code != 0); |
|
|
|
($port_offset) = $shell_output =~ /:(\d+)$/; |
|
$port = $port_base + int($port_offset); |
|
} |
|
else |
|
{ |
|
$port = int($port); |
|
} |
|
|
|
$vnc_info = { host_name => $host_name, port => $port }; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
port_offset => $port_offset, |
|
vnc_port => $vnc_info->{port} |
|
}, prefix => "get_vnc_info" }); |
|
|
|
return $vnc_info; |
|
} |
|
|
|
sub get_available_port |
|
{ |
|
my $parameters = shift; |
|
my $start_port = $parameters->{start_port}; |
|
my $host_name = $parameters->{host_name}; |
|
my $step_operator = ((defined $parameters->{step_operator}) and ($parameters->{step_operator} =~ /^[+-]$/)) ? $parameters->{step_operator} : "+"; |
|
my $step_size = ( |
|
(defined $parameters->{step_size}) |
|
and ($parameters->{step_size} =~ /^\d+$/) |
|
and ($parameters->{step_size} > 0) |
|
) ? $parameters->{step_size} : 1; |
|
|
|
my $available_port; |
|
|
|
my $shell_call = "ss_output=\$(ss --all --tcp --numeric) && port=".$start_port." && while egrep -q \":\${port}[[:space:]]+[^[:space:]]+\" <<<\$ss_output; do (( port ".$step_operator."= ".$step_size." )); done && echo \$port"; |
|
|
|
my ($shell_return_code, $shell_output) = call({ host_name => $host_name, call => $shell_call }); |
|
|
|
if ($shell_return_code == 0) |
|
{ |
|
$available_port = $shell_output; |
|
} |
|
|
|
return $available_port; |
|
} |
|
|
|
sub is_websockify_process |
|
{ |
|
my $parameters = shift; |
|
my $host_name = $parameters->{host_name}; |
|
my $ws_pid = $parameters->{ws_pid}; |
|
my $shell_call = "ps -e -o command -h -p ".$ws_pid; |
|
|
|
my ($shell_return_code, $shell_output) = call({ host_name => $host_name, call => $shell_call }); |
|
|
|
return $shell_output =~ /websockify/ ? 1 : 0; |
|
} |
|
|
|
sub is_ssh_process |
|
{ |
|
my $parameters = shift; |
|
my $host_name = $parameters->{host_name}; |
|
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
|
|
|
my $shell_call = "ps -e -o command -h -p ".$ssh_tunnel_pid; |
|
|
|
my ($shell_return_code, $shell_output) = call({ host_name => $host_name, call => $shell_call }); |
|
|
|
return $shell_output =~ /anvil-manage-tunnel/ ? 1 : 0; |
|
} |
|
|
|
sub is_websockify_exists |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $host_uuid = $parameters->{host_uuid}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
my $server_vnc_port = $parameters->{server_vnc_port}; |
|
|
|
# Expect 0 to 1 record(s) because each VM server can only have 1 |
|
# websockify instance (not considering shared VNC sessions). |
|
my $query = " |
|
SELECT |
|
a.server_vnc_port, |
|
b.host_name, |
|
a.ws_host_uuid, |
|
a.ws_pid, |
|
a.ws_dest_port, |
|
a.ssh_tunnel_host_uuid |
|
FROM vnc_pipes AS a |
|
JOIN hosts AS b ON a.ws_host_uuid = b.host_uuid |
|
WHERE |
|
a.server_uuid = ".$anvil->Database->quote($server_uuid)." |
|
AND |
|
a.ws_host_uuid = ".$anvil->Database->quote($ws_host_uuid)." |
|
AND |
|
a.server_vnc_port IS NOT NULL |
|
ORDER BY a.ws_pid DESC |
|
;"; |
|
|
|
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
my $count = @{$results}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { query => $query, count => $count }, prefix => "is_websockify_exists" }); |
|
|
|
my $ws_exists_info = { exists_code => 0 }; |
|
|
|
foreach my $row (@{$results}) |
|
{ |
|
my $server_vnc_port_in_record = $row->[0]; |
|
my $ws_host_name = $row->[1]; |
|
my $ws_host_uuid_in_record = $row->[2]; |
|
my $ws_pid = $row->[3]; |
|
my $ws_dest_port = $row->[4]; |
|
my $ssh_tunnel_host_uuid = $row->[5]; |
|
my $clean_up_parameters = { host_name => $ws_host_name, ws_pid => $ws_pid }; |
|
|
|
$ws_exists_info->{exists_code} = 1; |
|
|
|
if ($ws_host_uuid ne $ws_host_uuid_in_record) |
|
{ |
|
# VNC server host mismatch; try to stop the recorded instance. |
|
# Likely happens after a server migration. |
|
stop_websockify($clean_up_parameters); |
|
|
|
# No need to preserve the websockify source port in |
|
# this case because the tunnel will need to be replaced |
|
# as well. |
|
} |
|
elsif ($server_vnc_port != $server_vnc_port_in_record) |
|
{ |
|
# VNC server port mismatch; try to stop the recorded instance. |
|
stop_websockify($clean_up_parameters); |
|
|
|
$ws_exists_info->{ws_dest_port} = $ws_dest_port; |
|
} |
|
elsif (not is_websockify_process($clean_up_parameters)) |
|
{ |
|
# The recorded instance died. |
|
$ws_exists_info->{ws_dest_port} = $ws_dest_port; |
|
} |
|
else |
|
{ |
|
# Found one websockify instance that passed all tests. |
|
$ws_exists_info->{exists_code} = 2; |
|
$ws_exists_info->{ws_pid} = $ws_pid; |
|
$ws_exists_info->{ws_dest_port} = $ws_dest_port; |
|
} |
|
} |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $ws_exists_info, prefix => "is_websockify_exists" }); |
|
|
|
return $ws_exists_info; |
|
} |
|
|
|
sub is_ssh_tunnel_exists |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
my $ws_dest_port = $parameters->{ws_dest_port}; |
|
|
|
# Expect 0 to n record(s), where n is the number of subnodes: |
|
# n=0: no SSH tunnel connected to the server's ws instance |
|
# n>0: at least one SSH tunnel is known, regardless of whether it is |
|
# actually alive |
|
# |
|
# Order by clause ensures NULL record(s) are at first. |
|
my $query = " |
|
SELECT |
|
a.ws_host_uuid, |
|
a.ws_pid, |
|
a.ws_dest_port, |
|
a.ssh_tunnel_dest_port, |
|
b.host_name, |
|
a.ssh_tunnel_host_uuid, |
|
a.ssh_tunnel_pid, |
|
a.ssh_tunnel_source_port |
|
FROM vnc_pipes AS a |
|
JOIN hosts AS b ON a.ssh_tunnel_host_uuid = b.host_uuid |
|
WHERE |
|
a.server_uuid = ".$anvil->Database->quote($server_uuid)." |
|
AND |
|
a.ssh_tunnel_host_uuid IS NOT NULL |
|
ORDER BY a.ws_pid DESC |
|
;"; |
|
|
|
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
my $count = @{$results}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { query => $query, count => $count }, prefix => "is_ssh_tunnel_exists" }); |
|
|
|
my $ssh_tunnel_exists_info = { exists_code => 0 }; |
|
|
|
foreach my $row (@{$results}) |
|
{ |
|
my $row = $results->[0]; |
|
my $ws_host_uuid_in_record = $row->[0]; |
|
my $ws_pid_in_record = $row->[1]; |
|
my $ws_dest_port_in_record = $row->[2]; |
|
my $ssh_tunnel_dest_port = $row->[3]; |
|
my $ssh_tunnel_host_name = $row->[4]; |
|
my $ssh_tunnel_host_uuid_in_record = $row->[5]; |
|
my $ssh_tunnel_pid = $row->[6]; |
|
my $ssh_tunnel_source_port = $row->[7]; |
|
my $clean_up_parameters = { host_name => $ssh_tunnel_host_name, ssh_tunnel_pid => $ssh_tunnel_pid }; |
|
|
|
$ssh_tunnel_exists_info->{exists_code} = 1; |
|
|
|
if (not defined $ws_pid_in_record and defined $ssh_tunnel_pid) |
|
{ |
|
# Websockify instance doesn't exist but tunnel does. |
|
# Remove the tunnel that's not connected to anything. |
|
stop_ssh_tunnel($clean_up_parameters); |
|
|
|
clear_vnc_pipe_st({ server_uuid => $server_uuid, ws_host_uuid => $ws_host_uuid_in_record }); |
|
|
|
# No need to preserve any tunnel info because this |
|
# tunnel doesn't need to be recreated. |
|
} |
|
elsif ($ssh_tunnel_source_port != $ws_dest_port_in_record) |
|
{ |
|
# Tunnel is not pointing to the destination port of the |
|
# websockify instance; stop it and recreate. |
|
stop_ssh_tunnel($clean_up_parameters); |
|
|
|
# Port mismatch means we need to recreate the tunnel. |
|
$ssh_tunnel_exists_info->{ssh_tunnel_dest_port} = $ssh_tunnel_dest_port; |
|
} |
|
elsif (not is_ssh_process($clean_up_parameters)) |
|
{ |
|
# The tunnel pid doesn't point to an active tunnel; |
|
# mark as need-to-be recreated. |
|
$ssh_tunnel_exists_info->{ssh_tunnel_dest_port} = $ssh_tunnel_dest_port; |
|
} |
|
else |
|
{ |
|
# Passed all tests; tunnel considered exists. |
|
$ssh_tunnel_exists_info->{exists_code} = 2; |
|
$ssh_tunnel_exists_info->{ssh_tunnel_dest_port} = $ssh_tunnel_dest_port; |
|
$ssh_tunnel_exists_info->{ssh_tunnel_pid} = $ssh_tunnel_pid; |
|
} |
|
} |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $ssh_tunnel_exists_info, prefix => "is_ssh_tunnel_exists" }); |
|
|
|
return $ssh_tunnel_exists_info; |
|
} |
|
|
|
sub is_websockify_in_use_by_others |
|
{ |
|
my $parameters = shift; |
|
my $ws_pid = $parameters->{ws_pid}; |
|
|
|
my $query = "SELECT COUNT(*) FROM public.vnc_pipes WHERE ws_pid = ".$anvil->Database->quote($ws_pid).";"; |
|
|
|
my $count = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ })->[0]->[0]; |
|
|
|
return $count > 1 ? 1 : 0; |
|
} |
|
|
|
sub is_remote_host_name |
|
{ |
|
my $host_name = shift; |
|
|
|
return ((defined $host_name) and ($host_name ne $anvil->data->{sys}{host_name})) ? 1 : 0; |
|
} |
|
|
|
sub start_websockify |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $host_uuid = $parameters->{host_uuid}; |
|
my $ws_host_name = $parameters->{ws_host_name}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
my $target_port = $parameters->{target_port}; |
|
my $source_port = $parameters->{source_port}; |
|
my $ws_info; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $parameters, prefix => "start_websockify" }); |
|
|
|
my $ws_exists_info = is_websockify_exists({ |
|
server_uuid => $server_uuid, |
|
host_uuid => $host_uuid, |
|
ws_host_uuid => $ws_host_uuid, |
|
server_vnc_port => $target_port |
|
}); |
|
|
|
if ($ws_exists_info->{exists_code} == 2) |
|
{ |
|
$ws_info = {}; |
|
$ws_info->{pid} = $ws_exists_info->{ws_pid}; |
|
$ws_info->{source_port} = $ws_exists_info->{ws_dest_port}; |
|
} |
|
else |
|
{ |
|
if (not defined $source_port) |
|
{ |
|
if (defined $ws_exists_info->{ws_dest_port}) |
|
{ |
|
$source_port = $ws_exists_info->{ws_dest_port}; |
|
} |
|
else |
|
{ |
|
my $source_port_base = 10000; |
|
|
|
$source_port = $source_port_base + $target_port; |
|
} |
|
} |
|
|
|
$source_port = get_available_port({ start_port => $source_port, host_name => $ws_host_name }); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
source_port => $source_port |
|
}, prefix => "start_websockify" }); |
|
|
|
return if (not defined $source_port); |
|
|
|
my $shell_call = "websockify ".$source_port." :".$target_port." &>/dev/null & echo pid:\$!"; |
|
|
|
my ($shell_return_code, $shell_output) = call({ |
|
call => $shell_call, |
|
host_name => $ws_host_name, |
|
user => "admin", |
|
}); |
|
|
|
return if ($shell_return_code != 0); |
|
|
|
my ($ws_pid) = $shell_output =~ /pid:(\d+)$/; |
|
|
|
$ws_info = {}; |
|
$ws_info->{pid} = $ws_pid; |
|
$ws_info->{source_port} = $source_port; |
|
|
|
if ($ws_exists_info->{exists_code} == 1) |
|
{ |
|
$ws_info->{is_update} = 1; |
|
} |
|
else |
|
{ |
|
$ws_info->{is_new} = 1; |
|
} |
|
} |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $ws_info, prefix => "start_websockify" }); |
|
|
|
return $ws_info; |
|
} |
|
|
|
sub stop_websockify |
|
{ |
|
my $parameters = shift; |
|
my $host_name = $parameters->{host_name}; |
|
my $ws_pid = $parameters->{ws_pid}; |
|
|
|
return if (not is_websockify_process($parameters)); |
|
|
|
call({ host_name => $host_name, call => "kill $ws_pid || kill -9 $ws_pid" }); |
|
} |
|
|
|
sub start_ssh_tunnel |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $host_name = $parameters->{host_name}; |
|
my $host_uuid = $parameters->{host_uuid}; |
|
my $ws_host_name = $parameters->{ws_host_name}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
my $ws_dest_port = $parameters->{ws_dest_port}; |
|
my $ssh_tunnel_dest_port = $parameters->{ssh_tunnel_dest_port}; |
|
my $ssh_tunnel_info; |
|
|
|
my $ssh_tunnel_exists_info = is_ssh_tunnel_exists({ |
|
server_uuid => $server_uuid, |
|
ssh_tunnel_host_uuid => $host_uuid, |
|
ws_host_uuid => $ws_host_uuid, |
|
ws_dest_port => $ws_dest_port |
|
}); |
|
|
|
if ($ssh_tunnel_exists_info->{exists_code} == 2) |
|
{ |
|
$ssh_tunnel_info = {}; |
|
$ssh_tunnel_info->{pid} = $ssh_tunnel_exists_info->{ssh_tunnel_pid}; |
|
$ssh_tunnel_info->{forward_port} = $ssh_tunnel_exists_info->{ssh_tunnel_dest_port}; |
|
} |
|
else |
|
{ |
|
if (not defined $ssh_tunnel_dest_port) |
|
{ |
|
if (defined $ssh_tunnel_exists_info->{ssh_tunnel_dest_port}) |
|
{ |
|
$ssh_tunnel_dest_port = $ssh_tunnel_exists_info->{ssh_tunnel_dest_port}; |
|
} |
|
else |
|
{ |
|
$ssh_tunnel_dest_port = $ws_dest_port; |
|
} |
|
} |
|
|
|
$ssh_tunnel_dest_port = get_available_port({ start_port => $ssh_tunnel_dest_port }); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
ssh_tunnel_dest_port => $ssh_tunnel_dest_port |
|
}, prefix => "start_ssh_tunnel" }); |
|
|
|
if (not defined $ssh_tunnel_dest_port) |
|
{ |
|
return; |
|
} |
|
|
|
my $shell_call = $anvil->data->{path}{exe}{'anvil-manage-tunnel'} |
|
." --remote-user admin --target ".$ws_host_name |
|
." --forward-local-port ".$ssh_tunnel_dest_port |
|
." --forward-remote-port ".$ws_dest_port |
|
." &>/dev/null & echo pid:\$!"; |
|
|
|
my ($shell_return_code, $shell_output) = call({ host_name => $host_name, call => $shell_call }); |
|
|
|
if ($shell_return_code == 0) |
|
{ |
|
my ($ssh_tunnel_pid) = $shell_output =~ /pid:(\d+)$/; |
|
|
|
$ssh_tunnel_info = {}; |
|
$ssh_tunnel_info->{pid} = $ssh_tunnel_pid; |
|
$ssh_tunnel_info->{forward_port} = $ssh_tunnel_dest_port; |
|
|
|
if ($ssh_tunnel_exists_info->{exists_code} == 1) |
|
{ |
|
$ssh_tunnel_info->{is_update} = 1; |
|
} |
|
else |
|
{ |
|
$ssh_tunnel_info->{is_new} = 1; |
|
} |
|
} |
|
} |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $ssh_tunnel_info, prefix => "start_ssh_tunnel" }); |
|
|
|
return $ssh_tunnel_info; |
|
} |
|
|
|
sub stop_ssh_tunnel |
|
{ |
|
my $parameters = shift; |
|
my $host_name = $parameters->{host_name}; |
|
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
|
|
|
return if (not is_ssh_process($parameters)); |
|
|
|
call({ host_name => $host_name, call => "kill $ssh_tunnel_pid || kill -9 $ssh_tunnel_pid" }); |
|
} |
|
|
|
sub create_vnc_pipes_table |
|
{ |
|
my $query = " |
|
CREATE TABLE IF NOT EXISTS public.vnc_pipes ( |
|
uuid uuid not null primary key, |
|
server_uuid uuid not null, |
|
server_vnc_port numeric, |
|
ws_host_uuid uuid, |
|
ws_pid numeric, |
|
ws_dest_port numeric, |
|
ssh_tunnel_dest_port numeric, |
|
ssh_tunnel_host_uuid uuid, |
|
ssh_tunnel_pid numeric, |
|
ssh_tunnel_source_port numeric, |
|
modified_date timestamp with time zone not null, |
|
unique(server_uuid, ws_host_uuid) |
|
);"; |
|
|
|
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
} |
|
|
|
sub drop_vnc_pipes_table |
|
{ |
|
my $query = "DROP TABLE IF EXISTS public.vnc_pipes;"; |
|
|
|
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
} |
|
|
|
sub insert_or_update_vnc_pipe |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
return (1) if (not defined $server_uuid); |
|
|
|
my $server_vnc_port = $parameters->{server_vnc_port}; |
|
my $ssh_tunnel_dest_port = $parameters->{ssh_tunnel_dest_port}; |
|
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid}; |
|
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
|
my $ssh_tunnel_source_port = $parameters->{ssh_tunnel_source_port}; |
|
my $vnc_pipe_uuid = $parameters->{vnc_pipe_uuid} // $anvil->Get->uuid(); |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
my $ws_pid = $parameters->{ws_pid}; |
|
my $ws_dest_port = $parameters->{ws_dest_port}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $parameters, prefix => "insert_or_update_vnc_pipe" }); |
|
|
|
my $vnc_pipe_mdate = $anvil->Database->refresh_timestamp(); |
|
my $quoted_vnc_pipe_mdate = $anvil->Database->quote($vnc_pipe_mdate); |
|
|
|
my $insert_columns = ""; |
|
my $insert_values = ""; |
|
my $update_values = ""; |
|
|
|
foreach my $column_name ( |
|
"server_vnc_port", |
|
"ws_host_uuid", |
|
"ws_pid", |
|
"ws_dest_port", |
|
"ssh_tunnel_dest_port", |
|
"ssh_tunnel_host_uuid", |
|
"ssh_tunnel_pid", |
|
"ssh_tunnel_source_port" |
|
) |
|
{ |
|
my $column_value = $parameters->{$column_name}; |
|
|
|
next if (not defined $column_value); |
|
|
|
my $quoted_value = ($column_value eq "NULL") |
|
? $column_value |
|
: $anvil->Database->quote($column_value); |
|
|
|
$insert_columns .= "$column_name,\n\t"; |
|
$insert_values .= "$quoted_value,\n\t"; |
|
$update_values .= "$column_name = $quoted_value,\n\t"; |
|
} |
|
|
|
my $query = " |
|
INSERT INTO public.vnc_pipes ( |
|
uuid, |
|
server_uuid, |
|
$insert_columns |
|
modified_date |
|
) VALUES ( |
|
".$anvil->Database->quote($vnc_pipe_uuid).", |
|
".$anvil->Database->quote($server_uuid).", |
|
$insert_values |
|
$quoted_vnc_pipe_mdate |
|
) ON CONFLICT (server_uuid, ws_host_uuid) DO UPDATE SET |
|
$update_values |
|
modified_date = $quoted_vnc_pipe_mdate;"; |
|
|
|
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
|
|
return (0); |
|
} |
|
|
|
sub clear_vnc_pipe_ws |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
|
|
return (1) if (not defined $ws_host_uuid); |
|
|
|
insert_or_update_vnc_pipe({ |
|
server_uuid => $server_uuid, |
|
server_vnc_port => "NULL", |
|
ws_dest_port => "NULL", |
|
ws_host_uuid => $ws_host_uuid, |
|
ws_pid => "NULL" |
|
}); |
|
|
|
return (0); |
|
} |
|
|
|
sub clear_vnc_pipe_st |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
|
|
return (1) if (not defined $ws_host_uuid); |
|
|
|
insert_or_update_vnc_pipe({ |
|
server_uuid => $server_uuid, |
|
ssh_tunnel_dest_port => "NULL", |
|
ssh_tunnel_host_uuid => "NULL", |
|
ssh_tunnel_pid => "NULL", |
|
ssh_tunnel_source_port => "NULL", |
|
ws_host_uuid => $ws_host_uuid |
|
}); |
|
} |
|
|
|
sub get_vnc_pipe |
|
{ |
|
my $parameters = shift; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
return if (not defined $server_uuid); |
|
|
|
my $is_ws_pid = $parameters->{is_ws_pid}; |
|
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid}; |
|
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
|
|
|
my $vnc_pipe_info; |
|
|
|
my $cond_ws_pid = defined $is_ws_pid |
|
? "AND a.ws_pid IS NOT NULL" : ""; |
|
my $cond_ssht_huuid = defined $ssh_tunnel_host_uuid |
|
? "AND a.ssh_tunnel_host_uuid = ".$anvil->Database->quote($ssh_tunnel_host_uuid) : ""; |
|
my $cond_ws_huuid = defined $ws_host_uuid |
|
? "AND a.ws_host_uuid = ".$anvil->Database->quote($ws_host_uuid) : ""; |
|
|
|
my $query = " |
|
SELECT |
|
a.server_vnc_port, |
|
b.host_name, |
|
a.ws_host_uuid, |
|
a.ws_pid, |
|
a.ws_dest_port, |
|
a.ssh_tunnel_host_uuid, |
|
a.ssh_tunnel_pid, |
|
a.ssh_tunnel_dest_port |
|
FROM |
|
public.vnc_pipes AS a |
|
JOIN |
|
public.hosts AS b |
|
ON |
|
a.ws_host_uuid = b.host_uuid |
|
WHERE |
|
server_uuid = ".$anvil->Database->quote($server_uuid)." |
|
$cond_ws_pid |
|
$cond_ssht_huuid |
|
$cond_ws_huuid |
|
ORDER BY |
|
a.modified_date DESC |
|
;"; |
|
|
|
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
|
my $count = @{$results}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { query => $query, count => $count }, prefix => "get_vnc_pipe" }); |
|
|
|
if ($count == 1) |
|
{ |
|
my $row = $results->[0]; |
|
|
|
$vnc_pipe_info = {}; |
|
$vnc_pipe_info->{server_vnc_port} = $row->[0]; |
|
$vnc_pipe_info->{host_name} = $row->[1]; |
|
$vnc_pipe_info->{ws_host_uuid} = $row->[2]; |
|
$vnc_pipe_info->{ws_pid} = $row->[3]; |
|
$vnc_pipe_info->{ws_dest_port} = $row->[4]; |
|
$vnc_pipe_info->{ssh_tunnel_host_uuid} = $row->[5]; |
|
$vnc_pipe_info->{ssh_tunnel_pid} = $row->[6]; |
|
$vnc_pipe_info->{ssh_tunnel_dest_port} = $row->[7]; |
|
} |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $vnc_pipe_info, prefix => "get_vnc_pipe" }); |
|
|
|
return $vnc_pipe_info; |
|
} |
|
|
|
sub open_ws |
|
{ |
|
my $parameters = shift; |
|
my $server_host_name = $parameters->{server_host_name}; |
|
my $server_host_uuid = $parameters->{server_host_uuid}; |
|
my $server_name = $parameters->{server_name}; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
my $server_vnc_port = $parameters->{server_vnc_port}; |
|
|
|
my $server_info = $parameters->{server_info} // get_server_info($parameters); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $parameters, prefix => "open_ws" }); |
|
|
|
return (1, "error_0313") if (not defined $server_info); |
|
|
|
my $vnc_info = get_vnc_info({ |
|
host_name => $server_info->{host_name}, |
|
port => $server_vnc_port, |
|
server_name => $server_info->{server_name}, |
|
server_uuid => $server_uuid |
|
}); |
|
|
|
return (1, "error0314") if (not defined $vnc_info); |
|
|
|
my $ws_info = start_websockify({ |
|
server_uuid => $server_uuid, |
|
target_port => $vnc_info->{port}, |
|
ws_host_name => $server_info->{host_name}, |
|
ws_host_uuid => $server_info->{host_uuid} |
|
}); |
|
|
|
return (1, "error_0315") if (not defined $ws_info); |
|
|
|
if ($ws_info->{is_new} or $ws_info->{is_update}) |
|
{ |
|
insert_or_update_vnc_pipe({ |
|
server_uuid => $server_uuid, |
|
server_vnc_port => $vnc_info->{port}, |
|
ws_host_uuid => $server_info->{host_uuid}, |
|
ws_pid => $ws_info->{pid}, |
|
ws_dest_port => $ws_info->{source_port} |
|
}); |
|
} |
|
|
|
return (0, $ws_info); |
|
} |
|
|
|
sub close_ws |
|
{ |
|
my $parameters = shift; |
|
my $host_uuid = $parameters->{host_uuid}; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
my $vnc_pipe_info = $parameters->{vnc_pipe_info} // get_vnc_pipe({ server_uuid => $server_uuid, ws_host_uuid => $host_uuid }); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $parameters, prefix => "close_ws" }); |
|
|
|
return (1, "error_0317") if (not defined $vnc_pipe_info); |
|
|
|
return (0) if is_websockify_in_use_by_others({ ws_pid => $vnc_pipe_info->{ws_pid} }); |
|
|
|
stop_websockify({ host_name => $vnc_pipe_info->{host_name}, ws_pid => $vnc_pipe_info->{ws_pid} }); |
|
|
|
clear_vnc_pipe_ws({ server_uuid => $server_uuid, ws_host_uuid => $vnc_pipe_info->{ws_host_uuid} }); |
|
|
|
return (0); |
|
} |
|
|
|
sub open_st |
|
{ |
|
my $parameters = shift; |
|
my $host_uuid = $parameters->{host_uuid}; |
|
my $is_print = $parameters->{print} // (not $anvil->data->{switches}{'job-uuid'}); |
|
my $server_host_uuid = $parameters->{server_host_uuid}; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
my $server_info = $parameters->{server_info} // get_server_info($parameters); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $parameters, prefix => "open_st" }); |
|
|
|
return (1, "error_0313") if (not defined $server_info); |
|
|
|
my $vnc_pipe_info = get_vnc_pipe({ is_ws_pid => 1, server_uuid => $server_uuid, ws_host_uuid => $server_info->{host_uuid} }); |
|
|
|
return (1, "error_0317") if (not defined $vnc_pipe_info); |
|
|
|
my $ssh_tunnel_info = start_ssh_tunnel({ |
|
server_uuid => $server_uuid, |
|
host_uuid => $host_uuid, |
|
ws_host_name => $vnc_pipe_info->{host_name}, |
|
ws_host_uuid => $vnc_pipe_info->{ws_host_uuid}, |
|
ws_dest_port => $vnc_pipe_info->{ws_dest_port} |
|
}); |
|
|
|
return (1, "error_0316") if (not defined $ssh_tunnel_info); |
|
|
|
if ($ssh_tunnel_info->{is_new} or $ssh_tunnel_info->{is_update}) |
|
{ |
|
insert_or_update_vnc_pipe({ |
|
server_uuid => $server_uuid, |
|
ssh_tunnel_dest_port => $ssh_tunnel_info->{forward_port}, |
|
ssh_tunnel_host_uuid => $host_uuid, |
|
ssh_tunnel_pid => $ssh_tunnel_info->{pid}, |
|
ssh_tunnel_source_port => $vnc_pipe_info->{ws_dest_port}, |
|
ws_host_uuid => $vnc_pipe_info->{ws_host_uuid} |
|
}); |
|
} |
|
|
|
my $forward_port = $ssh_tunnel_info->{forward_port} // ""; |
|
|
|
$ssh_tunnel_info->{forward_port} = $forward_port; |
|
|
|
print "protocol:ws,forward_port:$forward_port\n" if ($is_print); |
|
|
|
return (0, $ssh_tunnel_info); |
|
} |
|
|
|
sub close_st |
|
{ |
|
my $parameters = shift; |
|
my $host_uuid = $parameters->{host_uuid}; |
|
my $server_uuid = $parameters->{server_uuid}; |
|
|
|
my $vnc_pipe_info = $parameters->{vnc_pipe_info} // get_vnc_pipe({ |
|
server_uuid => $server_uuid, |
|
ssh_tunnel_host_uuid => $host_uuid |
|
}); |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $parameters, prefix => "close_st" }); |
|
|
|
return (1, "error_0317") if (not defined $vnc_pipe_info); |
|
|
|
stop_ssh_tunnel({ ssh_tunnel_pid => $vnc_pipe_info->{ssh_tunnel_pid} }); |
|
|
|
clear_vnc_pipe_st({ server_uuid => $server_uuid, ws_host_uuid => $vnc_pipe_info->{ws_host_uuid} }); |
|
|
|
return (0); |
|
} |
|
|
|
$anvil->Get->switches; |
|
|
|
$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 }); |
|
} |
|
|
|
# Try to get a job UUID if not given. |
|
if (not $anvil->data->{switches}{'job-uuid'}) |
|
{ |
|
$anvil->data->{switches}{'job-uuid'} = $anvil->Job->get_job_uuid({ program => $THIS_FILE }); |
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
"switches::job-uuid" => $anvil->data->{switches}{'job-uuid'} |
|
} }); |
|
} |
|
|
|
# Handle this script as a job when job UUID is provided. |
|
if ($anvil->data->{switches}{'job-uuid'}) |
|
{ |
|
$anvil->Job->clear(); |
|
$anvil->Job->get_job_details(); |
|
$anvil->Job->update_progress({ |
|
progress => 1, |
|
job_picked_up_by => $$, |
|
job_picked_up_at => time, |
|
message => "message_0259" |
|
}); |
|
|
|
foreach my $line (split/\n/, $anvil->data->{jobs}{job_data}) |
|
{ |
|
if ($line =~ /server-uuid=(.*?)$/) |
|
{ |
|
$anvil->data->{switches}{'server-uuid'} = $1; |
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
'switches::server-uuid' => $anvil->data->{switches}{'server-uuid'} |
|
} }); |
|
} |
|
|
|
if ($line =~ /open=(.*?)$/) |
|
{ |
|
$anvil->data->{switches}{'open'} = $1; |
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
'switches::open' => $anvil->data->{switches}{'open'} |
|
} }); |
|
} |
|
|
|
if ($line =~ /drop-table=(.*?)$/) |
|
{ |
|
$anvil->data->{switches}{'drop-table'} = $1; |
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
'switches::drop-table' => $anvil->data->{switches}{'drop-table'} |
|
} }); |
|
} |
|
} |
|
} |
|
|
|
$anvil->Database->get_hosts(); |
|
$anvil->Database->get_anvils(); |
|
|
|
my $component = $anvil->data->{switches}{'component'} // "all"; |
|
my $is_drop_table = $anvil->data->{switches}{'drop-table'}; |
|
my $is_open = $anvil->data->{switches}{'open'}; |
|
my $server = $anvil->data->{switches}{'server'}; |
|
my $server_host_uuid = $anvil->data->{switches}{'server-host-uuid'}; |
|
my $server_uuid = $anvil->data->{switches}{'server-uuid'} // $anvil->Get->server_uuid_from_name({ server_name => $server }); |
|
my $server_vnc_port = $anvil->data->{switches}{'server-vnc-port'}; |
|
|
|
my $server_host_name; |
|
|
|
if (defined $server_host_uuid) |
|
{ |
|
if ($server_host_uuid eq "local") |
|
{ |
|
$server_host_uuid = $anvil->data->{sys}{host_uuid}; |
|
$server_host_name = $anvil->data->{sys}{host_name}; |
|
} |
|
else |
|
{ |
|
$server_host_name = $anvil->Get->host_name_from_uuid({ host_uuid => $server_host_uuid }); |
|
} |
|
} |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
|
component => $component, |
|
is_open => $is_open, |
|
is_drop_table => $is_drop_table, |
|
server => $server, |
|
server_host_name => $server_host_name, |
|
server_host_uuid => $server_host_uuid, |
|
server_uuid => $server_uuid, |
|
server_vnc_port => $server_vnc_port |
|
} }); |
|
|
|
my $map_to_operation = { |
|
st => { close => \&close_st, open => \&open_st }, |
|
ws => { close => \&close_ws, open => \&open_ws }, |
|
}; |
|
|
|
if ($server_uuid =~ /[a-f0-9]{8}-[a-f0-9]{4}-[1-5][a-f0-9]{3}-[89ab][a-f0-9]{3}-[a-f0-9]{12}/) |
|
{ |
|
create_vnc_pipes_table(); |
|
|
|
my $ops = $map_to_operation->{$component}; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => $ops }); |
|
|
|
$anvil->nice_exit({ exit_code => 1 }) if (not defined $ops); |
|
|
|
my $op = ($is_open) ? "open" : "close"; |
|
|
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { operation => $op } }); |
|
|
|
my $host_uuid = $anvil->data->{sys}{host_uuid}; |
|
|
|
my ($is_error, $error) = $ops->{$op}({ |
|
host_uuid => $host_uuid, |
|
server_host_name => $server_host_name, |
|
server_host_uuid => $server_host_uuid, |
|
server_name => $server, |
|
server_uuid => $server_uuid, |
|
server_vnc_port => $server_vnc_port |
|
}); |
|
|
|
if ($is_error) |
|
{ |
|
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 1, list => { |
|
error => $error, |
|
host_uuid => $host_uuid, |
|
server_uuid => $server_uuid |
|
} }); |
|
|
|
if (defined $error and $error =~ /^error_/) |
|
{ |
|
$anvil->Job->update_progress({ |
|
progress => 100, |
|
message => "$error,!!server_uuid!$server_uuid!!,!!host_uuid!$host_uuid!!", |
|
job_status => "failed" |
|
}); |
|
} |
|
|
|
$anvil->nice_exit({ exit_code => 2 }); |
|
} |
|
|
|
$anvil->Job->update_progress({ |
|
progress => 100, |
|
message => "message_0260,!!operation!$op!!,!!server_uuid!$server_uuid!!,!!host_uuid!$host_uuid!!" |
|
}); |
|
} |
|
elsif ($is_drop_table) |
|
{ |
|
drop_vnc_pipes_table(); |
|
|
|
$anvil->Job->update_progress({ progress => 100, message => "message_0261" }); |
|
} |
|
else |
|
{ |
|
$anvil->Job->update_progress({ progress => 100, message => "message_0262" }); |
|
|
|
$anvil->nice_exit({ exit_code => 3 }); |
|
} |
|
|
|
$anvil->nice_exit({ exit_code => 0 });
|
|
|