📄 workflow.java
字号:
/* JSPWiki - a JSP-based WikiWiki clone. Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to you under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */package com.ecyrd.jspwiki.workflow;import java.io.Serializable;import java.security.Principal;import java.util.*;import com.ecyrd.jspwiki.WikiException;import com.ecyrd.jspwiki.event.WikiEventListener;import com.ecyrd.jspwiki.event.WikiEventManager;import com.ecyrd.jspwiki.event.WorkflowEvent;/** * <p> * Sequence of {@link Step} objects linked together. Workflows are always * initialized with a message key that denotes the name of the Workflow, and a * Principal that represents its owner. * </p> * <h2>Workflow lifecycle</h2> * A Workflow's state (obtained by {@link #getCurrentState()}) will be one of the * following: * </p> * <ul> * <li><strong>{@link #CREATED}</strong>: after the Workflow has been * instantiated, but before it has been started using the {@link #start()} * method.</li> * <li><strong>{@link #RUNNING}</strong>: after the Workflow has been started * using the {@link #start()} method, but before it has finished processing all * Steps. Note that a Workflow can only be started once; attempting to start it * again results in an IllegalStateException. Callers can place the Workflow * into the WAITING state by calling {@link #waitstate()}.</li> * <li><strong>{@link #WAITING}</strong>: when the Workflow has temporarily * paused, for example because of a pending Decision. Once the responsible actor * decides what to do, the caller can change the Workflow back to the RUNNING * state by calling the {@link #restart()} method (this is done automatically by * the Decision class, for instance, when the {@link Decision#decide(Outcome)} * method is invoked)</li> * <li><strong>{@link #COMPLETED}</strong>: after the Workflow has finished * processing all Steps, without errors.</li> * <li><strong>{@link #ABORTED}</strong>: if a Step has elected to abort the * Workflow.</li> * </ul> * <h2>Steps and processing algorithm</h2> * <p> * Workflow Step objects can be of type {@link Decision}, {@link Task} or other * Step subclasses. Decisions require user input, while Tasks do not. See the * {@link Step} class for more details. * </p> * <p> * After instantiating a new Workflow (but before telling it to {@link #start()}), * calling classes should specify the first Step by executing the * {@link #setFirstStep(Step)} method. Additional Steps can be chained by * invoking the first step's {@link Step#addSuccessor(Outcome, Step)} method. * </p> * <p> * When a Workflow's <code>start</code> method is invoked, the Workflow * retrieves the first Step and processes it. This Step, and subsequent ones, * are processed as follows: * </p> * <ul> * <li>The Step's {@link Step#start()} method executes, which sets the start * time.</li> * <li>The Step's {@link Step#execute()} method is called to begin processing, * which will return an Outcome to indicate completion, continuation or errors:</li> * <ul> * <li>{@link Outcome#STEP_COMPLETE} indicates that the execution method ran * without errors, and that the Step should be considered "completed."</li> * <li>{@link Outcome#STEP_CONTINUE} indicates that the execution method ran * without errors, but that the Step is not "complete" and should be put into * the WAITING state.</li> * <li>{@link Outcome#STEP_ABORT} indicates that the execution method * encountered errors, and should abort the Step <em>and</em> the Workflow as * a whole. When this happens, the Workflow will set the current Step's Outcome * to {@link Outcome#STEP_ABORT} and invoke the Workflow's {@link #abort()} * method. The Step's processing errors, if any, can be retrieved by * {@link Step#getErrors()}.</li> * </ul> * <li>The Outcome of the <code>execute</code> method also affects what * happens next. Depending on the result (and assuming the Step did not abort), * the Workflow will either move on to the next Step or put the Workflow into * the {@link Workflow#WAITING} state:</li> * <ul> * <li>If the Outcome denoted "completion" (<em>i.e.</em>, its * {@link Step#isCompleted()} method returns <code>true</code>) then the Step * is considered complete; the Workflow looks up the next Step by calling the * current Step's {@link Step#getSuccessor(Outcome)} method. If * <code>successor()</code> returns a non-<code>null</code> Step, the * return value is marked as the current Step and added to the Workflow's Step * history. If <code>successor()</code> returns <code>null</code>, then the * Workflow has no more Steps and it enters the {@link #COMPLETED} state.</li> * <li>If the Outcome did not denote "completion" (<em>i.e.</em>, its * {@link Step#isCompleted()} method returns <code>false</code>), then the * Step still has further work to do. The Workflow enters the {@link #WAITING} * state and stops further processing until a caller restarts it.</li> * </ul> * </ul> * </p> * <p> * The currently executing Step can be obtained by {@link #getCurrentStep()}. The * actor for the current Step is returned by {@link #getCurrentActor()}. * </p> * <p> * To provide flexibility for specific implementations, the Workflow class * provides two additional features that enable Workflow participants (<em>i.e.</em>, * Workflow subclasses and Step/Task/Decision subclasses) to share context and * state information. These two features are <em>named attributes</em> and * <em>message arguments</em>: * </p> * <ul> * <li><strong>Named attributes</strong> are simple key-value pairs that * Workflow participants can get or set. Keys are Strings; values can be any * Object. Named attributes are set with {@link #setAttribute(String, Object)} * and retrieved with {@link #getAttribute(String)}.</li> * <li><strong>Message arguments</strong> are used in combination with * JSPWiki's {@link com.ecyrd.jspwiki.i18n.InternationalizationManager} to * create language-independent user interface messages. The message argument * array is retrieved via {@link #getMessageArguments()}; the first two array * elements will always be these: a String representing work flow owner's name, * and a String representing the current actor's name. Workflow participants * can add to this array by invoking {@link #addMessageArgument(Serializable)}.</li> * </ul> * <h2>Example</h2> * <p> * Workflow Steps can be very powerful when linked together. JSPWiki provides * two abstract subclasses classes that you can use to build your own Workflows: * Tasks and Decisions. As noted, Tasks are Steps that execute without user * intervention, while Decisions require actors (<em>aka</em> Principals) to * take action. Decisions and Tasks can be mixed freely to produce some highly * elaborate branching structures. * </p> * <p> * Here is a simple case. For example, suppose you would like to create a * Workflow that (a) executes a initialization Task, (b) pauses to obtain an * approval Decision from a user in the Admin group, and if approved, (c) * executes a "finish" Task. Here's sample code that illustrates how to do it: * </p> * * <pre> * // Create workflow; owner is current user * 1 Workflow workflow = new Workflow("workflow.myworkflow", context.getCurrentUser()); * * // Create custom initialization task * 2 Step initTask = new InitTask(this); * * // Create finish task * 3 Step finishTask = new FinishTask(this); * * // Create an intermediate decision step * 4 Principal actor = new GroupPrincipal("Admin"); * 5 Step decision = new SimpleDecision(this, "decision.AdminDecision", actor); * * // Hook the steps together * 6 initTask.addSuccessor(Outcome.STEP_COMPLETE, decision); * 7 decision.addSuccessor(Outcome.DECISION_APPROVE, finishTask); * * // Set workflow's first step * 8 workflow.setFirstStep(initTask); * </pre> * * <p> * Some comments on the source code: * </p> * <ul> * <li>Line 1 instantiates the workflow with a sample message key and * designated owner Principal, in this case the current wiki user</li> * <li>Lines 2 and 3 instantiate the custom Task subclasses, which contain the * business logic</li> * <li>Line 4 creates the relevant GroupPrincipal for the <code>Admin</code> * group, who will be the actor in the Decision step</li> * <li>Line 5 creates the Decision step, passing the Workflow, sample message * key, and actor in the constructor</li> * <li>Line 6 specifies that if the InitTask's Outcome signifies "normal * completion" (STEP_COMPLETE), the SimpleDecision step should be invoked next</li> * <li>Line 7 specifies that if the actor (anyone possessing the * <code>Admin</code> GroupPrincipal) selects DECISION_APPROVE, the FinishTask * step should be invoked</li> * <li>Line 8 adds the InitTask (and all of its successor Steps, nicely wired * together) to the workflow</li> * </ul> * * @author Andrew Jaquith */public class Workflow implements Serializable{ private static final long serialVersionUID = 5228149040690660032L; /** Time value: the start or end time has not been set. */ public static final Date TIME_NOT_SET = new Date( 0 ); /** ID value: the workflow ID has not been set. */ public static final int ID_NOT_SET = 0; /** State value: Workflow completed all Steps without errors. */ public static final int COMPLETED = 50; /** State value: Workflow aborted before completion. */ public static final int ABORTED = 40; /** * State value: Workflow paused, typically because a Step returned an * Outcome that doesn't signify "completion." */ public static final int WAITING = 30; /** State value: Workflow started, and is running. */ public static final int RUNNING = -1; /** State value: Workflow instantiated, but not started. */ public static final int CREATED = -2; /** Lazily-initialized attribute map. */ private Map<String, Object> m_attributes; /** The initial Step for this Workflow. */ private Step m_firstStep; /** Flag indicating whether the Workflow has started yet. */ private boolean m_started; private final LinkedList<Step> m_history; private int m_id; private final String m_key; private final Principal m_owner; private final List<Serializable> m_messageArgs; private int m_state; private Step m_currentStep; private WorkflowManager m_manager; /** * Constructs a new Workflow object with a supplied message key, owner * Principal, and undefined unique identifier {@link #ID_NOT_SET}. Once * instantiated the Workflow is considered to be in the {@link #CREATED} * state; a caller must explicitly invoke the {@link #start()} method to * begin processing. * * @param messageKey * the message key used to construct a localized workflow name, * such as <code>workflow.saveWikiPage</code> * @param owner * the Principal who owns the Workflow. Typically, this is the * user who created and submitted it */ public Workflow(String messageKey, Principal owner) { super(); m_attributes = null; m_currentStep = null; m_history = new LinkedList<Step>(); m_id = ID_NOT_SET; m_key = messageKey; m_manager = null; m_messageArgs = new ArrayList<Serializable>(); m_owner = owner; m_started = false; m_state = CREATED; } /** * Aborts the Workflow by setting the current Step's Outcome to * {@link Outcome#STEP_ABORT}, and the Workflow's overall state to * {@link #ABORTED}. It also appends the aborted Step into the workflow * history, and sets the current step to <code>null</code>. If the Step * is a Decision, it is removed from the DecisionQueue. This method * can be called at any point in the lifecycle prior to completion, but it * cannot be called twice. It finishes by calling the {@link #cleanup()} * method to flush retained objects. If the Workflow had been previously * aborted, this method throws an IllegalStateException. */ public final synchronized void abort() { // Check corner cases: previous abort or completion if ( m_state == ABORTED ) { throw new IllegalStateException( "The workflow has already been aborted." ); } if ( m_state == COMPLETED ) { throw new IllegalStateException( "The workflow has already completed." ); } if ( m_currentStep != null ) { if ( m_manager != null && m_currentStep instanceof Decision ) { Decision d = (Decision)m_currentStep; m_manager.getDecisionQueue().remove( d ); } m_currentStep.setOutcome( Outcome.STEP_ABORT ); m_history.addLast( m_currentStep ); } m_state = ABORTED; fireEvent( WorkflowEvent.ABORTED ); cleanup(); } /** * Appends a message argument object to the array returned by * {@link #getMessageArguments()}. The object <em>must</em> be an type * used by the {@link java.text.MessageFormat}: String, Date, or Number * (BigDecimal, BigInteger, Byte, Double, Float, Integer, Long, Short). * If the object is not of type String, Number or Date, this method throws * an IllegalArgumentException. * @param obj the object to add */ public final void addMessageArgument( Serializable obj ) { if ( obj instanceof String || obj instanceof Date || obj instanceof Number ) { m_messageArgs.add( obj ); return; } throw new IllegalArgumentException( "Message arguments must be of type String, Date or Number." ); } /** * Returns the actor Principal responsible for the current Step. If there is * no current Step, this method returns <code>null</code>. * * @return the current actor */ public final synchronized Principal getCurrentActor() { if ( m_currentStep == null ) { return null; } return m_currentStep.getActor(); } /** * Returns the workflow state: {@link #CREATED}, {@link #RUNNING}, * {@link #WAITING}, {@link #COMPLETED} or {@link #ABORTED}. * * @return the workflow state */ public final int getCurrentState() { return m_state; } /** * Returns the current Step, or <code>null</code> if the workflow has not * started or already completed. * * @return the current step */ public final Step getCurrentStep() { return m_currentStep; } /** * Retrieves a named Object associated with this Workflow. If the Workflow * has completed or aborted, this method always returns <code>null</code>. * * @param attr * the name of the attribute * @return the value */ public final synchronized Object getAttribute( String attr ) { if ( m_attributes == null ) { return null; } return m_attributes.get( attr ); } /** * The end time for this Workflow, expressed as a system time number. This * value is equal to the end-time value returned by the final Step's * {@link Step#getEndTime()} method, if the workflow has completed. * Otherwise, this method returns {@link #TIME_NOT_SET}. * * @return the end time */ public final Date getEndTime() { if ( isCompleted() ) { Step last = m_history.getLast(); if ( last != null ) { return last.getEndTime(); } } return TIME_NOT_SET; } /** * Returns the unique identifier for this Workflow. If not set, this method * returns ID_NOT_SET ({@value #ID_NOT_SET}). * * @return the unique identifier */ public final synchronized int getId() { return m_id; }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -