com.arjuna.mwlabs.wsas.activity.ActivityImple Maven / Gradle / Ivy
The newest version!
/*
Copyright The Narayana Authors
SPDX-License-Identifier: Apache-2.0
*/
package com.arjuna.mwlabs.wsas.activity;
import com.arjuna.mw.wsas.activity.HLS;
import com.arjuna.mw.wsas.completionstatus.Success;
import com.arjuna.mw.wsas.logging.wsasLogger;
import com.arjuna.mw.wsas.activity.Outcome;
import com.arjuna.mw.wsas.common.GlobalId;
// TODO: obtain via configuration
import com.arjuna.mwlabs.wsas.common.arjunacore.GlobalIdImple;
import com.arjuna.mw.wsas.status.Created;
import com.arjuna.mw.wsas.status.Active;
import com.arjuna.mw.wsas.status.Completing;
import com.arjuna.mw.wsas.status.Completed;
import com.arjuna.mw.wsas.completionstatus.Failure;
import com.arjuna.mw.wsas.completionstatus.FailureOnly;
import com.arjuna.mw.wsas.completionstatus.CompletionStatus;
import com.arjuna.mw.wsas.status.Status;
import com.arjuna.mw.wsas.exceptions.WrongStateException;
import com.arjuna.mw.wsas.exceptions.SystemException;
import com.arjuna.mw.wsas.exceptions.InvalidActivityException;
import com.arjuna.mw.wsas.exceptions.InvalidTimeoutException;
import com.arjuna.mw.wsas.exceptions.ProtocolViolationException;
import com.arjuna.mw.wsas.exceptions.NoActivityException;
import com.arjuna.mw.wsas.exceptions.NoPermissionException;
import com.arjuna.mw.wsas.exceptions.HLSException;
import java.util.Hashtable;
import java.util.Stack;
import java.util.Enumeration;
import java.util.Iterator;
/**
* The Activity.
*
* @author Mark Little ([email protected])
* @version $Id: ActivityImple.java,v 1.6 2005/05/19 12:13:18 nmcl Exp $
* @since 1.0.
*/
public class ActivityImple
{
public ActivityImple (String coordinationType)
{
this(null, coordinationType);
}
public ActivityImple (ActivityImple parent, String serviceType)
{
_parent = parent;
_children = new Hashtable();
_status = Created.instance();
_completionStatus = Failure.instance();
_activityId = new GlobalIdImple();
_timeout = 0;
_result = null;
_serviceType = serviceType;
}
/**
* Start a new activity.
*
* @exception WrongStateException
* Thrown if the currently associated activity is in a state
* that does not allow a new activity to be enlisted as a
* child.
* @exception InvalidTimeoutException
* Thrown if the specified timeout is invalid within the
* current working environment.
* @exception SystemException
* Thrown in any other situation.
*/
public void start () throws WrongStateException, SystemException
{
try
{
start(0);
}
catch (InvalidTimeoutException ex)
{
}
}
/**
* Start a new activity. If there is already an activity associated with the
* thread then it will be nested.
*
* @param timeout
* timeout The timeout associated with the activity. If the
* activity has not been terminated by the time this period
* elapses, then it will automatically be terminated.
* @exception WrongStateException
* Thrown if the currently associated activity is in a state
* that does not allow a new activity to be enlisted as a
* child.
* @exception InvalidTimeoutException
* Thrown if the specified timeout is invalid within the
* current working environment.
* @exception SystemException
* Thrown in any other situation.
*
*/
public void start (int timeout) throws WrongStateException,
InvalidTimeoutException, SystemException
{
if (timeout < 0)
throw new InvalidTimeoutException();
synchronized (this)
{
_timeout = timeout;
if (_status.equals(Created.instance()))
{
try
{
if (_parent != null)
_parent.addChild(this);
}
catch (InvalidActivityException ex)
{
_status = Completed.instance();
throw new WrongStateException(ex.toString());
}
if (_timeout > 0)
{
if (!ActivityReaper.activityReaper(true).insert(this, _timeout))
{
setCompletionStatus(FailureOnly.instance());
}
}
_status = Active.instance();
}
else
throw new WrongStateException(
wsasLogger.i18NLogger.get_activity_ActivityImple_1()
+ " " + this + " " + _status);
}
}
/**
* Complete the activity with the current completion status.
*
* @exception InvalidActivityException
* Thrown if the current activity is not known about by the
* activity system.
* @exception WrongStateException
* Thrown if the current activity is not in a state that
* allows it to be completed.
* @exception ProtocolViolationException
* Thrown if the a violation of the activity service or HLS
* protocol occurs.
* @exception NoPermissionException
* Thrown if the invoking thread does not have permission to
* terminate the transaction.
* @exception SystemException
* Thrown if some other error occurred.
*
* @return the result of completing the activity. Null is valid and must be
* interpreted within the context of any HLS that may exist.
*
* @see com.arjuna.mw.wsas.Outcome
*/
public Outcome end () throws InvalidActivityException, WrongStateException,
ProtocolViolationException, NoPermissionException, SystemException
{
return end(_completionStatus);
}
/**
* Complete the activity with the completion status provided.
*
*
* @param cs The CompletionStatus to use.
*
* @exception InvalidActivityException
* Thrown if the current activity is not known about by the
* activity system.
* @exception WrongStateException
* Thrown if the current activity is not in a state that
* allows it to be completed, or is incompatible with the
* completion status provided.
* @exception ProtocolViolationException
* Thrown if the a violation of the activity service or HLS
* protocol occurs.
* @exception NoPermissionException
* Thrown if the invoking thread does not have permission to
* terminate the transaction.
* @exception SystemException
* Thrown if some other error occurred.
*
* @return the result of completing the activity. Null is valid and must be
* interpreted within the context of any HLS that may exist.
*
* @see com.arjuna.mw.wsas.Outcome
*
*/
public Outcome end (com.arjuna.mw.wsas.completionstatus.CompletionStatus cs)
throws InvalidActivityException, WrongStateException,
ProtocolViolationException, NoPermissionException, SystemException
{
/*
* TODO
*
* We need an exception that can be thrown to say that the activity is
* completing.
*/
synchronized (this)
{
if (_status.equals(Active.instance()))
{
if (activeChildren())
{
/*
* Can we do equivalent of rollback on all children and then
* rollback this?
*/
throw new InvalidActivityException(
wsasLogger.i18NLogger.get_activity_ActivityImple_2()
+ " " + this);
}
Outcome result = null;
try
{
setCompletionStatus(cs);
}
catch (Exception ex)
{
// ignore and complete with the status we have.
}
_status = Completing.instance();
try
{
HLS hls = HLSManager.getHighLevelService(_serviceType);
if (hls != null) {
result = hls.complete(getCompletionStatus());
} else {
result = new OutcomeImple(Failure.instance());
}
}
catch (SystemException ex)
{
/*
* Currently if an exception occurs and we get here, then we
* forget all of the other outcomes and just return the
* exception. Does this make sense? How will applications be
* able to tell which HLSes have processed the outcome and
* which have not?
*/
result = new OutcomeImple(new HLSException(ex),
Failure.instance());
}
if (_parent != null)
{
_parent.removeChild(this);
_parent = null;
}
_status = Completed.instance();
_result = result;
return _result;
}
else
{
if (_result != null)
return _result;
else
{
// we can't have terminated yet!
throw new WrongStateException(
wsasLogger.i18NLogger.get_activity_ActivityImple_3()
+ " " + _status);
}
}
}
}
/**
* Set the termination status for the current activity, if any.
*
* @param CompletionStatus
* endStatus The state in which the activity should attempt to
* terminate. This may be one of the default values provided by
* WSAS or may be extended in an implementation specific manner
* by an HLS.
*
* @exception NoActivityException
* Thrown if there is no activity associated with the
* invoking thread.
* @exception WrongStateException
* Thrown if the completion status is incompatible with the
* current state of the activity.
* @exception SystemException
* Thrown if any other error occurs.
*
*/
public void setCompletionStatus (CompletionStatus endStatus)
throws WrongStateException, SystemException
{
synchronized (this)
{
if (_status.equals(Active.instance()))
{
completionValid(endStatus);
_completionStatus = endStatus;
}
else
throw new WrongStateException(
wsasLogger.i18NLogger.get_activity_ActivityImple_4()
+ " " + this + " " + _status);
}
}
/**
* Get the completion status currently associated with the activity.
*
* @exception NoActivityException
* Thrown if there is no activity associated with the current
* thread.
* @exception SystemException
* Thrown if any other error occurs.
*
* @return the termination status for the current activity, if any.
*/
public CompletionStatus getCompletionStatus () throws SystemException
{
synchronized (this)
{
return _completionStatus;
}
}
/**
* Get the timeout value currently associated with activities.
*
* @exception SystemException
* Thrown if any error occurs.
*
* @return the timeout value in seconds, or 0 if no application specified
* timeout has been provided.
*/
public int getTimeout () throws SystemException
{
return _timeout;
}
/**
* @exception SystemException
* Thrown if any error occurs.
*
* @return the status of the current activity. If there is no activity
* associated with the thread then NoActivity will be returned.
*
* @see com.arjuna.mw.wsas.status.Status
*/
public com.arjuna.mw.wsas.status.Status status () throws SystemException
{
synchronized (this)
{
return _status;
}
}
/**
* What is the name of the current activity? Use only for debugging
* purposes!
*
* @exception NoActivityException
* Thrown if there is no activity associated with the
* invoking thread.
* @exception SystemException
* Thrown if any other error occurs.
*
* @return the name of the activity.
*/
public String activityName () throws NoActivityException, SystemException
{
return "ActivityImple: " + toString();
}
public String toString ()
{
return _activityId.stringForm();
}
/**
* @return the unique identifier for this activity.
*/
public GlobalId getGlobalId ()
{
return _activityId;
}
/**
* @return The parent of the activity, or null if it is top-level.
*/
public ActivityImple parent ()
{
return _parent;
}
/**
*
* @return the type of the coordinator employed for this activity
*/
public String serviceType()
{
return _serviceType;
}
/**
*/
public boolean equals (Object obj)
{
if (obj == null)
return false;
if (obj == this)
return true;
if (obj instanceof ActivityImple) {
if (((ActivityImple) obj).getGlobalId().equals(getGlobalId()))
return true;
}
return false;
}
/**
* Return the activity hierarchy that this activity is within. The zeroth
* element is the parent.
*/
public ActivityImple[] hierarchy ()
{
Stack hier = new Stack();
ActivityImple ptr = this;
while (ptr != null)
{
hier.push(ptr);
ptr = ptr.parent();
}
int hierSize = hier.size();
ActivityImple[] toReturn = new ActivityImple[hierSize];
for (int i = 0; i < hierSize; i++)
toReturn[i] = (ActivityImple) hier.pop();
return toReturn;
}
/**
* Check whether the specified completion status is compatible with the one
* currently assigned to the activity.
*
* @param CompletionStatus
* cs The completion status to check.
*
* @exception WrongStateException
* Thrown if the specified status is incompatible with that
* currently possessed by this activity.
*
*/
public final void completionValid (CompletionStatus cs)
throws WrongStateException
{
if (!_completionStatus.equals(cs))
{
if (_completionStatus.equals(FailureOnly.instance()))
throw new WrongStateException(
wsasLogger.i18NLogger.get_activity_ActivityImple_5()
+ " " + _completionStatus + " " + cs);
}
}
public int hashCode ()
{
return _activityId.hashCode();
}
/**
* Add the specified activity as a child of this activity.
*
* @param ActivityImple
* child The child activity.
*
* @exception WrongStateException
* Thrown if the parent activity is not in a state that
* allows children to be added.
* @exception InvalidActivityException
* Thrown if the child activity is invalid.
* @exception SystemException
* Thrown if some other error occurs.
*
*/
final void addChild (ActivityImple child) throws WrongStateException,
InvalidActivityException, SystemException
{
if (child == null)
throw new InvalidActivityException(
wsasLogger.i18NLogger.get_activity_ActivityImple_6());
synchronized (this)
{
if (_status.equals(Active.instance()))
{
_children.put(child.getGlobalId(), child);
}
else
throw new WrongStateException(
wsasLogger.i18NLogger.get_activity_ActivityImple_7()
+ " " + _status);
}
}
/**
* Remove the specified child activity from this activity.
*
* @param ActivityImple
* child The child activity to remove.
*
* @exception WrongStateException
* Thrown if the parent activity is not in a state that
* allows children to be removed.
* @exception InvalidActivityException
* Thrown if the child activity is invalid.
* @exception SystemException
* Thrown if some other error occurs.
*
*/
final void removeChild (ActivityImple child) throws WrongStateException,
InvalidActivityException, SystemException
{
if (child == null)
throw new InvalidActivityException(
wsasLogger.i18NLogger.get_activity_ActivityImple_8());
synchronized (this)
{
if (_status.equals(Active.instance()))
{
if (_children.get(child.getGlobalId()) == null)
throw new InvalidActivityException(
wsasLogger.i18NLogger.get_activity_ActivityImple_9()
+ child);
}
else
throw new WrongStateException(
wsasLogger.i18NLogger.get_activity_ActivityImple_10()
+ _status);
}
}
/**
* @return true
if this activity has active children,
* false
* otherwise.
*/
private final boolean activeChildren ()
{
Enumeration e = _children.keys();
while (e.hasMoreElements())
{
ActivityImple child = (ActivityImple) _children.get(e.nextElement());
try
{
if ((child != null)
&& (child.status().equals(Active.instance())))
{
return true;
}
}
catch (Exception ex)
{
return true; // what else can we do?
}
}
return false;
}
private ActivityImple _parent;
private Hashtable _children;
private Status _status;
private CompletionStatus _completionStatus;
private GlobalIdImple _activityId;
private int _timeout;
private Outcome _result;
private String _serviceType;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy