fix(cgi-bin): draft logic for websockify setup and clean up

main
Tsu-ba-me 4 years ago
parent 39ecca7eaf
commit df33c2d193
  1. 165
      cgi-bin/manage_vnc_ports

@ -31,11 +31,12 @@ sub get_server_info
my $query = " my $query = "
SELECT SELECT
ser.server_name, ser.server_name,
hos.host_name hos.host_name,
hos.host_uuid
FROM FROM
servers AS ser public.servers AS ser
JOIN JOIN
hosts AS hos public.hosts AS hos
ON ON
ser.server_host_uuid = hos.host_uuid ser.server_host_uuid = hos.host_uuid
WHERE WHERE
@ -49,14 +50,17 @@ WHERE
my $row = $results->[0]; my $row = $results->[0];
my $server_name = $row->[0]; my $server_name = $row->[0];
my $host_name = $row->[1]; my $host_name = $row->[1];
my $host_uuid = $row->[2];
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { $anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => {
server_name => $server_name, server_name => $server_name,
host_name => $host_name host_name => $host_name,
host_uuid => $host_uuid
} }); } });
$info_hash->{server_name} = $server_name; $info_hash->{server_name} = $server_name;
$info_hash->{host_name} = $host_name; $info_hash->{host_name} = $host_name;
$info_hash->{host_uuid} = $host_uuid;
} }
return $info_hash; return $info_hash;
@ -69,10 +73,10 @@ sub get_vnc_info
my $server_name = $parameters->{server_name}; my $server_name = $parameters->{server_name};
my $port_base = 5900; my $port_base = 5900;
my $vnc_info = { host_name => $host_name }; my $vnc_info = { host_name => $host_name };
my $shell_call = "ssh root@".$host_name." \"virsh vncdisplay ".$server_name."\""; # Requires root to access VM information.
my $shell_call = "ssh -n root@".$host_name." \"virsh vncdisplay ".$server_name."\"";
my ($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call }); my ($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call });
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { $anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => {
shell_output => $shell_output, shell_output => $shell_output,
shell_return_code => $shell_return_code shell_return_code => $shell_return_code
@ -92,16 +96,126 @@ sub get_vnc_info
return $vnc_info; 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 = "ssh -n ".$host_name." \"ps -o comm -h -p ".$ws_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_output => $shell_output,
shell_return_code => $shell_return_code
} });
return $shell_output eq "websockify" ? 1 : 0;
}
sub is_websockify_exists
{
my $parameters = shift;
my $server_uuid = $parameters->{server_uuid};
my $server_vnc_port = $parameters->{server_vnc_port};
my $query = "
SELECT
vnc.server_vnc_port, hos.host_name, vnc.ws_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)."
;";
my $results = $anvil->Database->query({ query => $query, source => $THIS_FILE, line => __LINE__ });
my $count = @{$results};
my $is_exists = 0;
if ($count > 1)
{
my $row = $results->[0];
my $server_vnc_port_in_record = $row->[0];
my $host_name = $row->[1];
my $ws_pid = $row->[2];
my $clean_up_parameters = { host_name => $host_name, ws_pid => $ws_pid };
if ($server_vnc_port != $server_vnc_port_in_record)
{
# VNC server port mismatch/oudated; require clean up.
stop_websockify($clean_up_parameters);
return $is_exists;
}
if (not is_websockify_process($clean_up_parameters))
{
# Process died; require clean up.
return $is_exists;
}
# Passed all tests; process considered exists.
$is_exists = 1;
}
return $is_exists;
}
sub start_websockify sub start_websockify
{ {
my $parameters = shift; my $parameters = shift;
my $server_uuid = $parameters->{server_uuid};
my $host_name = $parameters->{host_name}; my $host_name = $parameters->{host_name};
my $target_port = $parameters->{target_port}; my $target_port = $parameters->{target_port};
my $ws_info;
if (not is_websockify_exists({ server_uuid => $server_uuid, server_vnc_port => $target_port }))
{
my $source_port_base = 10000; my $source_port_base = 10000;
my $source_port = $source_port_base + $target_port; my $source_port = $source_port_base + $target_port;
my $shell_call = "ssh root@".$host_name." \"websockify ".$source_port." :".$target_port." & echo \$!\""; my $shell_call = "ssh -n ".$host_name." 'websockify ".$source_port." :".$target_port." & 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_output => $shell_output,
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;
$anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, list => { ws_pid => $ws_pid } });
}
}
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 = "ssh -n ".$host_name." \"kill -9 ".$ws_pid."\"";
my ($shell_output, $shell_return_code) = $anvil->System->call({ shell_call => $shell_call }); 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_output => $shell_output,
shell_return_code => $shell_return_code
} });
}
} }
sub create_vnc_pipes_table sub create_vnc_pipes_table
@ -110,10 +224,11 @@ sub create_vnc_pipes_table
CREATE TABLE IF NOT EXISTS public.vnc_pipes ( CREATE TABLE IF NOT EXISTS public.vnc_pipes (
uuid uuid not null primary key, uuid uuid not null primary key,
server_uuid uuid not null, server_uuid uuid not null,
server_vnc_port numeric not null,
ws_host_uuid uuid not null, ws_host_uuid uuid not null,
ws_pid uuid not null, ws_pid numeric not null,
ssh_tunnel_host_uuid uuid not null, ssh_tunnel_host_uuid uuid not null,
ssh_tunnel_pid uuid not null, ssh_tunnel_pid numeric not null,
modified_date timestamp with time zone not null modified_date timestamp with time zone not null
);"; );";
@ -131,6 +246,7 @@ sub insert_vnc_pipe
{ {
my $parameters = shift; my $parameters = shift;
my $server_uuid = $parameters->{server_uuid}; my $server_uuid = $parameters->{server_uuid};
my $server_vnc_port = $parameters->{server_vnc_port};
my $ws_host_uuid = $parameters->{ws_host_uuid}; my $ws_host_uuid = $parameters->{ws_host_uuid};
my $ws_pid = $parameters->{ws_pid}; my $ws_pid = $parameters->{ws_pid};
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid}; my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid};
@ -138,7 +254,7 @@ sub insert_vnc_pipe
my $query = " my $query = "
INSERT INTO public.vnc_pipes ( INSERT INTO public.vnc_pipes (
server_uuid, ws_host_uuid, ws_pid, ssh_tunnel_host_uuid, ssh_tunnel_pid server_uuid, server_vnc_port, ws_host_uuid, ws_pid, ssh_tunnel_host_uuid, ssh_tunnel_pid
) VALUES ( ) VALUES (
".$anvil->Database->quote($server_uuid).", ".$anvil->Database->quote($server_uuid).",
".$anvil->Database->quote($ws_host_uuid).", ".$anvil->Database->quote($ws_host_uuid).",
@ -150,6 +266,24 @@ INSERT INTO public.vnc_pipes (
$anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ }); $anvil->Database->write({ query => $query, source => $THIS_FILE, line => __LINE__ });
} }
sub get_vnc_pipe
{
my $parameters = shift;
my $server_uuid = $parameters->{server_uuid};
my $ssh_tunnel_host_uuid = $parameters->{ssh_tunnel_host_uuid};
my $query = "
SELECT
ws_host_uuid, ws_pid, ssh_tunnel_pid
FROM
public.vnc_pipes
WHERE
server_uuid = ".$anvil->Database->quote($server_uuid)."
AND
ssh_tunnel_host_uuid = ".$anvil->Database->quote($ssh_tunnel_host_uuid)."
;";
}
sub delete_vnc_pipe sub delete_vnc_pipe
{ {
my $parameters = shift; my $parameters = shift;
@ -218,11 +352,20 @@ $anvil->Log->variables({ source => $THIS_FILE, line => __LINE__, level => 2, lis
} }); } });
if ($server_uuid) if ($server_uuid)
{
if ($is_open)
{ {
my $server_info = get_server_info({ server_uuid => $server_uuid }); my $server_info = get_server_info({ server_uuid => $server_uuid });
my $vnc_port = get_vnc_info($server_info); my $vnc_info = get_vnc_info($server_info);
create_vnc_pipes_table(); create_vnc_pipes_table();
my $ws_info = start_websockify({ server_uuid => $server_uuid, host_name => $server_info->{host_name}, target_port => $vnc_info->{port} });
}
else
{
# TODO: fill-in clean up section.
}
} }
elsif ($anvil->data->{switches}{'drop-table'}) elsif ($anvil->data->{switches}{'drop-table'})
{ {

Loading…
Cancel
Save