Workflow - Simple, flexible system to implement workflows |
get_current_actions()
get_history()
get_unsaved_history()
clear_history()
Workflow - Simple, flexible system to implement workflows
use Workflow::Factory qw( FACTORY ); # Defines a workflow of type 'myworkflow' my $workflow_conf = 'workflow.xml'; # contents of 'workflow.xml' <workflow> <type>myworkflow</type> <state name="INITIAL"> <action name="upload file" resulting_state="uploaded" /> </state> <state name="uploaded" autorun="yes"> <action name="verify file" resulting_state="verified file"> <!-- everyone other than 'CWINTERS' must verify --> <condition test="$context->{user} ne 'CWINTERS'" /> </action> <action name="null" resulting_state="annotated"> <condition test="$context->{user} eq 'CWINTERS'" /> </action> </state> <state name="verified file"> <action name="annotate"> <condition name="can_annotate" /> </action> <action name="null"> <condition name="!can_annotate" /> </action> </state> <state name="annotated" autorun="yes" may_stop="yes"> <action name="null" resulting_state="finished"> <condition name="completed" /> </action> </state> <state name="finished" /> </workflow> # Defines actions available to the workflow my $action_conf = 'action.xml'; # contents of 'action.xml' <actions> <action name="upload file" class="MyApp::Action::Upload"> <field name="path" label="File Path" description="Path to file" is_required="yes" /> </action>
<action name="verify file" class="MyApp::Action::Verify"> <validator name="filesize_cap"> <arg>$file_size</arg> </validator> </action>
<action name="annotate" class="MyApp::Action::Annotate" />
<action name="null" class="Workflow::Action::Null" /> </actions> # Defines conditions available to the workflow my $condition_conf = 'condition.xml'; # contents of 'condition.xml' <conditions> <condition name="can_annotate" class="MyApp::Condition::CanAnnotate" /> </conditions>
# Defines validators available to the actions my $validator_conf = 'validator.xml'; # contents of 'validator.xml' <validators> <validator name="filesize_cap" class="MyApp::Validator::FileSizeCap"> <param name="max_size" value="20M" /> </validator> </validators>
# Stock the factory with the configurations; we can add more later if # we want FACTORY->add_config_from_file( workflow => $workflow_conf, action => $action_conf, condition => $condition_conf, validator => $validator_conf ); # Instantiate a new workflow... my $workflow = FACTORY->create_workflow( 'myworkflow' ); print "Workflow ", $workflow->id, " ", "currently at state ", $workflow->state, "\n"; # Display available actions... print "Available actions: ", $workflow->get_current_actions, "\n"; # Get the data needed for action 'upload file' (assumed to be # available in the current state) and display the fieldname and # description print "Action 'upload file' requires the following fields:\n"; foreach my $field ( $workflow->get_action_fields( 'FOO' ) ) { print $field->name, ": ", $field->description, "(Required? ", $field->is_required, ")\n"; } # Add data to the workflow context for the validators, conditions and # actions to work with my $context = $workflow->context; $context->param( current_user => $user ); $context->param( sections => \@sections ); $context->param( path => $path_to_file ); # Execute one of them $workflow->execute_action( 'upload file' ); print "New state: ", $workflow->state, "\n"; # Later.... fetch an existing workflow my $id = get_workflow_id_from_user( ... ); my $workflow = FACTORY->fetch_workflow( 'myworkflow', $id ); print "Current state: ", $workflow->state, "\n";
This documentation is for version '0.15' of the Workflow module.
This is a standalone workflow system. It is designed to fit into your system rather than force your system to fit to it. You can save workflow information to a database or the filesystem (or a custom storage). The different components of a workflow system can be included separately as libraries to allow for maximum reusibility.
As a user you only see two components, plus a third which is really embedded into another:
The workflow system has four basic components:
This is represented by the Workflow object. You normally do not need to subclass this object for customization.
action - The action is defined by you or in a separate library. The action is triggered by moving from one state to another and has access to the workflow and more importantly its context.The base class for actions is the the Workflow::Action manpage class.
condition - Within the workflow you can attach one or more conditions to an action. These ensure that actions only get executed when certain conditions are met. Conditions are completely arbitrary: typically they will ensure the user has particular access rights, but you can also specify that an action can only be executed at certain times of the day, or from certain IP addresses, and so forth. Each condition is created once at startup then passed a context to check every time an action is checked to see if it can be executed.The base class for conditions is the the Workflow::Condition manpage class.
validator - An action can specify one or more validators to ensure that the data available to the action is correct. The data to check can be as simple or complicated as you like. Each validator is created once then passed a context and data to check every time an action is executed.The base class for validators is the the Workflow::Validator manpage class.
A workflow is just a bunch of states with rules on how to move between them. These are known as transitions and are triggered by some sort of event. A state is just a description of object properties. You can describe a surprisingly large number of processes as a series of states and actions to move between them. The application shipped with this distribution uses a fairly common application to illustrate: the trouble ticket.
When you create a workflow you have one action available to you: create a new ticket ('create issue'). The workflow has a state 'INITIAL' when it is first created, but this is just a bootstrapping exercise since the workflow must always be in some state.
The workflow action 'create issue' has a property 'resulting_state', which just means: if you execute me properly the workflow will be in the new state 'CREATED'.
All this talk of 'states' and 'transitions' can be confusing, but just match them to what happens in real life -- you move from one action to another and at each step ask: what happens next?
You create a trouble ticket: what happens next? Anyone can add comments to it and attach files to it while administrators can edit it and developers can start working on it. Adding comments does not really change what the ticket is, it just adds information. Attachments are the same, as is the admin editing the ticket.
But when someone starts work on the ticket, that is a different matter. When someone starts work they change the answer to: what happens next? Whenever the answer to that question changes, that means the workflow has changed state.
In addition to declaring what the resulting state will be from an action the action also has a number of 'field' properties that describe that data it required to properly execute it.
This is an example of discoverability. This workflow system is setup so you can ask it what you can do next as well as what is required to move on. So to use our ticket example we can do this, creating the workflow and asking it what actions we can execute right now:
my $wf = Workflow::Factory->create_workflow( 'Ticket' ); my @actions = $wf->get_current_actions;
We can also interrogate the workflow about what fields are necessary to execute a particular action:
print "To execute the action 'create issue' you must provide:\n\n"; my @fields = $wf->get_action_fields( 'create issue' ); foreach my $field ( @fields ) { print $field->name, " (Required? ", $field->is_required, ")\n", $field->description, "\n\n"; }
To allow the workflow to run into multiple environments we must have a common way to move data between your application, the workflow and the code that moves it from one state to another.
Whenever the the Workflow::Factory manpage creates a new workflow it associates the workflow with a the Workflow::Context manpage object. The context is what moves the data from your application to the workflow and the workflow actions.
For instance, the workflow has no idea what the 'current user' is. Not only is it unaware from an application standpoint but it does not presume to know where to get this information. So you need to tell it, and you do so through the context.
The fact that the workflow system proscribes very little means it can be used in lots of different applications and interfaces. If a system is too closely tied to an interface (like the web) then you have to create some potentially ugly hacks to create a more convenient avenue for input to your system (such as an e-mail approving a document).
The the Workflow::Context manpage object is extremely simple to use -- you ask a workflow for its context and just get/set parameters on it:
# Get the username from the Apache object my $username = $r->connection->user; # ...set it in the context $wf->context->param( user => $username ); # somewhere else you'll need the username: $news_object->{created_by} = $wf->context->param( 'user' );
A typical process for executing an action is:
When you execute the action a number of checks occur. The action needs to ensure:
Once the action passes these checks and successfully executes we update the permanent workflow storage with the new state, as long as the application has declared it.
It's useful to have your workflow generate events so that other parts of a system can see what's going on and react. For instance, say you have a new user creation process. You want to email the records of all users who have a first name of 'Sinead' because you're looking for your long-lost sister named 'Sinead'. You'd create an observer class like:
package FindSinead; sub update { my ( $class, $wf, $event, $new_state ) = @_; return unless ( $event eq 'state change' ); return unless ( $new_state eq 'CREATED' ); my $context = $wf->context; return unless ( $context->param( 'first_name' ) eq 'Sinead' );
my $user = $context->param( 'user' ); my $username = $user->username; my $email = $user->email; my $mailer = get_mailer( ... ); $mailer->send( 'foo@bar.com','Found her!', "We found Sinead under '$username' at '$email' ); }
And then associate it with your workflow:
<workflow> <type>SomeFlow</type> <observer class="FindSinead" /> ...
Every time you create/fetch a workflow the associated observers are attached to it.
You can attach listeners to workflows and catch events at a few points in the workflow lifecycle; these are the events fired:
No additional parameters.
fetch - Issued after a workflow is fetched from the persister.No additional parameters.
save - Issued after a workflow is successfully saved.No additional parameters.
execute - Issued after a workflow is successfully executed and saved.Adds the parameters $old_state
, $action_name
and $autorun
.
$old_state
includes the state of the workflow before the action
was executed, $action_name
is the action name that was executed and
$autorun
is set to 1 if the action just executed was started
using autorun.
Adds the parameters $old_state
, $action
and $autorun
.
$old_state
includes the state of the workflow before the action
was executed, $action
is the action name that was executed and
$autorun
is set to 1 if the action just executed was autorun.
The additional argument is an arrayref of all the Workflow::History manpage objects added to the workflow. (Note that these will not be persisted until the workflow is persisted.)
You configure the observers directly in the 'workflow' configuration item. Each 'observer' may have either a 'class' or 'sub' entry within it that defines the observer's location.
We load these classes at startup time. So if you specify an observer that doesn't exist you see the error when the workflow system is initialized rather than the system tries to use the observer.
For instance, the following defines two observers:
<workflow> <type>ObservedItem</type> <description>This is...</description> <observer class="SomeObserver" /> <observer sub="SomeOtherObserver::Functions::other_sub" />
In the first declaration we specify the class ('SomeObserver') that
will catch observations using its update()
method. In the second
we're naming exactly the subroutine ('other_sub()' in the class
'SomeOtherObserver::Functions') that will catch observations.
All configured observers get all events. It's up to each observer to figure out what it wants to handle.
The following documentation is for the workflow object itself rather than the entire system.
Execute the action $action_name
. Typically this changes the state
of the workflow. If $action_name
is not in the current state, fails
one of the conditions on the action, or fails one of the validators on
the action an exception is thrown. $autorun is used internally and
is set to 1 if the action was executed using autorun.
After the action has been successfully executed and the workflow saved we issue a 'execute' observation with the old state, action name and an autorun flag as additional parameters. So if you wanted to write an observer you could create a method with the signature:
sub update { my ( $class, $workflow, $action, $old_state, $action_name, $autorun ) = @_; if ( $action eq 'execute' ) { .... } }
We also issue a 'change state' observation if the executed action resulted in a new state. See WORKFLOWS ARE OBSERVABLE above for how we use and register observers and the Class::Observable manpage for more general information about observers as well as implementation details.
Returns: new state of workflow
get_current_actions()
Returns a list of action names available from the current state for
the given environment. So if you keep your context()
the same if
you call execute_action()
with one of the action names you should
not trigger any condition error since the action has already been
screened for conditions.
Returns: list of strings representing available actions
Return a list of the Workflow::Action::InputField manpage objects for the given
$action_name
. If $action_name
not in the current state or not
accessible by the environment an exception is thrown.
Returns: list of the Workflow::Action::InputField manpage objects
Adds any number of histories to the workflow, typically done by an
action in execute_action()
or one of the observers of that
action. This history will not be saved until execute_action()
is
complete.
You can add a list of either hashrefs with history information in them or full the Workflow::History manpage objects. Trying to add anything else will result in an exception and none of the items being added.
Successfully adding the history objects results in a 'add history' observation being thrown. See WORKFLOWS ARE OBSERVABLE above for more.
Returns: nothing
get_history()
Returns list of history objects for this workflow. Note that some may
be unsaved if you call this during the execute_action()
process.
get_unsaved_history()
Returns list of all unsaved history objects for this workflow.
clear_history()
Clears all transient history objects from the workflow object, not from the long-term storage.
Method used to overwrite the Class::Accessor manpage so only certain callers can set properties caller has to be a Workflow namespace package.
Sets property to value or throws the Workflow::Exception manpage
Unless otherwise noted properties are read-only.
id
ID of this workflow. This will always be defined, since when the the Workflow::Factory manpage creates a new workflow it first saves it to long-term storage.
type
Type of workflow this is. You may have many individual workflows associated with a type.
description
Description (usually brief, hopefully with a URL...) of this workflow.
state
The current state of the workflow.
last_update (read-write)
Date of the workflow's last update.
A the Workflow::Context manpage object associated with this workflow. This should never be undefined as the the Workflow::Factory manpage sets an empty context into the workflow when it is instantiated.
If you add a context to a workflow and one already exists, the values from the new workflow will overwrite values in the existing workflow. This is a shallow merge, so with the following:
$wf->context->param( drinks => [ 'coke', 'pepsi' ] ); my $context = Workflow::Context->new(); $context->param( drinks => [ 'beer', 'wine' ] ); $wf->context( $context ); print 'Current drinks: ', join( ', ', @{ $wf->context->param( 'drinks' ) } );
You will see:
Current drinks: beer, wine
THIS SHOULD ONLY BE CALLED BY THE the Workflow::Factory manpage. Do not call
this or the new()
method yourself -- you will only get an
exception. Your only interface for creating and fetching workflows is
through the factory.
This is called by the inherited constructor and sets the
$current_state
value to the property state
and uses the other
non-state values from \%config
to set parameters via the inherited
param()
.
Retrieves the action object associated with $action_name
in the
current workflow state. This will throw an exception if:
$action_name
exists in the current state.
No action $action_name
exists in the workflow universe.
One of the conditions for the action in this state is not met.
Return the the Workflow::State manpage object corresponding to $state
, which
defaults to the current state.
Assign the the Workflow::State manpage object $wf_state
to the workflow.
Returns the name of the next state given the action
$action_name
. Throws an exception if $action_name
not contained
in the current state.
October 2004 talk 'Workflows in Perl' given to pgh.pm: http://www.cwinters.com/pdf/workflow_pgh_pm.pdf
Copyright (c) 2003 Chris Winters and Arvato Direct; Copyright (c) 2004-2007 Chris Winters. All rights reserved.
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
Jonas B. Nielsen (jonasbn) <jonasbn@cpan.org>, current maintainer.
Chris Winters <chris@cwinters.com>, original author.
The following folks have also helped out:
Jim Brandt, for patch to Workflow::Config::XML. See Changes file, 0.27
Alexander Klink, for: patches resulting in 0.23, 0.24, 0.25, 0.26 and 0.27
Michael Bell, for patch resulting in 0.22
Martin Bartosch, for bug reporting and giving the solution not even using a patch (0.19 to 0.20) and a patch resulting in 0.21
Randal Schwartz, for testing 0.18 and swiftly giving feedback (0.18 to 0.19)
Chris Brown, for a patch to the Workflow::Config::Perl manpage (0.17 to 0.18)
Dietmar Hanisch <Dietmar.Hanisch@Bertelsmann.de> - Provided most of the good ideas for the module and an excellent example of everyday use.
Tom Moertel <tmoertel@cpan.org> gave me the idea for being able to attach event listeners (observers) to the process.
Michael Roberts <michael@vivtek.com> graciously released the 'Workflow' namespace on CPAN; check out his Workflow toolkit at http://www.vivtek.com/wftk.html.
Michael Schwern <schwern@pobox.org> barked via RT about a dependency problem and CPAN naming issue.
Jim Smith <jgsmith@tamu.edu> - Contributed patches (being able to subclass the Workflow::Factory manpage) and good ideas.
Martin Winkler <mw@arsnavigandi.de> - Pointed out a bug and a few other items.
Workflow - Simple, flexible system to implement workflows |