parent
5cceb09f2a
commit
0935b9a990
1 changed files with 971 additions and 0 deletions
@ -0,0 +1,971 @@ |
||||
#!/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(); |
||||
|
||||
$anvil->Log->level({ set => 2 }); |
||||
|
||||
sub get_server_info |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $server_info; |
||||
|
||||
my $query = " |
||||
SELECT |
||||
ser.server_name, hos.host_name, hos.host_uuid |
||||
FROM |
||||
public.servers AS ser |
||||
JOIN |
||||
public.hosts AS hos |
||||
ON |
||||
ser.server_host_uuid = hos.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}; |
||||
|
||||
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_name => $server_info->{server_name}, |
||||
host_name => $server_info->{host_name}, |
||||
host_uuid => $server_info->{host_uuid} |
||||
} }); |
||||
} |
||||
|
||||
return $server_info; |
||||
} |
||||
|
||||
sub get_vnc_info |
||||
{ |
||||
my $parameters = shift; |
||||
my $host_name = $parameters->{host_name}; |
||||
my $server_name = $parameters->{server_name}; |
||||
my $port_base = 5900; |
||||
# Requires root to access VM information. |
||||
my $shell_call = "virsh vncdisplay ".$server_name; |
||||
my $vnc_info; |
||||
|
||||
my ($shell_output, $shell_error, $shell_return_code) = $anvil->Remote->call({ |
||||
target => $host_name, |
||||
remote_user => "root", |
||||
shell_call => $shell_call, |
||||
'close' => 1 |
||||
}); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_error => $shell_error, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
if ($shell_return_code == 0) |
||||
{ |
||||
my ($port_offset) = $shell_output =~ /:(\d+)$/; |
||||
|
||||
$vnc_info = { host_name => $host_name }; |
||||
$vnc_info->{port} = $port_base + int($port_offset); |
||||
|
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
port_offset => $port_offset, |
||||
vnc_port => $vnc_info->{port} |
||||
} }); |
||||
} |
||||
|
||||
return $vnc_info; |
||||
} |
||||
|
||||
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_output, $shell_error, $shell_return_code) = $anvil->Remote->call({ |
||||
target => $host_name, |
||||
remote_user => "admin", |
||||
shell_call => $shell_call, |
||||
'close' => 1 |
||||
}); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_error => $shell_error, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
return $shell_output =~ /websockify/ ? 1 : 0; |
||||
} |
||||
|
||||
sub is_ssh_process |
||||
{ |
||||
my $parameters = shift; |
||||
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
||||
my $shell_call = "ps -e -o command -h -p ".$ssh_tunnel_pid; |
||||
|
||||
my ($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call }); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
return $shell_output =~ /striker-open-ssh-tunnel/ ? 1 : 0; |
||||
} |
||||
|
||||
sub is_websockify_exists |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $host_uuid = $parameters->{host_uuid}; |
||||
my $server_vnc_port = $parameters->{server_vnc_port}; |
||||
|
||||
my $query = " |
||||
SELECT |
||||
vnc.server_vnc_port, hos.host_name, hos.host_uuid, vnc.ws_pid, vnc.ws_source_port |
||||
FROM |
||||
public.vnc_pipes AS vnc |
||||
JOIN |
||||
public.hosts AS hos |
||||
ON |
||||
vnc.ws_host_uuid = hos.host_uuid |
||||
WHERE |
||||
vnc.server_uuid = ".$anvil->Database->quote($server_uuid)." |
||||
;"; |
||||
|
||||
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
my $count = @{$results}; |
||||
my $ws_exists_info = { exists_code => 0 }; |
||||
|
||||
if ($count > 0) |
||||
{ |
||||
my $row = $results->[0]; |
||||
my $server_vnc_port_in_record = $row->[0]; |
||||
my $host_name = $row->[1]; |
||||
my $host_uuid_in_record = $row->[2]; |
||||
my $ws_pid = $row->[3]; |
||||
my $ws_source_port = $row->[4]; |
||||
my $clean_up_parameters = { host_name => $host_name, ws_pid => $ws_pid }; |
||||
|
||||
$ws_exists_info->{ws_pid} = $ws_pid; |
||||
$ws_exists_info->{ws_source_port} = $ws_source_port; |
||||
$ws_exists_info->{exists_code} = 1; |
||||
|
||||
if ($host_uuid ne $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. |
||||
delete $ws_exists_info->{ws_source_port}; |
||||
|
||||
return $ws_exists_info; |
||||
} |
||||
|
||||
if ($server_vnc_port != $server_vnc_port_in_record) |
||||
{ |
||||
# VNC server port mismatch; try to stop the recorded instance. |
||||
stop_websockify($clean_up_parameters); |
||||
|
||||
return $ws_exists_info; |
||||
} |
||||
|
||||
if (not is_websockify_process($clean_up_parameters)) |
||||
{ |
||||
# The recorded instance died. |
||||
return $ws_exists_info; |
||||
} |
||||
|
||||
# Passed all tests; process considered exists. |
||||
$ws_exists_info->{exists_code} = 2; |
||||
} |
||||
|
||||
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_source_port = $parameters->{ws_source_port}; |
||||
|
||||
my $query = " |
||||
SELECT |
||||
ws_host_uuid, ws_source_port, ssh_tunnel_pid, ssh_tunnel_forward_port |
||||
FROM |
||||
public.vnc_pipes |
||||
WHERE |
||||
server_uuid = ".$anvil->Database->quote($server_uuid)." |
||||
AND |
||||
ssh_tunnel_host_uuid = ".$anvil->Database->quote($ssh_tunnel_host_uuid)." |
||||
;"; |
||||
|
||||
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
my $count = @{$results}; |
||||
my $ssh_tunnel_exists_info = { exists_code => 0 }; |
||||
|
||||
if ($count == 1) |
||||
{ |
||||
my $row = $results->[0]; |
||||
my $ws_host_uuid_in_record = $row->[0]; |
||||
my $ws_source_port_in_record = $row->[1]; |
||||
my $ssh_tunnel_pid = $row->[2]; |
||||
my $ssh_tunnel_forward_port = $row->[3]; |
||||
my $clean_up_parameters = { ssh_tunnel_pid => $ssh_tunnel_pid }; |
||||
|
||||
$ssh_tunnel_exists_info->{ssh_tunnel_pid} = $ssh_tunnel_pid; |
||||
$ssh_tunnel_exists_info->{ssh_tunnel_forward_port} = $ssh_tunnel_forward_port; |
||||
$ssh_tunnel_exists_info->{exists_code} = 1; |
||||
|
||||
if ($ws_host_uuid ne $ws_host_uuid_in_record) |
||||
{ |
||||
# Websockify host mismatch; try to stop the recorded instance. |
||||
# Likely happens after a server migration. |
||||
stop_ssh_tunnel($clean_up_parameters); |
||||
|
||||
# No need to preserve the SSH tunnel forward port in |
||||
# this case because the websockify instance will need |
||||
# to be replaced as well. |
||||
delete $ssh_tunnel_exists_info->{ssh_tunnel_forward_port}; |
||||
|
||||
return $ssh_tunnel_exists_info; |
||||
} |
||||
|
||||
if ($ws_source_port != $ws_source_port_in_record) |
||||
{ |
||||
# Websockify source port mismatch; try to stop the recorded instance. |
||||
stop_ssh_tunnel($clean_up_parameters); |
||||
|
||||
return $ssh_tunnel_exists_info; |
||||
} |
||||
|
||||
if (not is_ssh_process($clean_up_parameters)) |
||||
{ |
||||
# The recorded tunnel died. |
||||
return $ssh_tunnel_exists_info; |
||||
} |
||||
|
||||
# Passed all tests; tunnel considered exists. |
||||
$ssh_tunnel_exists_info->{exists_code} = 2; |
||||
} |
||||
|
||||
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 start_websockify |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $host_name = $parameters->{host_name}; |
||||
my $host_uuid = $parameters->{host_uuid}; |
||||
my $target_port = $parameters->{target_port}; |
||||
my $source_port = $parameters->{source_port}; |
||||
my $ws_info; |
||||
|
||||
my $ws_exists_info = is_websockify_exists({ |
||||
server_uuid => $server_uuid, |
||||
host_uuid => $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_source_port}; |
||||
} |
||||
else |
||||
{ |
||||
if (not defined $source_port) |
||||
{ |
||||
if (defined $ws_exists_info->{ws_source_port}) |
||||
{ |
||||
$source_port = $ws_exists_info->{ws_source_port}; |
||||
} |
||||
else |
||||
{ |
||||
my $source_port_base = 10000; |
||||
|
||||
$source_port = $source_port_base + $target_port; |
||||
} |
||||
} |
||||
|
||||
my $shell_call = "websockify ".$source_port." :".$target_port." &>/dev/null & echo pid:\$!"; |
||||
|
||||
my ($shell_output, $shell_error, $shell_return_code) = $anvil->Remote->call({ |
||||
target => $host_name, |
||||
remote_user => "admin", |
||||
shell_call => $shell_call, |
||||
'close' => 1 |
||||
}); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_error => $shell_error, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
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_pid => $ws_pid, |
||||
ws_source_port => $source_port, |
||||
ws_is_update => $ws_info->{is_update}, |
||||
ws_is_new => $ws_info->{is_new} |
||||
} }); |
||||
} |
||||
} |
||||
|
||||
return $ws_info; |
||||
} |
||||
|
||||
sub stop_websockify |
||||
{ |
||||
my $parameters = shift; |
||||
my $host_name = $parameters->{host_name}; |
||||
my $ws_pid = $parameters->{ws_pid}; |
||||
|
||||
if (is_websockify_process($parameters)) |
||||
{ |
||||
my $shell_call = "kill ".$ws_pid; |
||||
my $remote_call_parameters = { |
||||
target => $host_name, |
||||
remote_user => "admin", |
||||
shell_call => $shell_call, |
||||
'close' => 1 |
||||
}; |
||||
my $shell_output; |
||||
my $shell_error; |
||||
my $shell_return_code; |
||||
|
||||
($shell_output, $shell_error, $shell_return_code) = $anvil->Remote->call($remote_call_parameters); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_error => $shell_error, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
sleep(2); |
||||
|
||||
if (is_websockify_process($parameters)) |
||||
{ |
||||
$shell_call = $shell_call =~ s/kill/kill -9/; |
||||
$remote_call_parameters->{shell_call} = $shell_call; |
||||
|
||||
($shell_output, $shell_error, $shell_return_code) = $anvil->Remote->call($remote_call_parameters); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
} |
||||
} |
||||
} |
||||
|
||||
sub start_ssh_tunnel |
||||
{ |
||||
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 $ws_source_port = $parameters->{ws_source_port}; |
||||
my $ssh_tunnel_forward_port = $parameters->{ssh_tunnel_forward_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_source_port => $ws_source_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_forward_port}; |
||||
} |
||||
else |
||||
{ |
||||
if (not defined $ssh_tunnel_forward_port) |
||||
{ |
||||
if (defined $ssh_tunnel_exists_info->{ssh_tunnel_forward_port}) |
||||
{ |
||||
$ssh_tunnel_forward_port = $ssh_tunnel_exists_info->{ssh_tunnel_forward_port}; |
||||
} |
||||
else |
||||
{ |
||||
$ssh_tunnel_forward_port = $ws_source_port; |
||||
} |
||||
} |
||||
|
||||
my $shell_call = $anvil->data->{path}{exe}{'striker-open-ssh-tunnel'} |
||||
." --remote-user admin --target ".$ws_host_name |
||||
." --forward-local-port ".$ssh_tunnel_forward_port |
||||
." --forward-remote-port ".$ws_source_port |
||||
." &>/dev/null & echo pid:\$!"; |
||||
|
||||
my ($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call }); |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
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_forward_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_pid => $ssh_tunnel_pid, |
||||
ssh_tunnel_forward_port => $ssh_tunnel_forward_port, |
||||
ssh_tunnel_is_update => $ssh_tunnel_info->{is_update}, |
||||
ssh_tunnel_is_new => $ssh_tunnel_info->{is_new} |
||||
} }); |
||||
} |
||||
} |
||||
|
||||
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}; |
||||
|
||||
if (is_ssh_process($parameters)) |
||||
{ |
||||
my $shell_call = "kill ".$ssh_tunnel_pid; |
||||
my $is_remote_call = (defined $host_name) and ($host_name ne $anvil->Get->host_name()) ? 1 : 0; |
||||
my $remote_call_parameters = { |
||||
target => $host_name, |
||||
remote_user => "admin", |
||||
shell_call => $shell_call, |
||||
'close' => 1 |
||||
}; |
||||
my $shell_output; |
||||
my $shell_error; |
||||
my $shell_return_code; |
||||
|
||||
if ($is_remote_call) |
||||
{ |
||||
($shell_output, $shell_error, $shell_return_code) = $anvil->Remote->call($remote_call_parameters); |
||||
} |
||||
else |
||||
{ |
||||
($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call }); |
||||
} |
||||
|
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_error => $shell_error, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
|
||||
sleep(2); |
||||
|
||||
if (is_ssh_process($parameters)) |
||||
{ |
||||
$shell_call = $shell_call =~ s/kill/kill -9/; |
||||
$remote_call_parameters->{shell_call} = $shell_call; |
||||
|
||||
if ($is_remote_call) |
||||
{ |
||||
($shell_output, $shell_error, $shell_return_code) = $anvil->Remote->call($remote_call_parameters); |
||||
} |
||||
else |
||||
{ |
||||
($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call }); |
||||
} |
||||
|
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
shell_call => $shell_call, |
||||
shell_output => $shell_output, |
||||
shell_error => $shell_error, |
||||
shell_return_code => $shell_return_code |
||||
} }); |
||||
} |
||||
} |
||||
} |
||||
|
||||
sub stop_related |
||||
{ |
||||
my $parameters = shift; |
||||
my $ws_pid = $parameters->{ws_pid}; |
||||
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
||||
my $select_pid_field; |
||||
my $condition_pid_field; |
||||
my $condition_pid_value; |
||||
my $stop_function; |
||||
|
||||
if (defined $ws_pid) |
||||
{ |
||||
$select_pid_field = "ssh_tunnel_pid"; |
||||
$condition_pid_field = "ws_pid"; |
||||
$condition_pid_value = $ws_pid; |
||||
$stop_function = \&stop_ssh_tunnel |
||||
} |
||||
elsif (defined $ssh_tunnel_pid) |
||||
{ |
||||
$select_pid_field = "ws_pid"; |
||||
$condition_pid_field = "ssh_tunnel_pid"; |
||||
$condition_pid_value = $ssh_tunnel_pid; |
||||
$stop_function = \&stop_websockify |
||||
} |
||||
|
||||
my $query = " |
||||
SELECT |
||||
hos.host_name, vnc.".$select_pid_field." |
||||
FROM |
||||
public.vnc_pipes AS vnc |
||||
JOIN |
||||
public.hosts AS hos |
||||
ON |
||||
vnc.ssh_tunnel_host_uuid = hos.host_uuid |
||||
WHERE |
||||
vnc.".$condition_pid_field." = ".$anvil->Database->quote($condition_pid_value)." |
||||
;"; |
||||
|
||||
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
|
||||
foreach my $row (@{$results}) |
||||
{ |
||||
$stop_function->({ host_name => $row->[0], ws_pid => $row->[1], ssh_tunnel_pid => $row->[1] }); |
||||
} |
||||
} |
||||
|
||||
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 not null, |
||||
ws_host_uuid uuid not null, |
||||
ws_pid numeric not null, |
||||
ws_source_port numeric not null, |
||||
ssh_tunnel_host_uuid uuid not null, |
||||
ssh_tunnel_pid numeric not null, |
||||
ssh_tunnel_forward_port numeric not null, |
||||
modified_date timestamp with time zone not null |
||||
);"; |
||||
|
||||
$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_vnc_pipe |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $server_vnc_port = $parameters->{server_vnc_port}; |
||||
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
||||
my $ws_pid = $parameters->{ws_pid}; |
||||
my $ws_source_port = $parameters->{ws_source_port}; |
||||
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid}; |
||||
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
||||
my $ssh_tunnel_forward_port = $parameters->{ssh_tunnel_forward_port}; |
||||
|
||||
my $record_uuid = $anvil->Get->uuid(); |
||||
my $record_modified_date = $anvil->Database->refresh_timestamp(); |
||||
|
||||
my $query = " |
||||
INSERT INTO public.vnc_pipes ( |
||||
uuid, |
||||
server_uuid, |
||||
server_vnc_port, |
||||
ws_host_uuid, |
||||
ws_pid, |
||||
ws_source_port, |
||||
ssh_tunnel_host_uuid, |
||||
ssh_tunnel_pid, |
||||
ssh_tunnel_forward_port, |
||||
modified_date |
||||
) VALUES ( |
||||
".$anvil->Database->quote($record_uuid).", |
||||
".$anvil->Database->quote($server_uuid).", |
||||
".$anvil->Database->quote($server_vnc_port).", |
||||
".$anvil->Database->quote($ws_host_uuid).", |
||||
".$anvil->Database->quote($ws_pid).", |
||||
".$anvil->Database->quote($ws_source_port).", |
||||
".$anvil->Database->quote($ssh_tunnel_host_uuid).", |
||||
".$anvil->Database->quote($ssh_tunnel_pid).", |
||||
".$anvil->Database->quote($ssh_tunnel_forward_port).", |
||||
".$anvil->Database->quote($record_modified_date)." |
||||
);"; |
||||
|
||||
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
} |
||||
|
||||
sub update_vnc_pipe |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $server_vnc_port = $parameters->{server_vnc_port}; |
||||
my $ws_host_uuid = $parameters->{ws_host_uuid}; |
||||
my $ws_pid = $parameters->{ws_pid}; |
||||
my $ws_source_port = $parameters->{ws_source_port}; |
||||
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid}; |
||||
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
||||
my $ssh_tunnel_forward_port = $parameters->{ssh_tunnel_forward_port}; |
||||
my $set_string; |
||||
my $condition_string = "server_uuid = ".$anvil->Database->quote($server_uuid); |
||||
|
||||
if ((defined $ws_host_uuid) and (defined $ws_pid) and (defined $ws_source_port)) |
||||
{ |
||||
$set_string = " |
||||
server_vnc_port = ".$anvil->Database->quote($server_vnc_port).", |
||||
ws_host_uuid = ".$anvil->Database->quote($ws_host_uuid).", |
||||
ws_pid = ".$anvil->Database->quote($ws_pid).", |
||||
ws_source_port = ".$anvil->Database->quote($ws_source_port)." |
||||
"; |
||||
} |
||||
elsif ((defined $ssh_tunnel_host_uuid) and (defined $ssh_tunnel_pid) and (defined $ssh_tunnel_forward_port)) |
||||
{ |
||||
$set_string = " |
||||
ssh_tunnel_host_uuid = ".$anvil->Database->quote($ssh_tunnel_host_uuid).", |
||||
ssh_tunnel_pid = ".$anvil->Database->quote($ssh_tunnel_pid).", |
||||
ssh_tunnel_forward_port = ".$anvil->Database->quote($ssh_tunnel_forward_port)." |
||||
"; |
||||
$condition_string = $condition_string." AND ssh_tunnel_host_uuid = ".$anvil->Database->quote($ssh_tunnel_host_uuid); |
||||
} |
||||
|
||||
my $query = " |
||||
UPDATE public.vnc_pipes |
||||
SET ".$set_string." |
||||
WHERE ".$condition_string." |
||||
;"; |
||||
|
||||
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
} |
||||
|
||||
sub get_vnc_pipe |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $host_uuid = $parameters->{host_uuid}; |
||||
my $vnc_pipe_info; |
||||
|
||||
my $query = " |
||||
SELECT |
||||
hos.host_name, vnc.ws_pid, vnc.ssh_tunnel_pid |
||||
FROM |
||||
public.vnc_pipes AS vnc |
||||
JOIN |
||||
public.hosts AS hos |
||||
ON |
||||
vnc.ws_host_uuid = hos.host_uuid |
||||
WHERE |
||||
server_uuid = ".$anvil->Database->quote($server_uuid)." |
||||
AND |
||||
ssh_tunnel_host_uuid = ".$anvil->Database->quote($host_uuid)." |
||||
;"; |
||||
|
||||
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
my $count = @{$results}; |
||||
|
||||
if ($count == 1) |
||||
{ |
||||
my $row = $results->[0]; |
||||
|
||||
$vnc_pipe_info = {}; |
||||
$vnc_pipe_info->{host_name} = $row->[0]; |
||||
$vnc_pipe_info->{ws_pid} = $row->[1]; |
||||
$vnc_pipe_info->{ssh_tunnel_pid} = $row->[2]; |
||||
|
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
host_name => $vnc_pipe_info->{host_name}, |
||||
ws_pid => $vnc_pipe_info->{ws_pid}, |
||||
ssh_tunnel_pid => $vnc_pipe_info->{ssh_tunnel_pid} |
||||
} }); |
||||
} |
||||
|
||||
return $vnc_pipe_info; |
||||
} |
||||
|
||||
sub delete_vnc_pipe |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $host_uuid = $parameters->{host_uuid}; |
||||
my $ws_pid = $parameters->{ws_pid}; |
||||
my $ssh_tunnel_pid = $parameters->{ssh_tunnel_pid}; |
||||
|
||||
my $query = "DELETE FROM public.vnc_pipes "; |
||||
|
||||
if (defined $ws_pid) |
||||
{ |
||||
$query = $query."WHERE ws_pid = ".$anvil->Database->quote($ws_pid).";"; |
||||
} |
||||
elsif (defined $ssh_tunnel_pid) |
||||
{ |
||||
$query = $query."WHERE ssh_tunnel_pid = ".$anvil->Database->quote($ssh_tunnel_pid).";"; |
||||
} |
||||
else |
||||
{ |
||||
$query = $query." |
||||
WHERE |
||||
server_uuid = ".$anvil->Database->quote($server_uuid)." |
||||
AND |
||||
ssh_tunnel_host_uuid = ".$anvil->Database->quote($host_uuid)." |
||||
;"; |
||||
} |
||||
|
||||
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); |
||||
} |
||||
|
||||
sub open_vnc_pipe |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $host_uuid = $anvil->Get->host_uuid(); |
||||
my $vnc_pipe_info; |
||||
|
||||
my $server_info = get_server_info({ server_uuid => $server_uuid }); |
||||
|
||||
if (not defined $server_info) |
||||
{ |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
message => "Failed to get server VM information." |
||||
} }); |
||||
|
||||
return; |
||||
} |
||||
|
||||
my $vnc_info = get_vnc_info($server_info); |
||||
|
||||
if (not defined $vnc_info) |
||||
{ |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
message => "Failed to get server VM VNC information." |
||||
} }); |
||||
|
||||
return; |
||||
} |
||||
|
||||
my $ws_info = start_websockify({ |
||||
server_uuid => $server_uuid, |
||||
host_name => $server_info->{host_name}, |
||||
host_uuid => $server_info->{host_uuid}, |
||||
target_port => $vnc_info->{port} |
||||
}); |
||||
|
||||
if (not defined $ws_info) |
||||
{ |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
message => "Failed to get websockify instance information." |
||||
} }); |
||||
|
||||
return; |
||||
} |
||||
|
||||
my $ssh_tunnel_info = start_ssh_tunnel({ |
||||
server_uuid => $server_uuid, |
||||
host_uuid => $host_uuid, |
||||
ws_host_name => $server_info->{host_name}, |
||||
ws_host_uuid => $server_info->{host_uuid}, |
||||
ws_source_port => $ws_info->{source_port} |
||||
}); |
||||
|
||||
if (not defined $ssh_tunnel_info) |
||||
{ |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
message => "Failed to get SSH tunnel instance information." |
||||
} }); |
||||
|
||||
return; |
||||
} |
||||
|
||||
if ($ws_info->{is_new} or $ssh_tunnel_info->{is_new}) |
||||
{ |
||||
insert_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_source_port => $ws_info->{source_port}, |
||||
ssh_tunnel_host_uuid => $host_uuid, |
||||
ssh_tunnel_pid => $ssh_tunnel_info->{pid}, |
||||
ssh_tunnel_forward_port => $ssh_tunnel_info->{forward_port} |
||||
}); |
||||
} |
||||
else |
||||
{ |
||||
if ($ws_info->{is_update}) |
||||
{ |
||||
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_source_port => $ws_info->{source_port} |
||||
}); |
||||
} |
||||
|
||||
if ($ssh_tunnel_info->{is_update}) |
||||
{ |
||||
update_vnc_pipe({ |
||||
server_uuid => $server_uuid, |
||||
ssh_tunnel_host_uuid => $host_uuid, |
||||
ssh_tunnel_pid => $ssh_tunnel_info->{pid}, |
||||
ssh_tunnel_forward_port => $ssh_tunnel_info->{forward_port} |
||||
}); |
||||
} |
||||
} |
||||
|
||||
$vnc_pipe_info = { forward_port => $ssh_tunnel_info->{forward_port} }; |
||||
|
||||
return $vnc_pipe_info; |
||||
} |
||||
|
||||
sub close_vnc_pipe |
||||
{ |
||||
my $parameters = shift; |
||||
my $server_uuid = $parameters->{server_uuid}; |
||||
my $vnc_pipe_parameters = { server_uuid => $server_uuid, host_uuid => $anvil->Get->host_uuid() }; |
||||
my $vnc_pipe_info = get_vnc_pipe($vnc_pipe_parameters); |
||||
|
||||
if (not defined $vnc_pipe_info) |
||||
{ |
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
message => "Failed to get VNC pipe information." |
||||
} }); |
||||
|
||||
return; |
||||
} |
||||
|
||||
if (not 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} }); |
||||
} |
||||
|
||||
stop_ssh_tunnel({ ssh_tunnel_pid => $vnc_pipe_info->{ssh_tunnel_pid} }); |
||||
|
||||
delete_vnc_pipe($vnc_pipe_parameters); |
||||
} |
||||
|
||||
$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 }); |
||||
} |
||||
|
||||
$anvil->Database->get_hosts(); |
||||
$anvil->Database->get_anvils(); |
||||
|
||||
my $server_uuid = $anvil->data->{switches}{'server-uuid'}; |
||||
my $is_open = $anvil->data->{switches}{'open'}; |
||||
my $is_drop_table = $anvil->data->{switches}{'drop-table'}; |
||||
|
||||
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { |
||||
server_uuid => $server_uuid, |
||||
is_open => $is_open, |
||||
is_drop_table => $is_drop_table |
||||
} }); |
||||
|
||||
if ($server_uuid) |
||||
{ |
||||
create_vnc_pipes_table(); |
||||
|
||||
my $vnc_pipe_parameters = { server_uuid => $server_uuid }; |
||||
|
||||
if ($is_open) |
||||
{ |
||||
my $vnc_pipe_info = open_vnc_pipe($vnc_pipe_parameters); |
||||
|
||||
print "protocol:ws,forward_port:".$vnc_pipe_info->{forward_port}."\n"; |
||||
} |
||||
else |
||||
{ |
||||
close_vnc_pipe($vnc_pipe_parameters); |
||||
} |
||||
} |
||||
elsif ($is_drop_table) |
||||
{ |
||||
drop_vnc_pipes_table(); |
||||
} |
Loading…
Reference in new issue