adding debug to spectre

This commit is contained in:
JT Smith 2006-03-16 23:32:18 +00:00
parent baf9a200af
commit 71b2634741
5 changed files with 215 additions and 51 deletions

View file

@ -29,16 +29,16 @@ Initializes the workflow manager.
=cut
sub _start {
print "Starting WebGUI Spectre Workflow Manager...";
my ( $kernel, $self, $publicEvents) = @_[ KERNEL, OBJECT, ARG0 ];
$self->debug("Starting workflow manager.");
my $serviceName = "workflow";
$kernel->alias_set($serviceName);
$kernel->call( IKC => publish => $serviceName, $publicEvents );
my $configs = WebGUI::Config->readAllConfigs($self->{_config}->getWebguiRoot);
$self->debug("Reading workflow configs.");
my $configs = WebGUI::Config->readAllConfigs($self->config->getWebguiRoot);
foreach my $config (keys %{$configs}) {
$kernel->yield("loadWorkflows", $config);
}
print "OK\n";
$kernel->yield("checkJobs");
}
@ -51,10 +51,9 @@ Gracefully shuts down the workflow manager.
=cut
sub _stop {
my ($kernel, $self) = @_[KERNEL, OBJECT];
print "Stopping WebGUI Spectre Workflow Manager...";
my ($kernel, $self) = @_[KERNEL, OBJECT];
$self->debug("Stopping workflow manager.");
undef $self;
print "OK\n";
}
@ -77,6 +76,7 @@ A hash reference containing a row of data from the WorkflowInstance table.
sub addJob {
my ($self, $config, $job) = @_[OBJECT, ARG0, ARG1];
$self->debug->("Adding workflow instance ".$job->{instanceId}." from ".$config." to job queue at priority ".$job->{priority}.".");
# job list
$self->{_jobs}{$job->{instanceId}} = {
instanceId=>$job->{instanceId},
@ -97,7 +97,8 @@ Checks to see if there are any open job slots available, and if there are assign
sub checkJobs {
my ($kernel, $self) = @_[KERNEL, OBJECT];
if ($self->countRunningJobs < $self->{_config}->get("maxWorkers")) {
$self->debug("Checking to see if we can run anymore jobs right now.");
if ($self->countRunningJobs < $self->config->get("maxWorkers")) {
my $job = $self->getNextJob;
if (defined $job) {
$job->{status} = "running";
@ -109,6 +110,19 @@ sub checkJobs {
#-------------------------------------------------------------------
=head2 config
Returns a reference to the config object.
=cut
sub config {
my $self = shift;
return $self->{_config};
}
#-------------------------------------------------------------------
=head2 countRunningJobs ( )
Returns an integer representing the number of running jobs.
@ -118,7 +132,27 @@ Returns an integer representing the number of running jobs.
sub countRunningJobs {
my $self = shift;
my $runningJobs = $self->{_runningJobs} || [];
return scalar(@{$runningJobs});
my $jobCount = scalar(@{$runningJobs});
$self->debug("There are $jobCount running jobs.");
return $jobCount;
}
#-------------------------------------------------------------------
=head2 debug ( output )
Prints out debug information if debug is enabled.
=head3
=cut
sub debug {
my $self = shift;
my $output = shift;
if ($self->{_debug}) {
print "WORKFLOW: ".$output."\n";
}
}
#-------------------------------------------------------------------
@ -131,6 +165,7 @@ Removes a workflow job from the processing queue.
sub deleteJob {
my ($self, $instanceId) = @_[OBJECT, ARG0];
$self->debug("Deleting workflow instance $instanceId from job queue.");
my $priority = $self->{_jobs}{$instanceId}{priority};
delete $self->{_jobs}{$instanceId};
for (my $i=0; $i < scalar(@{$self->{"_priority".$priority}}); $i++) {
@ -148,13 +183,16 @@ sub deleteJob {
sub getNextJob {
my $self = shift;
$self->debug("Looking for a workflow instance to execute.");
foreach my $priority (1..3) {
foreach my $job (@{$self->{"_priority".$priority}}) {
if ($job->{status} eq "waiting") {
$self->debug("Looks like ".$job->{instanceId}." would be a good workflow instance to run.");
return $job;
}
}
}
$self->debug("Didn't see any workflow instances to run.");
return undef;
}
@ -166,7 +204,8 @@ sub getNextJob {
sub loadWorkflows {
my ($kernel, $self, $config) = @_[KERNEL, OBJECT, ARG0];
my $session = WebGUI::Session->open($self->{_config}->getWebguiRoot, $config);
$self->debug("Loading workflows for ".$config.".");
my $session = WebGUI::Session->open($self->config->getWebguiRoot, $config);
my $result = $session->db->read("select * from WorkflowInstance");
while (my $data = $result->hashRef) {
$kernel->yield("addJob", $config, $data);
@ -176,7 +215,7 @@ sub loadWorkflows {
#-------------------------------------------------------------------
=head2 new ( config )
=head2 new ( config [ , debug ] )
Constructor. Loads all active workflows from each WebGUI site and begins executing them.
@ -184,12 +223,17 @@ Constructor. Loads all active workflows from each WebGUI site and begins executi
The path to the root of the WebGUI installation.
=head3 debug
A boolean indicating Spectre should spew forth debug as it runs.
=cut
sub new {
my $class = shift;
my $config = shift;
my $self = {_config=>$config};
my $debug = shift;
my $self = {_debug=>$debug, _config=>$config};
bless $self, $class;
my @publicEvents = qw(addJob deleteJob);
POE::Session->create(
@ -208,6 +252,7 @@ Calls a worker to execute a workflow activity.
sub runWorker {
my ($kernel, $self, $job, $session) = @_[KERNEL, OBJECT, ARG0, SESSION];
$self->debug("Preparing to run workflow instance ".$job->{instanceId}.".");
POE::Component::Client::UserAgent->new;
my $url = $job->{sitename}.'/'.$job->{gateway};
$url =~ s/\/\//\//g;
@ -216,18 +261,20 @@ sub runWorker {
'do'=>'runWorkflow',
instanceId=>$job->{instanceId},
};
my $cipher = Crypt::Blowfish->new($self->{_config}->get("cryptoKey"));
my $cipher = Crypt::Blowfish->new($self->config->get("cryptoKey"));
my $request = HTTP::Request->new(POST => $url, Content => { op=>"spectre", payload=>$cipher->encrypt(objToJson($payload)) });
my $cookie = $self->{_cookies}{$job->{sitename}};
$request->header("Cookie","wgSession=".$cookie) if (defined $cookie);
$request->header("User-Agent","Spectre");
$request->header("X-JobId",$job->{instanceId});
$self->debug("Posting workflow instance ".$job->{instanceId}.".");
$kernel->post( useragent => 'request', { request => $request, response => $session->postback('workerResponse') });
$self->debug("Workflow instance ".$job->{instanceId}." posted.");
}
#-------------------------------------------------------------------
=head2 suspendJob ( jobId ) {
=head2 suspendJob ( jobId )
This method puts a running job back into the available jobs pool thusly freeing up a slot in the running jobs pool. This is done when a job has executed a workflow activity, but the entire workflow has not yet completed.
@ -240,6 +287,7 @@ The job being suspended.
sub suspendJob {
my $self = shift;
my $instanceId = shift;
$self->debug("Suspending workflow instance ".$instanceId.".");
$self->{_jobs}{$instanceId}{status} = "waiting";
for (my $i=0; $i < scalar(@{$self->{_runningJobs}}); $i++) {
if ($self->{_runningJobs}[$i]{instanceId} eq $instanceId) {
@ -258,28 +306,44 @@ This method is called when the response from the runWorker() method is received.
sub workerResponse {
my $self = $_[OBJECT];
$self->debug("Retrieving response from workflow instance job.");
my ($request, $response, $entry) = @{$_[ARG1]};
my $jobId = $request->header("X-JobId"); # got to figure out how to get this from the request, cuz the response may die
$self->debug("Response retrieved is for $jobId.");
if ($response->is_success) {
$self->debug("Response for $jobId retrieved successfully.");
if ($response->header("Cookie") ne "") {
$self->debug("Storing cookie for $jobId for later use.");
my $cookie = $response->header("Set-Cookie");
$cookie =~ s/wgSession=([a-zA-Z0-9\_\-]{22})/$1/;
$self->{_cookies}{$self->{_jobs}{$jobId}{sitename}} = $cookie;
}
my $cipher = Crypt::Blowfish->new($self->{_config}->get("cryptoKey"));
my $cipher = Crypt::Blowfish->new($self->config->get("cryptoKey"));
my $payload = jsonToObj($cipher->decrypt($response->content));
my $state = $payload->{state};
if ($state eq "continue") {
$self->suspendJob($jobId);
} elsif ($state eq "done") {
if ($state eq "waiting") {
$self->debug("Was told to wait on $jobId because we're still waiting on some external event.");
$self->suspendJob($jobId, $self->config->get("waitTime"));
} elsif ($state eq "complete") {
$self->debug("Workflow instance $jobId ran one of it's activities successfully.");
} elsif ($state eq "disabled") {
$self->debug("Workflow instance $jobId is disabled.");
$self->deleteJob($jobId);
} elsif ($state eq "done") {
$self->debug("Workflow instance $jobId is now complete.");
$self->deleteJob($jobId);
} elsif ($state eq "error") {
$self->debug("Got an error for $jobId.");
$self->suspendJob($jobId, $self->config->get("waitTime"));
} else {
$self->suspendJob($jobId);
$self->debug("Something bad happened on the return of $jobId, so we're suspending it's run for a while.");
$self->suspendJob($jobId, $self->config->get("waitTime"));
# something bad happened
}
} elsif ($response->is_redirect) {
# nothing to do, cuz we're following the redirect to see what happens
$self->debug("Response for $jobId was redirected.");
} elsif ($response->is_error) {
$self->debug("Response for $jobId had a communications error.");
$self->suspendJob($jobId)
# we should probably log something
}