Group
Extension

Rex-WebUI/lib/Rex/WebUI/Task.pm

package Rex::WebUI::Task;

use Mojo::Base 'Mojolicious::Controller';

use Data::Dumper;

our $TEST_DELAY_AFTER_RUN_TASK = 0;

# This action will render a template
sub view
{
	my $self = shift;

	my $id        = $self->param("id");
	my $task_name = $self->param("name");

	$self->app->log->debug("View task: $task_name");

	my $project = $self->config->{projects}->[$id];
	$self->rex->load_rexfile($project->{rexfile});

	$self->app->log->debug("Load task: $task_name");

	my $task = $self->rex->get_task($task_name);

	my $ws_url = $self->url_for('tail_ws')->to_abs;

	$self->stash(task => $task);
	$self->stash(tail_ws_url => $ws_url);

	$self->render;
}

# run a rex task in a conventional HTTP GET / POST
sub run
{
	my $self = shift;

	my $id        = $self->param("id");
	my $task_name = $self->param("name");

	my $project = $self->config->{projects}->[$id];
	$self->rex->load_rexfile($project->{rexfile});

	$self->app->log->debug("Load task: $task_name");

	my $task = $self->rex->get_task($task_name);

	unless ($task) {

		$self->stash(task => "Not Found: $task");
		$self->render;
	}

	my $servers = $task->{server};

	$servers = [{ name => '<local>'}] if scalar @$servers == 0;


	my $jobid = $self->logbook->add({
		task_name => $task_name,
		server    => $servers,
		userid    => 1,
	});

	$self->app->log->debug("Got jobid: $jobid");

	$self->render(json => { jobid => $jobid, status => "starting task: $task_name" });

	$self->app->log->debug("After render");

	my $temp_logfile     = "/tmp/rex_" . $jobid . '.log';
	my $temp_status_file = "/tmp/rex_" . $jobid . '.status';

	$self->_set_status($temp_status_file, 'init');

	# I wish we didn't have to fork here, if there's a better way please tell me

	# after forking parent thread can write to websocket, but child can't
	if (my $pid = $self->_fork_process()) {

		# parent thread
		$self->app->log->debug("Parent Thread - return response to browser");

		$self->logbook->set_pid($jobid, $pid);

		return;
	}

	$self->app->log->debug("calling rex: $task_name");

	$self->_set_status($temp_status_file, 'running');

	$self->logbook->update_status($jobid, 1);

	foreach my $server (@$servers) {

		my $server_name = $server->{name};

		$self->app->log->debug("running task: $task_name on server: $server_name");
		Rex::Logger::info("running task: $task_name on server: $server_name");

		my $result = $self->rex->do_run_task($task_name, $server_name, $temp_logfile);
	}

	sleep $TEST_DELAY_AFTER_RUN_TASK if $TEST_DELAY_AFTER_RUN_TASK;

	$self->app->log->debug("finished running task");

	$self->_set_status($temp_status_file, "done");

	$self->logbook->update_status($jobid, 2);

	exit(0);
}

# run a rex task in a websocket, sending back the log messages as we go
sub tail_ws
{
	my $self = shift;

	my $id    = $self->param("id"); # project id not really required here at the moment
	my $jobid = $self->param("jobid");

	$self->app->log->debug("Tail ws - jobid: $jobid");

	my $temp_logfile     = "/tmp/rex_" . $jobid . '.log';
	my $temp_status_file = "/tmp/rex_" . $jobid . '.status';

	$self->res->headers->content_type('text/event-stream');

	# give a little
	foreach my $i (1..5) {
		unless (-f $temp_logfile) {
			warn "Not Found: $temp_logfile - give a little";
			sleep 1;
		}
	}

	unless (-f $temp_logfile) {

		$self->send("ERROR: File Not Found: $temp_logfile");
		return;
	}

	Mojo::IOLoop->stream($self->tx->connection)->timeout(300);

	my $i = 0;
	my $log_position = 0;
	my $cb;

	$cb = sub {
		sleep 1;
		$i++;
		my $status = $self->_get_status($temp_status_file);

		my ($log_lines, $new_log_position) = $self->_read_log($temp_logfile, $log_position);

		$log_position = $new_log_position;

		foreach my $log_line (@$log_lines) {

			$_[0]->send($log_line);
		}

		if ($status =~ /^done/) {
			$_[0]->send("STATUS: $status [$i]");
			unlink $temp_logfile;
			unlink $temp_status_file;
		} else
		{
			$_[0]->send("STATUS: $status [$i]", $cb);
		}
	};

	$self->$cb;

	return;
}

sub _fork_process
{
	my $self = shift;

	# Block signals whilst we fork the new child process
	$SIG{CHLD} = 'IGNORE';

	my $pid;

	# Fork the new child process
	if (!defined($pid = fork)) {

		$self->app->log->debug("Fork: $!");
		die("Fork: $!");
	}

	# Am I the parent or the child?
	if ($pid) {
		# Ensure we return to caller

		$self->app->log->debug("Child - return pid: $pid");

		return $pid;
	}

	# I am the child - as I can't return from here - make sure sig(INT) kills me
	$SIG{INT} = 'DEFAULT';

	$self->app->log->debug("Parent lives");

	return undef;
}

sub _set_status
{
	my ($self, $temp_status_file, $status) = @_;

	open FILE, ">", $temp_status_file;
	print FILE $status;
	close FILE;

	#warn "WROTE: $temp_status_file, $status";
}

sub _get_status
{
	my ($self, $temp_status_file) = @_;

	open FILE, "<", $temp_status_file;

	my @lines = <FILE>;

	my $status = $lines[0];

	#warn "READ: $temp_status_file, $status";

	return $status;
}

sub _read_log
{
	my ($self, $logfile, $log_position) = @_;

	my $log_lines = [];
	my $total_lines = 0;

	# this is a very, very crude way to tail the log, but it will do fine for small log files

	if (-f $logfile) {
		open FILE, "<", $logfile;

		my @lines = <FILE>;

		$total_lines = scalar @lines;

		if ($total_lines >= $log_position) {

			foreach my $i ($log_position .. $total_lines) {

				push @$log_lines, $lines[$i-1];
			}
		}
		$total_lines++;
	}

	return ($log_lines, $total_lines);
}

1;


Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.