All Downloads are FREE. Search and download functionalities are using the official Maven repository.

bitronix.tm.twopc.AbstractPhaseEngine Maven / Gradle / Ivy

/*
 * Copyright (C) 2006-2013 Bitronix Software (http://www.bitronix.be)
 *
 * Licensed 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 bitronix.tm.twopc;

import bitronix.tm.TransactionManagerServices;
import bitronix.tm.internal.LogDebugCheck;
import bitronix.tm.internal.XAResourceHolderState;
import bitronix.tm.internal.XAResourceManager;
import bitronix.tm.twopc.executor.Executor;
import bitronix.tm.twopc.executor.Job;
import bitronix.tm.utils.CollectionUtils;
import bitronix.tm.utils.Decoder;

import javax.transaction.xa.XAException;
import java.util.*;
import java.util.logging.Level;

/**
 * Abstract phase execution engine.
 *
 * @author Ludovic Orban
 */
public abstract class AbstractPhaseEngine
{

	private static final java.util.logging.Logger log = java.util.logging.Logger.getLogger(AbstractPhaseEngine.class.toString());

	private final Executor executor;

	/**
	 * Constructor AbstractPhaseEngine creates a new AbstractPhaseEngine instance.
	 *
	 * @param executor
	 * 		of type Executor
	 */
	protected AbstractPhaseEngine(Executor executor)
	{
		this.executor = executor;
	}

	/**
	 * Method collectResourcesUniqueNames ...
	 *
	 * @param resources
	 * 		of type List XAResourceHolderState
	 *
	 * @return Set String
	 */
	protected static Set collectResourcesUniqueNames(List resources)
	{
		Set uniqueNames = new HashSet<>();

		for (XAResourceHolderState resourceHolderState : resources)
		{
			uniqueNames.add(resourceHolderState.getUniqueName());
		}

		return uniqueNames;
	}

	/**
	 * Method collectNotInterestedResources ...
	 *
	 * @param allResources
	 * 		of type List XAResourceHolderState
	 * @param interestedResources
	 * 		of type List XAResourceHolderState
	 *
	 * @return List XAResourceHolderState
	 */
	protected static List collectNotInterestedResources(List allResources, List interestedResources)
	{
		List result = new ArrayList<>();

		for (XAResourceHolderState resourceHolderState : allResources)
		{
			if (!CollectionUtils.containsByIdentity(interestedResources, resourceHolderState))
			{
				result.add(resourceHolderState);
			}
		}

		return result;
	}

	/**
	 * Execute the phase. Resources receive the phase command in position order (reversed or not). If there is more than
	 * once resource in a position, command is sent in enlistment order (again reversed or not).
	 * If {@link bitronix.tm.Configuration#isAsynchronous2Pc()} is true, all commands in a given position are sent
	 * in parallel by using the detected {@link Executor} implementation.
	 *
	 * @param resourceManager
	 * 		the {@link XAResourceManager} containing the enlisted resources to execute the phase on.
	 * @param reverse
	 * 		true if jobs should be executed in reverse position / enlistment order, false for natural position / enlistment order.
	 *
	 * @throws PhaseException
	 * 		if one or more resource threw an exception during phase execution.
	 * @see bitronix.tm.twopc.executor.SyncExecutor
	 * @see bitronix.tm.twopc.executor.AsyncExecutor
	 */
	protected void executePhase(XAResourceManager resourceManager, boolean reverse) throws PhaseException
	{
		SortedSet positions;
		if (reverse)
		{
			positions = resourceManager.getReverseOrderPositions();
			if (LogDebugCheck.isDebugEnabled())
			{
				log.finer("executing phase on " + resourceManager.size() + " resource(s) enlisted in " + positions.size() + " position(s) in reverse position order");
			}
		}
		else
		{
			positions = resourceManager.getNaturalOrderPositions();
			if (LogDebugCheck.isDebugEnabled())
			{
				log.finer("executing phase on " + resourceManager.size() + " resource(s) enlisted in " + positions.size() + " position(s) in natural position order");
			}
		}

		List positionErrorReports = new ArrayList<>();

		for (Integer positionKey : positions)
		{
			List resources;
			if (reverse)
			{
				resources = resourceManager.getReverseOrderResourcesForPosition(positionKey);
			}
			else
			{
				resources = resourceManager.getNaturalOrderResourcesForPosition(positionKey);
			}

			if (LogDebugCheck.isDebugEnabled())
			{
				log.finer("running " + resources.size() + " job(s) for position '" + positionKey + "'");
			}
			JobsExecutionReport report = runJobsForPosition(resources);
			if (!report.getExceptions()
			           .isEmpty())
			{
				if (LogDebugCheck.isDebugEnabled())
				{
					log.finer(report.getExceptions()
					                .size() + " error(s) happened during execution of position '" + positionKey + "'");
				}
				positionErrorReports.add(report);
				break;
			}
			if (LogDebugCheck.isDebugEnabled())
			{
				log.finer("ran " + resources.size() + " job(s) for position '" + positionKey + "'");
			}
		}

		if (!positionErrorReports.isEmpty())
		{
			// merge all resources and exceptions lists
			List exceptions = new ArrayList<>();
			List resources = new ArrayList<>();

			for (JobsExecutionReport report : positionErrorReports)
			{
				exceptions.addAll(report.getExceptions());
				resources.addAll(report.getResources());
			}

			throw new PhaseException(exceptions, resources);
		}
	}

	/**
	 * Method runJobsForPosition ...
	 *
	 * @param resources
	 * 		of type List XAResourceHolderState
	 *
	 * @return JobsExecutionReport
	 */
	private JobsExecutionReport runJobsForPosition(List resources)
	{
		List jobs = new ArrayList<>();
		List exceptions = new ArrayList<>();
		List errorResources = new ArrayList<>();

		// start threads
		for (XAResourceHolderState resource : resources)
		{
			if (!isParticipating(resource))
			{
				if (LogDebugCheck.isDebugEnabled())
				{
					log.finer("skipping not participating resource " + resource);
				}
				continue;
			}

			Job job = createJob(resource);
			Object future = executor.submit(job);
			job.setFuture(future);
			jobs.add(job);
		}

		// wait for threads to finish and check results
		for (Job job : jobs)
		{
			Object future = job.getFuture();
			while (!executor.isDone(future))
			{
				executor.waitFor(future, 1000L);
			}

			XAException xaException = job.getXAException();
			RuntimeException runtimeException = job.getRuntimeException();

			if (xaException != null)
			{
				String extraErrorDetails = TransactionManagerServices.getExceptionAnalyzer()
				                                                     .extractExtraXAExceptionDetails(xaException);
				if (LogDebugCheck.isDebugEnabled())
				{
					log.finer("error executing " + job + ", errorCode=" + Decoder.decodeXAExceptionErrorCode(xaException) +
					          (extraErrorDetails == null ? "" : ", extra error=" + extraErrorDetails));
				}
				exceptions.add(xaException);
				errorResources.add(job.getResource());
			}
			else if (runtimeException != null)
			{
				if (LogDebugCheck.isDebugEnabled())
				{
					log.finer("error executing " + job);
				}
				exceptions.add(runtimeException);
				errorResources.add(job.getResource());
			}
		}

		if (LogDebugCheck.isDebugEnabled())
		{
			log.finer("phase executed with " + exceptions.size() + " exception(s)");
		}
		return new JobsExecutionReport(exceptions, errorResources);
	}

	/**
	 * Determine if a resource is participating in the phase or not. A participating resource gets
	 * a job created to execute the phase's command on it.
	 *
	 * @param xaResourceHolderState
	 * 		the resource to check for its participation.
	 *
	 * @return true if the resource must participate in the phase.
	 */
	protected abstract boolean isParticipating(XAResourceHolderState xaResourceHolderState);

	/**
	 * Create a {@link Job} that is going to execute the phase command on the given resource.
	 *
	 * @param xaResourceHolderState
	 * 		the resource that is going to receive a command.
	 *
	 * @return the {@link Job} that is going to execute the command.
	 */
	protected abstract Job createJob(XAResourceHolderState xaResourceHolderState);

	/**
	 * Log exceptions that happened during a phase failure.
	 *
	 * @param ex
	 * 		the phase exception.
	 */
	protected void logFailedResources(PhaseException ex)
	{
		List exceptions = ex.getExceptions();
		List resources = ex.getResourceStates();

		for (int i = 0; i < exceptions.size(); i++)
		{
			Exception e = exceptions.get(i);
			XAResourceHolderState holderState = resources.get(i);
			log.log(Level.SEVERE, "resource " + holderState.getUniqueName() + " failed on " + holderState.getXid(), e);
		}
	}

	private static final class JobsExecutionReport
	{
		private final List exceptions;
		private final List resources;

		/**
		 * Constructor JobsExecutionReport creates a new JobsExecutionReport instance.
		 *
		 * @param exceptions
		 * 		of type List Exception
		 * @param resources
		 * 		of type List XAResourceHolderState
		 */
		private JobsExecutionReport(List exceptions, List resources)
		{
			this.exceptions = Collections.unmodifiableList(exceptions);
			this.resources = Collections.unmodifiableList(resources);
		}

		/**
		 * Method getExceptions returns the exceptions of this JobsExecutionReport object.
		 *
		 * @return the exceptions (type List Exception ) of this JobsExecutionReport object.
		 */
		public List getExceptions()
		{
			return exceptions;
		}

		/**
		 * Method getResources returns the resources of this JobsExecutionReport object.
		 *
		 * @return the resources (type List XAResourceHolderState ) of this JobsExecutionReport object.
		 */
		public List getResources()
		{
			return resources;
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy